416 lines
13 KiB
Markdown
416 lines
13 KiB
Markdown
|
# Plug points
|
|||
|
|
|||
|
Swagger UI exposes most of its internal logic through the plugin system.
|
|||
|
|
|||
|
Often, it is beneficial to override the core internals to achieve custom behavior.
|
|||
|
|
|||
|
### Note: Semantic Versioning
|
|||
|
|
|||
|
Swagger UI's internal APIs are _not_ part of our public contract, which means that they can change without the major version change.
|
|||
|
|
|||
|
If your custom plugins wrap, extend, override, or consume any internal core APIs, we recommend specifying a specific minor version of Swagger UI to use in your application, because they will _not_ change between patch versions.
|
|||
|
|
|||
|
If you're installing Swagger UI via NPM, for example, you can do this by using a tilde:
|
|||
|
|
|||
|
```js
|
|||
|
{
|
|||
|
"dependencies": {
|
|||
|
"swagger-ui": "~3.11.0"
|
|||
|
}
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
### `fn.opsFilter`
|
|||
|
|
|||
|
When using the `filter` option, tag names will be filtered by the user-provided value. If you'd like to customize this behavior, you can override the default `opsFilter` function.
|
|||
|
|
|||
|
For example, you can implement a multiple-phrase filter:
|
|||
|
|
|||
|
```js
|
|||
|
const MultiplePhraseFilterPlugin = function() {
|
|||
|
return {
|
|||
|
fn: {
|
|||
|
opsFilter: (taggedOps, phrase) => {
|
|||
|
const phrases = phrase.split(", ")
|
|||
|
|
|||
|
return taggedOps.filter((val, key) => {
|
|||
|
return phrases.some(item => key.indexOf(item) > -1)
|
|||
|
})
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
```
|
|||
|
### Logo component
|
|||
|
While using the Standalone Preset the SwaggerUI logo is rendered in the Top Bar.
|
|||
|
The logo can be exchanged by replacing the `Logo` component via the plugin api:
|
|||
|
|
|||
|
```jsx
|
|||
|
import React from "react";
|
|||
|
const MyLogoPlugin = {
|
|||
|
components: {
|
|||
|
Logo: () => (
|
|||
|
<img alt="My Logo" height="40" src=""/>
|
|||
|
)
|
|||
|
}
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
|
|||
|
### JSON Schema components
|
|||
|
In swagger there are so called JSON Schema components. These are used to render inputs for parameters and components of request bodies with `application/x-www-form-urlencoded` or `multipart/*` media-type.
|
|||
|
|
|||
|
Internally swagger uses following mapping to find the JSON Schema component from OpenAPI Specification schema information:
|
|||
|
|
|||
|
For each schema’s type(eg. `string`, `array`, …) and if defined schema’s format (eg. ‘date’, ‘uuid’, …) there is a corresponding component mapping:
|
|||
|
|
|||
|
**If format defined:**
|
|||
|
```js
|
|||
|
`JsonSchema_${type}_${format}`
|
|||
|
```
|
|||
|
|
|||
|
**Fallback if `JsonSchema_${type}_${format}` component does not exist or format not defined:**
|
|||
|
```js
|
|||
|
`JsonSchema_${type}`
|
|||
|
```
|
|||
|
|
|||
|
**Default:**
|
|||
|
```js
|
|||
|
`JsonSchema_string`
|
|||
|
```
|
|||
|
|
|||
|
With this, one can define custom input components or override existing.
|
|||
|
|
|||
|
#### Example Date-Picker plugin
|
|||
|
|
|||
|
If one would like to input date values you could provide a custom plugin to integrate [react-datepicker](https://www.npmjs.com/package/react-datepicker) into swagger-ui.
|
|||
|
All you need to do is to create a component to wrap [react-datepicker](https://www.npmjs.com/package/react-datepicker) accordingly to the format.
|
|||
|
|
|||
|
**There are two cases:**
|
|||
|
- ```yaml
|
|||
|
type: string
|
|||
|
format: date
|
|||
|
```
|
|||
|
The resulting name for mapping to succeed: `JsonSchema_string_date`
|
|||
|
- ```yaml
|
|||
|
type: string
|
|||
|
format: date-time
|
|||
|
```
|
|||
|
The resulting name for mapping to succeed: `JsonSchema_string_date-time`
|
|||
|
|
|||
|
This creates the need for two components and simple logic to strip any time input in case the format is date:
|
|||
|
```js
|
|||
|
import React from "react";
|
|||
|
import DatePicker from "react-datepicker";
|
|||
|
import "react-datepicker/dist/react-datepicker.css";
|
|||
|
|
|||
|
const JsonSchema_string_date = (props) => {
|
|||
|
const dateNumber = Date.parse(props.value);
|
|||
|
const date = dateNumber
|
|||
|
? new Date(dateNumber)
|
|||
|
: new Date();
|
|||
|
|
|||
|
return (
|
|||
|
<DatePicker
|
|||
|
selected={date}
|
|||
|
onChange={d => props.onChange(d.toISOString().substring(0, 10))}
|
|||
|
/>
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
const JsonSchema_string_date_time = (props) => {
|
|||
|
const dateNumber = Date.parse(props.value);
|
|||
|
const date = dateNumber
|
|||
|
? new Date(dateNumber)
|
|||
|
: new Date();
|
|||
|
|
|||
|
return (
|
|||
|
<DatePicker
|
|||
|
selected={date}
|
|||
|
onChange={d => props.onChange(d.toISOString())}
|
|||
|
showTimeSelect
|
|||
|
timeFormat="p"
|
|||
|
dateFormat="Pp"
|
|||
|
/>
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
export const DateTimeSwaggerPlugin = {
|
|||
|
components: {
|
|||
|
JsonSchema_string_date: JsonSchema_string_date,
|
|||
|
"JsonSchema_string_date-time": JsonSchema_string_date_time
|
|||
|
}
|
|||
|
};
|
|||
|
```
|
|||
|
|
|||
|
### Request Snippets
|
|||
|
|
|||
|
SwaggerUI can be configured with the `requestSnippetsEnabled: true` option to activate Request Snippets.
|
|||
|
Instead of the generic curl that is generated upon doing a request. It gives you more granular options:
|
|||
|
- curl for bash
|
|||
|
- curl for cmd
|
|||
|
- curl for powershell
|
|||
|
|
|||
|
There might be the case where you want to provide your own snipped generator. This can be done by using the plugin api.
|
|||
|
A Request Snipped generator consists of the configuration and a `fn`,
|
|||
|
which takes the internal request object and transforms it to the desired snippet.
|
|||
|
|
|||
|
```js
|
|||
|
// Add config to Request Snippets Configuration with an unique key like "node_native"
|
|||
|
const snippetConfig = {
|
|||
|
requestSnippetsEnabled: true,
|
|||
|
requestSnippets: {
|
|||
|
generators: {
|
|||
|
"node_native": {
|
|||
|
title: "NodeJs Native",
|
|||
|
syntax: "javascript"
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
const SnippedGeneratorNodeJsPlugin = {
|
|||
|
fn: {
|
|||
|
// use `requestSnippetGenerator_` + key from config (node_native) for generator fn
|
|||
|
requestSnippetGenerator_node_native: (request) => {
|
|||
|
const url = new Url(request.get("url"))
|
|||
|
let isMultipartFormDataRequest = false
|
|||
|
const headers = request.get("headers")
|
|||
|
if(headers && headers.size) {
|
|||
|
request.get("headers").map((val, key) => {
|
|||
|
isMultipartFormDataRequest = isMultipartFormDataRequest || /^content-type$/i.test(key) && /^multipart\/form-data$/i.test(val)
|
|||
|
})
|
|||
|
}
|
|||
|
const packageStr = url.protocol === "https:" ? "https" : "http"
|
|||
|
let reqBody = request.get("body")
|
|||
|
if (request.get("body")) {
|
|||
|
if (isMultipartFormDataRequest && ["POST", "PUT", "PATCH"].includes(request.get("method"))) {
|
|||
|
return "throw new Error(\"Currently unsupported content-type: /^multipart\\/form-data$/i\");"
|
|||
|
} else {
|
|||
|
if (!Map.isMap(reqBody)) {
|
|||
|
if (typeof reqBody !== "string") {
|
|||
|
reqBody = JSON.stringify(reqBody)
|
|||
|
}
|
|||
|
} else {
|
|||
|
reqBody = getStringBodyOfMap(request)
|
|||
|
}
|
|||
|
}
|
|||
|
} else if (!request.get("body") && request.get("method") === "POST") {
|
|||
|
reqBody = ""
|
|||
|
}
|
|||
|
|
|||
|
const stringBody = "`" + (reqBody || "")
|
|||
|
.replace(/\\n/g, "\n")
|
|||
|
.replace(/`/g, "\\`")
|
|||
|
+ "`"
|
|||
|
|
|||
|
return `const http = require("${packageStr}");
|
|||
|
const options = {
|
|||
|
"method": "${request.get("method")}",
|
|||
|
"hostname": "${url.host}",
|
|||
|
"port": ${url.port || "null"},
|
|||
|
"path": "${url.pathname}"${headers && headers.size ? `,
|
|||
|
"headers": {
|
|||
|
${request.get("headers").map((val, key) => `"${key}": "${val}"`).valueSeq().join(",\n ")}
|
|||
|
}` : ""}
|
|||
|
};
|
|||
|
const req = http.request(options, function (res) {
|
|||
|
const chunks = [];
|
|||
|
res.on("data", function (chunk) {
|
|||
|
chunks.push(chunk);
|
|||
|
});
|
|||
|
res.on("end", function () {
|
|||
|
const body = Buffer.concat(chunks);
|
|||
|
console.log(body.toString());
|
|||
|
});
|
|||
|
});
|
|||
|
${reqBody ? `\nreq.write(${stringBody});` : ""}
|
|||
|
req.end();`
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
const ui = SwaggerUIBundle({
|
|||
|
"dom_id": "#swagger-ui",
|
|||
|
deepLinking: true,
|
|||
|
presets: [
|
|||
|
SwaggerUIBundle.presets.apis,
|
|||
|
SwaggerUIStandalonePreset
|
|||
|
],
|
|||
|
plugins: [
|
|||
|
SwaggerUIBundle.plugins.DownloadUrl,
|
|||
|
SnippedGeneratorNodeJsPlugin
|
|||
|
],
|
|||
|
layout: "StandaloneLayout",
|
|||
|
validatorUrl: "https://validator.swagger.io/validator",
|
|||
|
url: "https://petstore.swagger.io/v2/swagger.json",
|
|||
|
...snippetConfig,
|
|||
|
})
|
|||
|
```
|
|||
|
|
|||
|
### Error handling
|
|||
|
|
|||
|
SwaggerUI comes with a `safe-render` plugin that handles error handling allows plugging into error handling system and modify it.
|
|||
|
|
|||
|
The plugin accepts a list of component names that should be protected by error boundaries.
|
|||
|
|
|||
|
Its public API looks like this:
|
|||
|
|
|||
|
```js
|
|||
|
{
|
|||
|
fn: {
|
|||
|
componentDidCatch,
|
|||
|
withErrorBoundary: withErrorBoundary(getSystem),
|
|||
|
},
|
|||
|
components: {
|
|||
|
ErrorBoundary,
|
|||
|
Fallback,
|
|||
|
},
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
safe-render plugin is automatically utilized by [base](https://github.com/swagger-api/swagger-ui/blob/78f62c300a6d137e65fd027d850981b010009970/src/core/presets/base.js) and [standalone](https://github.com/swagger-api/swagger-ui/tree/78f62c300a6d137e65fd027d850981b010009970/src/standalone) SwaggerUI presets and
|
|||
|
should always be used as the last plugin, after all the components are already known to the SwaggerUI.
|
|||
|
The plugin defines a default list of components that should be protected by error boundaries:
|
|||
|
|
|||
|
```js
|
|||
|
[
|
|||
|
"App",
|
|||
|
"BaseLayout",
|
|||
|
"VersionPragmaFilter",
|
|||
|
"InfoContainer",
|
|||
|
"ServersContainer",
|
|||
|
"SchemesContainer",
|
|||
|
"AuthorizeBtnContainer",
|
|||
|
"FilterContainer",
|
|||
|
"Operations",
|
|||
|
"OperationContainer",
|
|||
|
"parameters",
|
|||
|
"responses",
|
|||
|
"OperationServers",
|
|||
|
"Models",
|
|||
|
"ModelWrapper",
|
|||
|
"Topbar",
|
|||
|
"StandaloneLayout",
|
|||
|
"onlineValidatorBadge"
|
|||
|
]
|
|||
|
```
|
|||
|
|
|||
|
As demonstrated below, additional components can be protected by utilizing the safe-render plugin
|
|||
|
with configuration options. This gets really handy if you are a SwaggerUI integrator and you maintain a number of
|
|||
|
plugins with additional custom components.
|
|||
|
|
|||
|
```js
|
|||
|
const swaggerUI = SwaggerUI({
|
|||
|
url: "https://petstore.swagger.io/v2/swagger.json",
|
|||
|
dom_id: '#swagger-ui',
|
|||
|
plugins: [
|
|||
|
() => ({
|
|||
|
components: {
|
|||
|
MyCustomComponent1: () => 'my custom component',
|
|||
|
},
|
|||
|
}),
|
|||
|
SwaggerUI.plugins.SafeRender({
|
|||
|
fullOverride: true, // only the component list defined here will apply (not the default list)
|
|||
|
componentList: [
|
|||
|
"MyCustomComponent1",
|
|||
|
],
|
|||
|
}),
|
|||
|
],
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
##### componentDidCatch
|
|||
|
|
|||
|
This static function is invoked after a component has thrown an error.
|
|||
|
It receives two parameters:
|
|||
|
|
|||
|
1. `error` - The error that was thrown.
|
|||
|
2. `info` - An object with a componentStack key containing [information about which component threw the error](https://reactjs.org/docs/error-boundaries.html#component-stack-traces).
|
|||
|
|
|||
|
It has precisely the same signature as error boundaries [componentDidCatch lifecycle method](https://reactjs.org/docs/react-component.html#componentdidcatch),
|
|||
|
except it's a static function and not a class method.
|
|||
|
|
|||
|
Default implement of componentDidCatch uses `console.error` to display the received error:
|
|||
|
|
|||
|
```js
|
|||
|
export const componentDidCatch = console.error;
|
|||
|
```
|
|||
|
|
|||
|
To utilize your own error handling logic (e.g. [bugsnag](https://www.bugsnag.com/)), create new SwaggerUI plugin that overrides componentDidCatch:
|
|||
|
|
|||
|
{% highlight js linenos %}
|
|||
|
const BugsnagErrorHandlerPlugin = () => {
|
|||
|
// init bugsnag
|
|||
|
|
|||
|
return {
|
|||
|
fn: {
|
|||
|
componentDidCatch = (error, info) => {
|
|||
|
Bugsnag.notify(error);
|
|||
|
Bugsnag.notify(info);
|
|||
|
},
|
|||
|
},
|
|||
|
};
|
|||
|
};
|
|||
|
{% endhighlight %}
|
|||
|
|
|||
|
##### withErrorBoundary
|
|||
|
|
|||
|
This function is HOC (Higher Order Component). It wraps a particular component into the `ErrorBoundary` component.
|
|||
|
It can be overridden via a plugin system to control how components are wrapped by the ErrorBoundary component.
|
|||
|
In 99.9% of situations, you won't need to override this function, but if you do, please read the source code of this function first.
|
|||
|
|
|||
|
##### Fallback
|
|||
|
|
|||
|
The component is displayed when the error boundary catches an error. It can be overridden via a plugin system.
|
|||
|
Its default implementation is trivial:
|
|||
|
|
|||
|
```js
|
|||
|
import React from "react"
|
|||
|
import PropTypes from "prop-types"
|
|||
|
|
|||
|
const Fallback = ({ name }) => (
|
|||
|
<div className="fallback">
|
|||
|
😱 <i>Could not render { name === "t" ? "this component" : name }, see the console.</i>
|
|||
|
</div>
|
|||
|
)
|
|||
|
Fallback.propTypes = {
|
|||
|
name: PropTypes.string.isRequired,
|
|||
|
}
|
|||
|
export default Fallback
|
|||
|
```
|
|||
|
|
|||
|
Feel free to override it to match your look & feel:
|
|||
|
|
|||
|
```js
|
|||
|
const CustomFallbackPlugin = () => ({
|
|||
|
components: {
|
|||
|
Fallback: ({ name } ) => `This is my custom fallback. ${name} failed to render`,
|
|||
|
},
|
|||
|
});
|
|||
|
|
|||
|
const swaggerUI = SwaggerUI({
|
|||
|
url: "https://petstore.swagger.io/v2/swagger.json",
|
|||
|
dom_id: '#swagger-ui',
|
|||
|
plugins: [
|
|||
|
CustomFallbackPlugin,
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
##### ErrorBoundary
|
|||
|
|
|||
|
This is the component that implements React error boundaries. Uses `componentDidCatch` and `Fallback`
|
|||
|
under the hood. In 99.9% of situations, you won't need to override this component, but if you do,
|
|||
|
please read the source code of this component first.
|
|||
|
|
|||
|
|
|||
|
##### Change in behavior
|
|||
|
|
|||
|
In prior releases of SwaggerUI (before v4.3.0), almost all components have been protected, and when thrown error,
|
|||
|
`Fallback` component was displayed. This changes with SwaggerUI v4.3.0. Only components defined
|
|||
|
by the `safe-render` plugin are now protected and display fallback. If a small component somewhere within
|
|||
|
SwaggerUI React component tree fails to render and throws an error. The error bubbles up to the closest
|
|||
|
error boundary, and that error boundary displays the `Fallback` component and invokes `componentDidCatch`.
|