diff --git a/docs/configuration/config-modification.md b/docs/configuration/config-modification.md index 3eb36a4cc..dce729574 100644 --- a/docs/configuration/config-modification.md +++ b/docs/configuration/config-modification.md @@ -36,6 +36,7 @@ interface ConfigInterface { receiveLabel?: string; requestLabel?: string; replyLabel?: string; + extensions?: Record>; } ``` @@ -100,6 +101,11 @@ interface ConfigInterface { This field contains configuration responsible for customizing the label for response operation. This takes effect when rendering AsyncAPI v3 documents. This field is set to `REPLY` by default. +- **extensions?: Record>** + + This field contains configuration responsible for adding custom extension components. + This field will contain default components. + ## Examples See exemplary component configuration in TypeScript and JavaScript. @@ -110,6 +116,7 @@ See exemplary component configuration in TypeScript and JavaScript. import * as React from "react"; import { render } from "react-dom"; import AsyncAPIComponent, { ConfigInterface } from "@asyncapi/react-component"; +import CustomExtension from "./CustomExtension"; import { schema } from "./mock"; @@ -126,6 +133,9 @@ const config: ConfigInterface = { expand: { messageExamples: false, }, + extensions: { + 'x-custom-extension': CustomExtension + } }; const App = () => ; @@ -133,6 +143,18 @@ const App = () => ; render(, document.getElementById("root")); ``` +```tsx +// CustomExtension.tsx +import { ExtensionComponentProps } from '@asyncapi/react-component/lib/types/components/Extensions'; + +export default function CustomExtension(props: ExtensionComponentProps) { + return
+

{props.propertyName}

+

{props.propertyValue}

+
+} +``` + ### JavaScript ```jsx @@ -162,6 +184,16 @@ const App = () => ; render(, document.getElementById("root")); ``` +```jsx +// CustomExtension.jsx +export default function CustomExtension(props) { + return
+

{props.propertyName}

+

{props.propertyValue}

