2019-11-19 05:18:28 +03:00
|
|
|
#!/usr/bin/env node
|
|
|
|
/**
|
|
|
|
* Copyright 2017 Google Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2020-12-28 18:03:09 +03:00
|
|
|
//@ts-check
|
|
|
|
|
2020-05-23 10:03:57 +03:00
|
|
|
const playwright = require('../../');
|
2020-12-03 00:50:10 +03:00
|
|
|
const fs = require('fs');
|
2019-11-19 05:18:28 +03:00
|
|
|
const path = require('path');
|
|
|
|
const Source = require('./Source');
|
2020-12-29 04:38:00 +03:00
|
|
|
const md = require('../markdown');
|
2020-12-03 20:11:48 +03:00
|
|
|
const { spawnSync } = require('child_process');
|
2020-12-04 20:03:33 +03:00
|
|
|
const preprocessor = require('./preprocessor');
|
2020-12-29 04:38:00 +03:00
|
|
|
const { MDOutline } = require('./MDBuilder');
|
|
|
|
const missingDocs = require('./missingDocs');
|
2020-12-30 04:35:01 +03:00
|
|
|
const Documentation = require('./Documentation');
|
2020-02-14 05:26:38 +03:00
|
|
|
|
2020-12-29 03:19:28 +03:00
|
|
|
/** @typedef {import('./Documentation').Type} Type */
|
2020-12-29 04:38:00 +03:00
|
|
|
/** @typedef {import('../markdown').MarkdownNode} MarkdownNode */
|
2020-12-28 18:03:09 +03:00
|
|
|
|
2019-11-19 05:18:28 +03:00
|
|
|
const PROJECT_DIR = path.join(__dirname, '..', '..');
|
|
|
|
const VERSION = require(path.join(PROJECT_DIR, 'package.json')).version;
|
|
|
|
|
|
|
|
const RED_COLOR = '\x1b[31m';
|
|
|
|
const YELLOW_COLOR = '\x1b[33m';
|
|
|
|
const RESET_COLOR = '\x1b[0m';
|
|
|
|
|
2020-12-03 20:21:56 +03:00
|
|
|
run().catch(e => {
|
|
|
|
console.error(e);
|
|
|
|
process.exit(1);
|
|
|
|
});;
|
2019-11-19 05:18:28 +03:00
|
|
|
|
|
|
|
async function run() {
|
|
|
|
const startTime = Date.now();
|
|
|
|
|
2020-12-05 05:05:35 +03:00
|
|
|
const api = await Source.readFile(path.join(PROJECT_DIR, 'docs', 'api.md'));
|
|
|
|
const readme = await Source.readFile(path.join(PROJECT_DIR, 'README.md'));
|
|
|
|
const binReadme = await Source.readFile(path.join(PROJECT_DIR, 'bin', 'README.md'));
|
|
|
|
const contributing = await Source.readFile(path.join(PROJECT_DIR, 'CONTRIBUTING.md'));
|
|
|
|
const docs = await Source.readdir(path.join(PROJECT_DIR, 'docs'), '.md');
|
|
|
|
const mdSources = [readme, binReadme, api, contributing, ...docs];
|
|
|
|
|
2020-12-29 03:19:28 +03:00
|
|
|
/** @type {!Array<string>} */
|
|
|
|
const errors = [];
|
2019-11-19 05:18:28 +03:00
|
|
|
let changedFiles = false;
|
|
|
|
|
2020-12-28 18:03:09 +03:00
|
|
|
const header = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-header.md')).toString();
|
|
|
|
const footer = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-footer.md')).toString();
|
|
|
|
const links = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-links.md')).toString();
|
2020-12-29 04:38:00 +03:00
|
|
|
const outline = new MDOutline(path.join(PROJECT_DIR, 'docs-src', 'api-body.md'), path.join(PROJECT_DIR, 'docs-src', 'api-params.md'));
|
2020-12-28 18:03:09 +03:00
|
|
|
|
2020-12-03 00:50:10 +03:00
|
|
|
// Produce api.md
|
|
|
|
{
|
2020-12-29 10:42:51 +03:00
|
|
|
outline.renderLinks(item => {
|
|
|
|
const { clazz, member, param, option } = item;
|
|
|
|
if (param)
|
|
|
|
return `\`${param}\``;
|
|
|
|
if (option)
|
|
|
|
return `\`${option}\``;
|
|
|
|
if (clazz)
|
|
|
|
return `[${clazz.name}]`;
|
|
|
|
if (member.kind === 'method')
|
|
|
|
return createMemberLink(`${member.clazz.varName}.${member.name}(${member.signature})`);
|
|
|
|
if (member.kind === 'event')
|
|
|
|
return createMemberLink(`${member.clazz.varName}.on('${member.name}')`);
|
|
|
|
if (member.kind === 'property')
|
|
|
|
return createMemberLink(`${member.clazz.varName}.${member.name}`);
|
|
|
|
throw new Error('Unknown member kind ' + member.kind);
|
|
|
|
});
|
|
|
|
|
2020-12-03 00:50:10 +03:00
|
|
|
const comment = '<!-- THIS FILE IS NOW GENERATED -->';
|
2020-12-04 20:03:33 +03:00
|
|
|
{
|
2020-12-29 04:38:00 +03:00
|
|
|
/** @type {MarkdownNode[]} */
|
2020-12-28 18:03:09 +03:00
|
|
|
const result = [];
|
|
|
|
for (const clazz of outline.classesArray) {
|
|
|
|
// Iterate over classes, create header node.
|
2020-12-29 04:38:00 +03:00
|
|
|
/** @type {MarkdownNode} */
|
2020-12-28 18:03:09 +03:00
|
|
|
const classNode = { type: 'h3', text: `class: ${clazz.name}` };
|
|
|
|
result.push(classNode);
|
|
|
|
// Append link shortcut to resolve text like [Browser]
|
|
|
|
result.push({
|
|
|
|
type: 'text',
|
|
|
|
text: `[${clazz.name}]: #class-${clazz.name.toLowerCase()} "${clazz.name}"`
|
|
|
|
});
|
|
|
|
// Append class comments
|
2020-12-29 04:38:00 +03:00
|
|
|
classNode.children = (clazz.spec || []).map(c => md.clone(c));
|
2020-12-30 04:35:01 +03:00
|
|
|
classNode.children.push({
|
|
|
|
type: 'text',
|
|
|
|
text: '<!-- TOC -->'
|
|
|
|
});
|
|
|
|
classNode.children.push(...generateToc(clazz));
|
|
|
|
if (clazz.extends && clazz.extends !== 'EventEmitter' && clazz.extends !== 'Error') {
|
|
|
|
const superClass = outline.documentation.classes.get(clazz.extends);
|
|
|
|
classNode.children.push(...generateToc(superClass));
|
|
|
|
}
|
2020-12-28 18:03:09 +03:00
|
|
|
|
|
|
|
for (const member of clazz.membersArray) {
|
|
|
|
// Iterate members
|
2020-12-29 04:38:00 +03:00
|
|
|
/** @type {MarkdownNode} */
|
2020-12-28 18:03:09 +03:00
|
|
|
const memberNode = { type: 'h4', children: [] };
|
|
|
|
if (member.kind === 'event') {
|
2020-12-29 10:42:51 +03:00
|
|
|
memberNode.text = `${clazz.varName}.on('${member.name}')`;
|
2020-12-28 18:03:09 +03:00
|
|
|
} else if (member.kind === 'property') {
|
2020-12-29 10:42:51 +03:00
|
|
|
memberNode.text = `${clazz.varName}.${member.name}`;
|
2020-12-28 18:03:09 +03:00
|
|
|
} else if (member.kind === 'method') {
|
|
|
|
// Patch method signatures
|
2020-12-29 10:42:51 +03:00
|
|
|
memberNode.text = `${clazz.varName}.${member.name}(${member.signature})`;
|
2020-12-28 18:03:09 +03:00
|
|
|
for (const arg of member.argsArray) {
|
|
|
|
if (arg.type)
|
|
|
|
memberNode.children.push(renderProperty(`\`${arg.name}\``, arg.type, arg.spec));
|
|
|
|
}
|
2020-12-04 22:09:20 +03:00
|
|
|
}
|
|
|
|
|
2020-12-28 18:03:09 +03:00
|
|
|
// Append type
|
|
|
|
if (member.type && member.type.name !== 'void') {
|
|
|
|
let name;
|
|
|
|
switch (member.kind) {
|
|
|
|
case 'event': name = 'type:'; break;
|
|
|
|
case 'property': name = 'type:'; break;
|
|
|
|
case 'method': name = 'returns:'; break;
|
2020-12-05 05:05:35 +03:00
|
|
|
}
|
2020-12-28 18:03:09 +03:00
|
|
|
memberNode.children.push(renderProperty(name, member.type));
|
2020-12-04 22:09:20 +03:00
|
|
|
}
|
|
|
|
|
2020-12-28 18:03:09 +03:00
|
|
|
// Append member doc
|
2020-12-29 04:38:00 +03:00
|
|
|
memberNode.children.push(...(member.spec || []).map(c => md.clone(c)));
|
2020-12-28 18:03:09 +03:00
|
|
|
classNode.children.push(memberNode);
|
2020-12-04 20:03:33 +03:00
|
|
|
}
|
2020-12-04 09:28:11 +03:00
|
|
|
}
|
2020-12-28 18:03:09 +03:00
|
|
|
result.push({
|
|
|
|
type: 'text',
|
|
|
|
text: links
|
2020-12-28 21:54:47 +03:00
|
|
|
});
|
2020-12-29 04:38:00 +03:00
|
|
|
api.setText([comment, header, md.render(result), footer].join('\n'));
|
2020-12-04 09:28:11 +03:00
|
|
|
}
|
2020-12-03 00:50:10 +03:00
|
|
|
}
|
|
|
|
|
2019-11-19 05:18:28 +03:00
|
|
|
// Documentation checks.
|
|
|
|
{
|
2020-03-13 00:03:01 +03:00
|
|
|
const browserVersions = await getBrowserVersions();
|
2020-12-29 03:19:28 +03:00
|
|
|
errors.push(...(await preprocessor.runCommands(mdSources, {
|
2020-02-14 05:26:38 +03:00
|
|
|
libversion: VERSION,
|
|
|
|
chromiumVersion: browserVersions.chromium,
|
|
|
|
firefoxVersion: browserVersions.firefox,
|
2020-12-22 05:09:55 +03:00
|
|
|
webkitVersion: browserVersions.webkit,
|
2020-02-14 05:26:38 +03:00
|
|
|
})));
|
2020-07-22 21:03:35 +03:00
|
|
|
|
2020-12-29 03:19:28 +03:00
|
|
|
errors.push(...preprocessor.autocorrectInvalidLinks(PROJECT_DIR, mdSources, getRepositoryFiles()));
|
2020-12-04 23:19:48 +03:00
|
|
|
for (const source of mdSources.filter(source => source.hasUpdatedText()))
|
2020-12-29 03:19:28 +03:00
|
|
|
errors.push(`WARN: updated ${source.projectPath()}`);
|
2020-12-04 23:19:48 +03:00
|
|
|
|
|
|
|
const jsSources = await Source.readdir(path.join(PROJECT_DIR, 'src', 'client'), '', []);
|
2020-12-29 04:38:00 +03:00
|
|
|
errors.push(...missingDocs(outline, jsSources, path.join(PROJECT_DIR, 'src', 'client', 'api.ts')));
|
2019-11-19 05:18:28 +03:00
|
|
|
|
|
|
|
for (const source of mdSources) {
|
|
|
|
if (!source.hasUpdatedText())
|
|
|
|
continue;
|
|
|
|
await source.save();
|
|
|
|
changedFiles = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Report results.
|
|
|
|
if (errors.length) {
|
|
|
|
for (let i = 0; i < errors.length; ++i) {
|
2020-12-29 03:19:28 +03:00
|
|
|
const error = errors[i].split('\n').join('\n ');
|
2019-11-19 05:18:28 +03:00
|
|
|
console.log(` ${i + 1}) ${RED_COLOR}${error}${RESET_COLOR}`);
|
|
|
|
}
|
|
|
|
}
|
2020-12-29 03:19:28 +03:00
|
|
|
let clearExit = errors.length === 0;
|
2019-11-19 05:18:28 +03:00
|
|
|
if (changedFiles) {
|
|
|
|
if (clearExit)
|
|
|
|
console.log(`${YELLOW_COLOR}Some files were updated.${RESET_COLOR}`);
|
|
|
|
clearExit = false;
|
|
|
|
}
|
2020-12-29 03:19:28 +03:00
|
|
|
console.log(`${errors.length} failures.`);
|
2019-11-19 05:18:28 +03:00
|
|
|
const runningTime = Date.now() - startTime;
|
|
|
|
console.log(`DocLint Finished in ${runningTime / 1000} seconds`);
|
2020-12-04 23:19:48 +03:00
|
|
|
process.exit(clearExit ? 0 : 1);
|
2019-11-19 05:18:28 +03:00
|
|
|
}
|
2020-02-14 05:26:38 +03:00
|
|
|
|
2020-03-13 00:03:01 +03:00
|
|
|
async function getBrowserVersions() {
|
2020-12-22 05:09:55 +03:00
|
|
|
const names = ['chromium', 'firefox', 'webkit'];
|
|
|
|
const browsers = await Promise.all(names.map(name => playwright[name].launch()));
|
|
|
|
const result = {};
|
|
|
|
for (let i = 0; i < names.length; i++) {
|
|
|
|
result[names[i]] = browsers[i].version();
|
2020-03-13 00:03:01 +03:00
|
|
|
}
|
2020-12-22 05:09:55 +03:00
|
|
|
await Promise.all(browsers.map(browser => browser.close()));
|
|
|
|
return result;
|
2020-03-13 00:03:01 +03:00
|
|
|
}
|
|
|
|
|
2020-04-24 05:52:06 +03:00
|
|
|
function getRepositoryFiles() {
|
|
|
|
const out = spawnSync('git', ['ls-files'], {cwd: PROJECT_DIR});
|
2020-12-04 22:09:20 +03:00
|
|
|
const files = out.stdout.toString().trim().split('\n').filter(f => !f.startsWith('docs-src'));
|
|
|
|
return files.map(file => path.join(PROJECT_DIR, file));
|
2020-04-24 05:52:06 +03:00
|
|
|
}
|
2020-12-28 18:03:09 +03:00
|
|
|
|
2020-12-30 04:35:01 +03:00
|
|
|
|
|
|
|
const memberLinks = new Map();
|
|
|
|
const rMemberLinks = new Map();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} text
|
|
|
|
*/
|
|
|
|
function createMemberLink(text) {
|
|
|
|
if (memberLinks.has(text))
|
|
|
|
return memberLinks.get(text);
|
|
|
|
const baseAnchor = text.toLowerCase().split(',').map(c => c.replace(/[^a-z]/g, '')).join('-');
|
|
|
|
let anchor = baseAnchor;
|
|
|
|
let index = 0;
|
|
|
|
while (rMemberLinks.has(anchor))
|
|
|
|
anchor = baseAnchor + '-' + (++index);
|
|
|
|
const result = `[${text}](#${anchor})`;
|
|
|
|
memberLinks.set(text, result);
|
|
|
|
rMemberLinks.set(anchor, text);
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Documentation.Class} clazz
|
|
|
|
* @return {MarkdownNode[]}
|
|
|
|
*/
|
|
|
|
function generateToc(clazz) {
|
|
|
|
/** @type {MarkdownNode[]} */
|
|
|
|
const result = [];
|
|
|
|
for (const member of clazz.membersArray) {
|
|
|
|
if (member.kind === 'property') {
|
|
|
|
result.push({
|
|
|
|
type: 'li',
|
|
|
|
liType: 'default',
|
|
|
|
text: createMemberLink(`${clazz.varName}.${member.name}`)
|
|
|
|
});
|
|
|
|
} else if (member.kind === 'event') {
|
|
|
|
result.push({
|
|
|
|
type: 'li',
|
|
|
|
liType: 'default',
|
|
|
|
text: createMemberLink(`${clazz.varName}.on('${member.name}')`)
|
|
|
|
});
|
|
|
|
} else if (member.kind === 'method') {
|
|
|
|
result.push({
|
|
|
|
type: 'li',
|
|
|
|
liType: 'default',
|
|
|
|
text: createMemberLink(`${clazz.varName}.${member.name}(${member.signature})`)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2020-12-28 18:03:09 +03:00
|
|
|
/**
|
|
|
|
* @param {string} name
|
|
|
|
* @param {Type} type
|
|
|
|
* @param {MarkdownNode[]} [spec]
|
|
|
|
*/
|
|
|
|
function renderProperty(name, type, spec) {
|
|
|
|
let comment = '';
|
|
|
|
if (spec && spec.length)
|
|
|
|
comment = spec[0].text;
|
|
|
|
let children;
|
|
|
|
if (type.properties && type.properties.length)
|
|
|
|
children = type.properties.map(p => renderProperty(`\`${p.name}\``, p.type, p.spec))
|
|
|
|
else if (spec && spec.length > 1)
|
2020-12-29 04:38:00 +03:00
|
|
|
children = spec.slice(1).map(s => md.clone(s));
|
2020-12-28 18:03:09 +03:00
|
|
|
|
2020-12-29 04:38:00 +03:00
|
|
|
/** @type {MarkdownNode} */
|
2020-12-28 18:03:09 +03:00
|
|
|
const result = {
|
|
|
|
type: 'li',
|
|
|
|
liType: 'default',
|
|
|
|
text: `${name} <${renderType(type.name)}>${comment ? ' ' + comment : ''}`,
|
|
|
|
children
|
|
|
|
};
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} type
|
|
|
|
*/
|
|
|
|
function renderType(type) {
|
|
|
|
if (type.includes('"'))
|
|
|
|
return type.replace(/,/g, '|').replace(/Array/, "[Array]").replace(/null/, "[null]").replace(/number/, "[number]");
|
|
|
|
const result = type.replace(/([\w]+)/g, '[$1]');
|
|
|
|
if (result === '[Promise]<[void]>')
|
|
|
|
return '[Promise]';
|
|
|
|
return result.replace(/[(]/g, '\\(').replace(/[)]/g, '\\)');
|
|
|
|
}
|