2019-03-12 22:07:57 +00:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright 2019 Balena Ltd.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
2019-04-02 11:26:21 +00:00
|
|
|
import { flagUsages } from '@oclif/parser';
|
2017-12-21 17:40:13 +00:00
|
|
|
import * as ent from 'ent';
|
2019-03-12 22:07:57 +00:00
|
|
|
import * as _ from 'lodash';
|
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
import { getManualSortCompareFunction } from '../../lib/utils/helpers';
|
2019-11-09 01:56:11 +00:00
|
|
|
import { capitanoizeOclifUsage } from '../../lib/utils/oclif-utils';
|
2019-04-02 11:26:21 +00:00
|
|
|
import { CapitanoCommand, Category, Document, OclifCommand } from './doc-types';
|
2017-12-21 17:40:13 +00:00
|
|
|
import * as utils from './utils';
|
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
function renderCapitanoCommand(command: CapitanoCommand): string[] {
|
|
|
|
const result = [`## ${ent.encode(command.signature)}`, command.help];
|
2017-12-21 17:40:13 +00:00
|
|
|
|
|
|
|
if (!_.isEmpty(command.options)) {
|
2019-04-02 11:26:21 +00:00
|
|
|
result.push('### Options');
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-03-12 22:07:57 +00:00
|
|
|
for (const option of command.options!) {
|
2019-05-28 16:31:18 +00:00
|
|
|
if (option == null) {
|
|
|
|
throw new Error(`Undefined option in markdown generation!`);
|
|
|
|
}
|
2019-04-02 11:26:21 +00:00
|
|
|
result.push(
|
|
|
|
`#### ${utils.parseCapitanoOption(option)}`,
|
|
|
|
option.description,
|
|
|
|
);
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
2019-04-02 11:26:21 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function renderOclifCommand(command: OclifCommand): string[] {
|
|
|
|
const result = [`## ${ent.encode(command.usage)}`];
|
|
|
|
const description = (command.description || '')
|
|
|
|
.split('\n')
|
|
|
|
.slice(1) // remove the first line, which oclif uses as help header
|
|
|
|
.join('\n')
|
|
|
|
.trim();
|
|
|
|
result.push(description);
|
|
|
|
|
|
|
|
if (!_.isEmpty(command.examples)) {
|
|
|
|
result.push('Examples:', command.examples!.map(v => `\t${v}`).join('\n'));
|
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
if (!_.isEmpty(command.args)) {
|
|
|
|
result.push('### Arguments');
|
|
|
|
for (const arg of command.args!) {
|
|
|
|
result.push(`#### ${arg.name.toUpperCase()}`, arg.description || '');
|
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
if (!_.isEmpty(command.flags)) {
|
|
|
|
result.push('### Options');
|
|
|
|
for (const [name, flag] of Object.entries(command.flags!)) {
|
|
|
|
if (name === 'help') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
flag.name = name;
|
|
|
|
const flagUsage = flagUsages([flag])
|
|
|
|
.map(([usage, _description]) => usage)
|
|
|
|
.join()
|
|
|
|
.trim();
|
|
|
|
result.push(`#### ${flagUsage}`);
|
|
|
|
result.push(flag.description || '');
|
|
|
|
}
|
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
return result;
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
function renderCategory(category: Category): string[] {
|
|
|
|
const result = [`# ${category.title}`];
|
2019-03-12 22:07:57 +00:00
|
|
|
for (const command of category.commands) {
|
2019-04-02 11:26:21 +00:00
|
|
|
result.push(
|
|
|
|
...(typeof command === 'object'
|
|
|
|
? renderCapitanoCommand(command)
|
|
|
|
: renderOclifCommand(command)),
|
|
|
|
);
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
|
|
|
return result;
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
function getAnchor(cmdSignature: string): string {
|
|
|
|
return `#${_.trim(cmdSignature.replace(/\W+/g, '-'), '-').toLowerCase()}`;
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
function renderToc(categories: Category[]): string[] {
|
|
|
|
const result = [`# CLI Command Reference`];
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-03-12 22:07:57 +00:00
|
|
|
for (const category of categories) {
|
2019-04-02 11:26:21 +00:00
|
|
|
result.push(`- ${category.title}`);
|
|
|
|
result.push(
|
|
|
|
category.commands
|
|
|
|
.map(command => {
|
|
|
|
const signature =
|
|
|
|
typeof command === 'object'
|
|
|
|
? command.signature // Capitano
|
2019-11-09 01:56:11 +00:00
|
|
|
: capitanoizeOclifUsage(command.usage); // oclif
|
2019-04-02 11:26:21 +00:00
|
|
|
return `\t- [${ent.encode(signature)}](${getAnchor(signature)})`;
|
|
|
|
})
|
|
|
|
.join('\n'),
|
|
|
|
);
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
|
|
|
return result;
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
const manualCategorySorting: { [category: string]: string[] } = {
|
|
|
|
'Environment Variables': ['envs', 'env rm', 'env add', 'env rename'],
|
2019-10-31 01:46:14 +00:00
|
|
|
OS: [
|
|
|
|
'os versions',
|
|
|
|
'os download',
|
|
|
|
'os build config',
|
|
|
|
'os configure',
|
|
|
|
'os initialize',
|
|
|
|
],
|
2019-04-02 11:26:21 +00:00
|
|
|
};
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
function sortCommands(doc: Document): void {
|
2019-03-12 22:07:57 +00:00
|
|
|
for (const category of doc.categories) {
|
2019-04-02 11:26:21 +00:00
|
|
|
if (category.title in manualCategorySorting) {
|
|
|
|
category.commands = category.commands.sort(
|
|
|
|
getManualSortCompareFunction<CapitanoCommand | OclifCommand, string>(
|
|
|
|
manualCategorySorting[category.title],
|
|
|
|
(cmd: CapitanoCommand | OclifCommand, x: string) =>
|
|
|
|
typeof cmd === 'object' // Capitano vs oclif command
|
|
|
|
? cmd.signature.replace(/\W+/g, ' ').includes(x)
|
|
|
|
: (cmd.usage || '')
|
|
|
|
.toString()
|
|
|
|
.replace(/\W+/g, ' ')
|
|
|
|
.includes(x),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
}
|
2019-04-02 11:26:21 +00:00
|
|
|
}
|
2017-12-21 17:40:13 +00:00
|
|
|
|
2019-04-02 11:26:21 +00:00
|
|
|
export function render(doc: Document) {
|
|
|
|
sortCommands(doc);
|
|
|
|
const result = [
|
|
|
|
`# ${doc.title}`,
|
|
|
|
doc.introduction,
|
|
|
|
...renderToc(doc.categories),
|
|
|
|
];
|
|
|
|
for (const category of doc.categories) {
|
|
|
|
result.push(...renderCategory(category));
|
|
|
|
}
|
|
|
|
return result.join('\n\n');
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|