+
+} +``` + In the above examples, after concatenation with the default configuration, the resulting configuration looks as follows: ```js @@ -188,6 +220,10 @@ In the above examples, after concatenation with the default configuration, the r sendLabel: 'SEND', receiveLabel: 'RECEIVE', requestLabel: 'REQUEST', - replyLabel: 'REPLY' + replyLabel: 'REPLY', + extensions: { + // default extensions... + 'x-custom-extension': CustomExtension + } } ``` diff --git a/library/src/__tests__/index.test.tsx b/library/src/__tests__/index.test.tsx index 3bb01e8b3..d8c11ba63 100644 --- a/library/src/__tests__/index.test.tsx +++ b/library/src/__tests__/index.test.tsx @@ -6,7 +6,7 @@ import React from 'react'; import { render, waitFor } from '@testing-library/react'; -import AsyncApiComponent from '..'; +import AsyncApiComponent, { ExtensionComponentProps } from '..'; import adoaKafka from './docs/v3/adeo-kafka-request-reply.json'; import krakenMessageFilter from './docs/v3/kraken-websocket-request-reply-message-filter-in-reply.json'; import krakenMultipleChannels from './docs/v3/kraken-websocket-request-reply-multiple-channels.json'; @@ -216,4 +216,54 @@ describe('AsyncAPI component', () => { expect(result.container.querySelector('#introduction')).toBeDefined(), ); }); + + test('should work with custom extensions', async () => { + const schema = { + asyncapi: '2.0.0', + info: { + title: 'Example AsyncAPI', + version: '0.1.0', + }, + channels: { + 'smartylighting/streetlights/1/0/event/{streetlightId}/lighting/measured': + { + subscribe: { + message: { + $ref: '#/components/messages/lightMeasured', + }, + }, + }, + }, + components: { + messages: { + lightMeasured: { + name: 'lightMeasured', + title: 'Light measured', + contentType: 'application/json', + 'x-custom-extension': 'Custom extension value', + }, + }, + }, + }; + + const CustomExtension = (props: ExtensionComponentProps) => ( +
{props.propertyValue}
+ ); + + const result = render( + , + ); + + await waitFor(() => { + expect(result.container.querySelector('#introduction')).toBeDefined(); + expect(result.container.querySelector('#custom-extension')).toBeDefined(); + }); + }); }); diff --git a/library/src/components/Extensions.tsx b/library/src/components/Extensions.tsx index b98d44c00..f71ae4081 100644 --- a/library/src/components/Extensions.tsx +++ b/library/src/components/Extensions.tsx @@ -1,11 +1,13 @@ /* eslint-disable @typescript-eslint/no-unsafe-return */ /* eslint-disable @typescript-eslint/no-unsafe-assignment */ -import React from 'react'; +import React, { useState } from 'react'; import { Schema } from './Schema'; import { SchemaHelpers } from '../helpers'; +import { useConfig, useSpec } from '../contexts'; +import { CollapseButton } from './CollapseButton'; interface Props { name?: string; @@ -17,18 +19,72 @@ export const Extensions: React.FunctionComponent = ({ name = 'Extensions', item, }) => { + const [expanded, setExpanded] = useState(false); + + const config = useConfig(); + const document = useSpec(); + const extensions = SchemaHelpers.getCustomExtensions(item); if (!extensions || !Object.keys(extensions).length) { return null; } - const schema = SchemaHelpers.jsonToSchema(extensions); + if (!config.extensions || !Object.keys(config.extensions).length) { + const schema = SchemaHelpers.jsonToSchema(extensions); + return ( + schema && ( +
+ +
+ ) + ); + } return ( - schema && ( -
- +
+
+
+ <> + setExpanded((prev) => !prev)} + expanded={expanded} + > + {name} + + +
+
+
+ {Object.keys(extensions) + .sort((extension1, extension2) => + extension1.localeCompare(extension2), + ) + .map((extensionKey) => { + if (config.extensions?.[extensionKey]) { + const CustomExtensionComponent = config.extensions[extensionKey]; + return ( + + ); + } else { + const extensionSchema = SchemaHelpers.jsonToSchema( + extensions[extensionKey], + ); + return ( +
+ +
+ ); + } + })}
- ) +
); }; diff --git a/library/src/components/supportedExtensions/XExtension.tsx b/library/src/components/supportedExtensions/XExtension.tsx new file mode 100644 index 000000000..5089eba32 --- /dev/null +++ b/library/src/components/supportedExtensions/XExtension.tsx @@ -0,0 +1,35 @@ +import React from 'react'; + +import { ExtensionComponentProps } from '../../types'; + +/** + * See . + */ +export default function XExtension({ + propertyValue, +}: ExtensionComponentProps) { + return ( + + + + + + ); +} diff --git a/library/src/config/config.ts b/library/src/config/config.ts index 9ad84af18..98fc24c2b 100644 --- a/library/src/config/config.ts +++ b/library/src/config/config.ts @@ -1,3 +1,5 @@ +import { ExtensionComponentProps } from '../types'; + export interface ConfigInterface { schemaID?: string; show?: ShowConfig; @@ -11,6 +13,7 @@ export interface ConfigInterface { receiveLabel?: string; requestLabel?: string; replyLabel?: string; + extensions?: Record>; } export interface ShowConfig { diff --git a/library/src/config/default.ts b/library/src/config/default.ts index 12e6dffa6..26025bba0 100644 --- a/library/src/config/default.ts +++ b/library/src/config/default.ts @@ -7,6 +7,7 @@ import { SEND_LABEL_DEFAULT_TEXT, SUBSCRIBE_LABEL_DEFAULT_TEXT, } from '../constants'; +import XExtension from '../components/supportedExtensions/XExtension'; export const defaultConfig: ConfigInterface = { schemaID: '', @@ -33,4 +34,7 @@ export const defaultConfig: ConfigInterface = { receiveLabel: RECEIVE_TEXT_LABEL_DEFAULT_TEXT, requestLabel: REQUEST_LABEL_DEFAULT_TEXT, replyLabel: REPLIER_LABEL_DEFAULT_TEXT, + extensions: { + 'x-x': XExtension, + }, }; diff --git a/library/src/containers/AsyncApi/Standalone.tsx b/library/src/containers/AsyncApi/Standalone.tsx index b4ac3665a..6228f93a8 100644 --- a/library/src/containers/AsyncApi/Standalone.tsx +++ b/library/src/containers/AsyncApi/Standalone.tsx @@ -69,6 +69,10 @@ class AsyncApiComponent extends Component { ...defaultConfig.sidebar, ...(!!config && config.sidebar), }, + extensions: { + ...defaultConfig.extensions, + ...(!!config && config.extensions), + }, }; if (!asyncapi) { diff --git a/library/src/containers/Info/Info.tsx b/library/src/containers/Info/Info.tsx index a32eeb269..dbc5dfbcb 100644 --- a/library/src/containers/Info/Info.tsx +++ b/library/src/containers/Info/Info.tsx @@ -1,6 +1,6 @@ import React from 'react'; -import { Href, Markdown, Tags } from '../../components'; +import { Extensions, Href, Markdown, Tags } from '../../components'; import { useSpec } from '../../contexts'; import { TERMS_OF_SERVICE_TEXT, @@ -23,6 +23,7 @@ export const Info: React.FunctionComponent = () => { const termsOfService = info.termsOfService(); const defaultContentType = asyncapi.defaultContentType(); const contact = info.contact(); + const extensions = info.extensions(); const showInfoList = license ?? termsOfService ?? defaultContentType ?? contact ?? externalDocs; @@ -128,6 +129,12 @@ export const Info: React.FunctionComponent = () => {
)} + + {extensions.length > 0 && ( +
+ +
+ )}
diff --git a/library/src/index.ts b/library/src/index.ts index f5fc0a8c1..be2b7ef65 100644 --- a/library/src/index.ts +++ b/library/src/index.ts @@ -3,7 +3,7 @@ import AsyncApiComponentWP from './containers/AsyncApi/Standalone'; export { AsyncApiProps } from './containers/AsyncApi/AsyncApi'; export { ConfigInterface } from './config/config'; -export { FetchingSchemaInterface } from './types'; +export { FetchingSchemaInterface, ExtensionComponentProps } from './types'; import { hljs } from './helpers'; diff --git a/library/src/types.ts b/library/src/types.ts index 9f58864e2..3d3c2cdbb 100644 --- a/library/src/types.ts +++ b/library/src/types.ts @@ -1,4 +1,4 @@ -import { AsyncAPIDocumentInterface } from '@asyncapi/parser'; +import { AsyncAPIDocumentInterface, BaseModel } from '@asyncapi/parser'; export type PropsSchema = | string @@ -77,3 +77,11 @@ export interface ErrorObject { endOffset: number; }[]; } + +// eslint-disable-next-line @typescript-eslint/no-explicit-any +export interface ExtensionComponentProps { + propertyName: string; + propertyValue: V; + document: AsyncAPIDocumentInterface; + parent: BaseModel; +} diff --git a/playground/specs/streetlights.ts b/playground/specs/streetlights.ts index 58ab8fb5f..fa178ed0d 100644 --- a/playground/specs/streetlights.ts +++ b/playground/specs/streetlights.ts @@ -1,6 +1,7 @@ export const streetlights = `asyncapi: '2.6.0' id: 'urn:com:smartylighting:streetlights:server' info: + x-x: AsyncAPISpec title: Streetlights API version: '1.0.0' description: |