feat: add build-docs command (#863)

This commit is contained in:
Alex Varchuk
2022-09-19 11:39:31 +03:00
committed by GitHub
parent bb8b9121c4
commit 6f21fc4cb8
16 changed files with 1866 additions and 294 deletions

View File

@@ -0,0 +1,116 @@
import * as React from 'react';
import { createStore, Redoc } from 'redoc';
import { parseYaml, findConfig, Config } from '@redocly/openapi-core';
import { renderToString } from 'react-dom/server';
import { ServerStyleSheet } from 'styled-components';
import { compile } from 'handlebars';
import { join } from 'path';
import { existsSync, lstatSync, readFileSync } from 'fs';
import type { BuildDocsOptions } from './types';
import { red, yellow } from 'colorette';
import { exitWithError } from '../../utils';
export function getObjectOrJSON(
options: string | Record<string, unknown>
): JSON | Record<string, unknown> | Config {
switch (typeof options) {
case 'object':
return options;
case 'string':
try {
if (existsSync(options) && lstatSync(options).isFile()) {
return JSON.parse(readFileSync(options, 'utf-8'));
} else {
return JSON.parse(options);
}
} catch (e) {
process.stderr.write(
red(
`Encountered error:\n\n${options}\n\nis neither a file with a valid JSON object neither a stringified JSON object.`
)
);
exitWithError(e);
}
break;
default: {
const configFile = findConfig();
if (configFile) {
process.stderr.write(`Found ${configFile} and using features.openapi options`);
try {
const config = parseYaml(readFileSync(configFile, 'utf-8')) as Config;
return config['features.openapi'];
} catch (e) {
process.stderr.write(yellow(`Found ${configFile} but failed to parse: ${e.message}`));
}
}
return {};
}
}
return {};
}
export async function getPageHTML(
api: any,
pathToApi: string,
{
cdn,
title,
disableGoogleFont,
templateFileName,
templateOptions,
redocOptions = {},
redocCurrentVersion,
}: BuildDocsOptions
) {
process.stderr.write('Prerendering docs');
const apiUrl = redocOptions.specUrl || (isURL(pathToApi) ? pathToApi : undefined);
const store = await createStore(api, apiUrl, redocOptions);
const sheet = new ServerStyleSheet();
const html = renderToString(sheet.collectStyles(React.createElement(Redoc, { store })));
const state = await store.toJS();
const css = sheet.getStyleTags();
templateFileName = templateFileName ? templateFileName : join(__dirname, './template.hbs');
const template = compile(readFileSync(templateFileName).toString());
return template({
redocHTML: `
<div id="redoc">${html || ''}</div>
<script>
${`const __redoc_state = ${sanitizeJSONString(JSON.stringify(state))};` || ''}
var container = document.getElementById('redoc');
Redoc.${'hydrate(__redoc_state, container)'};
</script>`,
redocHead:
(cdn
? '<script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"></script>'
: `<script src="https://cdn.redoc.ly/redoc/v${redocCurrentVersion}/bundles/redoc.standalone.js"></script>`) +
css,
title: title || api.info.title || 'ReDoc documentation',
disableGoogleFont,
templateOptions,
});
}
export function isURL(str: string): boolean {
return /^(https?:)\/\//m.test(str);
}
export function sanitizeJSONString(str: string): string {
return escapeClosingScriptTag(escapeUnicode(str));
}
// see http://www.thespanner.co.uk/2011/07/25/the-json-specification-is-now-wrong/
export function escapeClosingScriptTag(str: string): string {
return str.replace(/<\/script>/g, '<\\/script>');
}
// see http://www.thespanner.co.uk/2011/07/25/the-json-specification-is-now-wrong/
export function escapeUnicode(str: string): string {
return str.replace(/\u2028|\u2029/g, (m) => '\\u202' + (m === '\u2028' ? '8' : '9'));
}