OpenAPI & Swagger
Okapi provides automatic OpenAPI documentation generation with built-in interactive UIs. The documentation is dynamically generated from your route definitions, ensuring it always stays in sync with your API implementation.
Okapi serves both OpenAPI 3.1 (the default) and OpenAPI 3.0, and ships with three interactive UIs out of the box — Swagger UI (default), ReDoc, and Scalar — with the UI rendered at /docs fully selectable.
Quick Start
Using okapi.Default()
Documentation is enabled by default and served at /docs:
o := okapi.Default() // Docs available at /docs
Using okapi.New() with WithOpenAPIDocs()
If you initialize Okapi with okapi.New(), documentation is disabled by default. Enable it with WithOpenAPIDocs():
o := okapi.New()
if os.Getenv("ENABLE_DOCS") == "true" {
o.WithOpenAPIDocs()
}
Custom Configuration
Customize the OpenAPI documentation:
o := okapi.New().WithOpenAPIDocs(
okapi.OpenAPI{
Title: "Example API",
Version: "1.0.0",
Contact: okapi.Contact{
Name: "API Support",
Email: "support@example.com",
},
},
)
Security Schemes
Define authentication mechanisms for your API:
o.WithOpenAPIDocs(okapi.OpenAPI{
Title: "Okapi Web Framework Example",
Version: "1.0.0",
License: okapi.License{Name: "MIT"},
SecuritySchemes: okapi.SecuritySchemes{
{
Name: "basicAuth",
Type: "http",
Scheme: "basic",
},
{
Name: "bearerAuth",
Type: "http",
Scheme: "bearer",
BearerFormat: "JWT",
},
{
Name: "OAuth2",
Type: "oauth2",
Flows: &okapi.OAuthFlows{
AuthorizationCode: &okapi.OAuthFlow{
AuthorizationURL: "https://auth.example.com/authorize",
TokenURL: "https://auth.example.com/token",
Scopes: map[string]string{
"read": "Read access",
"write": "Write access",
},
},
},
},
},
})
Applying Security to Routes
Single Route
var bearerAuthSecurity = []map[string][]string{
{"bearerAuth": {}},
}
o.Get("/books", getBooksHandler).WithSecurity(bearerAuthSecurity...)
Route Group
api := o.Group("/api", jwtMiddleware).WithSecurity(bearerAuthSecurity)
api.Get("/", apiHandler)
Documenting Routes
Okapi offers multiple ways to document your routes.
1. Composable Functions (Direct Style)
Simple and readable approach for small to medium routes:
o.Get("/books", getBooksHandler,
okapi.DocSummary("List all available books"),
okapi.DocTags("Books"),
okapi.DocQueryParam("author", "string", "Filter by author name", false),
okapi.DocQueryParam("limit", "int", "Maximum results to return", false),
okapi.DocResponseHeader("X-Client-Id", "string", "Client ID"),
okapi.DocResponse([]Book{}),
okapi.DocResponse(400, ErrorResponse{}),
okapi.DocResponse(401, ErrorResponse{}),
)
2. Fluent Builder Style
For complex or dynamic documentation:
o.Post("/books", createBookHandler,
okapi.Doc().
Summary("Add a new book to the inventory").
Tags("Books").
BearerAuth().
ResponseHeader("X-Client-Id", "string", "Client ID").
RequestBody(BookRequest{}).
Response(201, Book{}).
Response(400, ErrorResponse{}).
Response(401, ErrorResponse{}).
Build(),
)
3. Body Field Style
Using structs with dedicated body fields:
type BookRequest struct {
Body struct {
Name string `json:"name" minLength:"4" maxLength:"50" required:"true"`
Price int `json:"price" required:"true"`
} `json:"body"`
ID int `param:"id" query:"id"`
APIKey string `header:"X-API-Key" required:"true"`
}
o.Post("/books", createBookHandler,
okapi.Request(&BookRequest{}),
okapi.Response(&BookResponse{}),
)
Using .WithIO(), .WithInput(), .WithOutput()
// Both request & response
o.Post("/books", handler).WithIO(&BookRequest{}, &BookResponse{})
// Request only
o.Post("/books", handler).WithInput(&BookRequest{})
// Response only
o.Get("/books", handler).WithOutput(&BooksResponse{})
Available Documentation Options
| Method | Description |
|---|---|
DocSummary() / Doc().Summary() | Short endpoint summary |
DocTags() / Doc().Tags() | Group endpoints under tags |
DocBearerAuth() / Doc().BearerAuth() | Enable Bearer token authentication |
DocRequestBody() / Doc().RequestBody() | Document request body schema |
DocResponse() / Doc().Response() | Document response schema or status codes |
DocPathParam() / Doc().PathParam() | Document path parameters |
DocQueryParam() / Doc().QueryParam() | Document query parameters |
DocHeader() / Doc().Header() | Document request headers |
DocResponseHeader() / Doc().ResponseHeader() | Document response headers |
DocDeprecated() / Doc().Deprecated() | Mark route as deprecated |
Choosing the Documentation UI
Okapi ships with three interactive UIs: Swagger UI (default), ReDoc, and Scalar. The UI rendered at /docs is selectable, while each UI also stays reachable at its own dedicated route (/swagger, /redoc, /scalar) regardless of the selection.
Select it with the UI field on okapi.OpenAPI:
o.WithOpenAPIDocs(okapi.OpenAPI{
Title: "My API",
UI: okapi.ScalarUI, // okapi.SwaggerUI (default) | okapi.RedocUI | okapi.ScalarUI
})
…or with the chainable WithDocUI method:
o := okapi.New().WithOpenAPIDocs().WithDocUI(okapi.ScalarUI)
Restricting to a single UI
By default every UI stays reachable at its own route regardless of which one /docs renders. Set StrictDocUI: true to register only the selected UI — the other UI routes then return 404:
o.WithOpenAPIDocs(okapi.OpenAPI{
Title: "My API",
UI: okapi.ScalarUI,
StrictDocUI: true, // only /docs and /scalar are served; /swagger and /redoc return 404
})
OpenAPI 3.1 and 3.0
Okapi serves the same API description as both OpenAPI 3.1 / JSON Schema 2020-12 and OpenAPI 3.0. The default endpoints (/openapi.json, /openapi.yaml) serve 3.1, and the documentation UIs render it. The 3.0 document remains available at /openapi-3.0.{json,yaml}, so 3.0-only consumers stay supported.
The 3.1 document is derived from the 3.0 base and adds these 3.1 features:
- Type-array nullability — pointer fields render as
nullable: truein 3.0 and astype: ["string", "null"]in 3.1. jsonSchemaDialect— set to the JSON Schema 2020-12 base dialect on the 3.1 document.- SPDX license identifier — set
License.Identifier(e.g."Apache-2.0"); it appears only on the 3.1 document and is mutually exclusive withLicense.URL. const— theconst:"value"struct tag becomes a JSON Schemaconston the 3.1 document.- Webhooks — declare outbound callbacks with
o.Webhook(...); they appear under thewebhooksfield of the 3.1 document only.
o.WithOpenAPIDocs(okapi.OpenAPI{
Title: "Example API",
Version: "1.0.0",
License: okapi.License{Name: "Apache 2.0", Identifier: "Apache-2.0"},
})
// A webhook is documentation-only: it is not added to the router.
o.Webhook("newBook", http.MethodPost,
okapi.DocSummary("Notifies subscribers about a newly added book"),
okapi.DocRequestBody(Book{}),
okapi.DocResponse(200, okapi.M{"received": true}),
)
Accessing Documentation
| Route | Content |
|---|---|
/docs | The selected UI (Swagger UI by default) |
/swagger | Swagger UI |
/redoc | ReDoc |
/scalar | Scalar API Reference |
/openapi.json | OpenAPI spec (JSON) — 3.1 by default |
/openapi.yaml | OpenAPI spec (YAML) — 3.1 by default |
/openapi-3.0.json | OpenAPI 3.0 spec (JSON) |
/openapi-3.0.yaml | OpenAPI 3.0 spec (YAML) |

