playwright/utils/doclint/api_parser.js
2022-11-23 08:40:47 -08:00

464 lines
15 KiB
JavaScript

/**
* 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.
*/
// @ts-check
const fs = require('fs');
const path = require('path');
const md = require('../markdown');
const docs = require('./documentation');
/** @typedef {import('../markdown').MarkdownNode} MarkdownNode */
/** @typedef {import('../markdown').MarkdownHeaderNode} MarkdownHeaderNode */
/** @typedef {import('../markdown').MarkdownLiNode} MarkdownLiNode */
/** @typedef {import('../markdown').MarkdownTextNode} MarkdownTextNode */
class ApiParser {
/**
* @param {string} apiDir
* @param {string=} paramsPath
*/
constructor(apiDir, paramsPath) {
let bodyParts = [];
for (const name of fs.readdirSync(apiDir)) {
if (!name.endsWith('.md'))
continue;
if (name === 'params.md')
paramsPath = path.join(apiDir, name);
else
bodyParts.push(fs.readFileSync(path.join(apiDir, name)).toString());
}
const body = md.parse(bodyParts.join('\n'));
const params = paramsPath ? md.parse(fs.readFileSync(paramsPath).toString()) : undefined;
checkNoDuplicateParamEntries(params);
const api = params ? applyTemplates(body, params) : body;
/** @type {Map<string, docs.Class>} */
this.classes = new Map();
md.visitAll(api, node => {
if (node.type === 'h1')
this.parseClass(node);
});
md.visitAll(api, node => {
if (node.type === 'h2')
this.parseMember(node);
});
md.visitAll(api, node => {
if (node.type === 'h3')
this.parseArgument(node);
});
this.documentation = new docs.Documentation([...this.classes.values()]);
this.documentation.index();
}
/**
* @param {MarkdownHeaderNode} node
*/
parseClass(node) {
let extendsName = null;
const name = node.text.substring('class: '.length);
for (const member of node.children) {
if (member.type.startsWith('h'))
continue;
if (member.type === 'li' && member.liType === 'bullet' && member.text.startsWith('extends: [')) {
extendsName = member.text.substring('extends: ['.length, member.text.indexOf(']'));
continue;
}
}
const clazz = new docs.Class(extractMetainfo(node), name, [], extendsName, extractComments(node));
this.classes.set(clazz.name, clazz);
}
/**
* @param {MarkdownHeaderNode} spec
*/
parseMember(spec) {
const match = spec.text.match(/(event|method|property|async method|optional method|optional async method): ([^.]+)\.(.*)/);
if (!match)
throw new Error('Invalid member: ' + spec.text);
const name = match[3];
let returnType = null;
let optional = false;
for (const item of spec.children || []) {
if (item.type === 'li' && item.liType === 'default') {
const parsed = this.parseType(item);
returnType = parsed.type;
optional = parsed.optional;
}
}
if (!returnType)
returnType = new docs.Type('void');
const comments = extractComments(spec);
let member;
if (match[1] === 'event')
member = docs.Member.createEvent(extractMetainfo(spec), name, returnType, comments);
if (match[1] === 'property')
member = docs.Member.createProperty(extractMetainfo(spec), name, returnType, comments, !optional);
if (['method', 'async method', 'optional method', 'optional async method'].includes(match[1])) {
member = docs.Member.createMethod(extractMetainfo(spec), name, [], returnType, comments);
if (match[1].includes('async'))
member.async = true;
if (match[1].includes('optional'))
member.required = false;
}
if (!member)
throw new Error('Unknown member: ' + spec.text);
const clazz = /** @type {docs.Class} */(this.classes.get(match[2]));
const existingMember = clazz.membersArray.find(m => m.name === name && m.kind === member.kind);
if (existingMember && isTypeOverride(existingMember, member)) {
for (const lang of member?.langs?.only || []) {
existingMember.langs.types = existingMember.langs.types || {};
existingMember.langs.types[lang] = returnType;
}
} else {
clazz.membersArray.push(member);
}
}
/**
* @param {MarkdownHeaderNode} spec
*/
parseArgument(spec) {
const match = spec.text.match(/(param|option): (.*)/);
if (!match)
throw `Something went wrong with matching ${spec.text}`;
// For "test.describe.only.title":
// - className is "test"
// - methodName is "describe.only"
// - argument name is "title"
const parts = match[2].split('.');
const className = parts[0];
const name = parts[parts.length - 1];
const methodName = parts.slice(1, parts.length - 1).join('.');
const clazz = this.classes.get(className);
if (!clazz)
throw new Error('Invalid class ' + className);
const method = clazz.membersArray.find(m => m.kind === 'method' && m.name === methodName);
if (!method)
throw new Error(`Invalid method ${className}.${methodName} when parsing: ${match[0]}`);
if (!name)
throw new Error('Invalid member name ' + spec.text);
if (match[1] === 'param') {
const arg = this.parseProperty(spec);
arg.name = name;
const existingArg = method.argsArray.find(m => m.name === arg.name);
if (existingArg && isTypeOverride(existingArg, arg)) {
if (!arg.langs || !arg.langs.only)
throw new Error('Override does not have lang: ' + spec.text);
for (const lang of arg.langs.only) {
existingArg.langs.overrides = existingArg.langs.overrides || {};
existingArg.langs.overrides[lang] = arg;
}
} else {
method.argsArray.push(arg);
}
} else {
// match[1] === 'option'
let options = method.argsArray.find(o => o.name === 'options');
if (!options) {
const type = new docs.Type('Object', []);
options = docs.Member.createProperty({ langs: {}, experimental: false, since: 'v1.0', deprecated: undefined, discouraged: undefined }, 'options', type, undefined, false);
method.argsArray.push(options);
}
const p = this.parseProperty(spec);
p.required = false;
// @ts-ignore
options.type.properties.push(p);
}
}
/**
* @param {MarkdownHeaderNode} spec
*/
parseProperty(spec) {
const param = childrenWithoutProperties(spec)[0];
const text = /** @type {string}*/(param.text);
let typeStart = text.indexOf('<');
while ('?e'.includes(text[typeStart - 1]))
typeStart--;
const name = text.substring(0, typeStart).replace(/\`/g, '').trim();
const comments = extractComments(spec);
const { type, optional } = this.parseType(/** @type {MarkdownLiNode} */(param));
return docs.Member.createProperty(extractMetainfo(spec), name, type, comments, !optional);
}
/**
* @param {MarkdownLiNode} spec
* @return {{ type: docs.Type, optional: boolean, experimental: boolean }}
*/
parseType(spec) {
const arg = parseVariable(spec.text);
const properties = [];
for (const child of /** @type {MarkdownLiNode[]} */ (spec.children) || []) {
const { name, text } = parseVariable(/** @type {string} */(child.text));
const comments = /** @type {MarkdownNode[]} */ ([{ type: 'text', text }]);
const childType = this.parseType(child);
properties.push(docs.Member.createProperty({ langs: {}, experimental: childType.experimental, since: 'v1.0', deprecated: undefined, discouraged: undefined }, name, childType.type, comments, !childType.optional));
}
const type = docs.Type.parse(arg.type, properties);
return { type, optional: arg.optional, experimental: arg.experimental };
}
}
/**
* @param {string} line
* @returns {{ name: string, type: string, text: string, optional: boolean, experimental: boolean }}
*/
function parseVariable(line) {
let match = line.match(/^`([^`]+)` (.*)/);
if (!match)
match = line.match(/^(returns): (.*)/);
if (!match)
match = line.match(/^(type): (.*)/);
if (!match)
match = line.match(/^(argument): (.*)/);
if (!match)
throw new Error('Invalid argument: ' + line);
const name = match[1];
let remainder = match[2];
let optional = false;
let experimental = false;
while ('?e'.includes(remainder[0])) {
if (remainder[0] === '?')
optional = true;
else if (remainder[0] === 'e')
experimental = true;
remainder = remainder.substring(1);
}
if (!remainder.startsWith('<'))
throw new Error(`Bad argument: "${name}" in "${line}"`);
let depth = 0;
for (let i = 0; i < remainder.length; ++i) {
const c = remainder.charAt(i);
if (c === '<')
++depth;
if (c === '>')
--depth;
if (depth === 0)
return { name, type: remainder.substring(1, i), text: remainder.substring(i + 2), optional, experimental };
}
throw new Error('Should not be reached, line: ' + line);
}
/**
* @param {MarkdownNode[]} body
* @param {MarkdownNode[]} params
*/
function applyTemplates(body, params) {
const paramsMap = new Map();
for (const node of params)
paramsMap.set('%%-' + node.text + '-%%', node);
const visit = (node, parent) => {
if (node.text && node.text.includes('-inline- = %%')) {
const [name, key] = node.text.split('-inline- = ');
const list = paramsMap.get(key);
const newChildren = [];
if (!list)
throw new Error('Bad template: ' + key);
for (const prop of list.children) {
const template = paramsMap.get(prop.text);
if (!template)
throw new Error('Bad template: ' + prop.text);
const children = childrenWithoutProperties(template);
const { name: argName } = parseVariable(children[0].text || '');
newChildren.push({
type: node.type,
text: name + argName,
children: [...node.children, ...template.children.map(c => md.clone(c))]
});
}
const nodeIndex = parent.children.indexOf(node);
parent.children = [...parent.children.slice(0, nodeIndex), ...newChildren, ...parent.children.slice(nodeIndex + 1)];
} else if (node.text && node.text.includes(' = %%')) {
const [name, key] = node.text.split(' = ');
node.text = name;
const template = paramsMap.get(key);
if (!template)
throw new Error('Bad template: ' + key);
node.children.push(...template.children.map(c => md.clone(c)));
} else if (node.text && node.text.includes('%%-template-')) {
node.text.replace(/%%-template-[^%]+-%%/, templateName => {
const template = paramsMap.get(templateName);
if (!template)
throw new Error('Bad template: ' + templateName);
const nodeIndex = parent.children.indexOf(node);
parent.children = [...parent.children.slice(0, nodeIndex), ...template.children, ...parent.children.slice(nodeIndex + 1)];
});
}
for (const child of node.children || [])
visit(child, node);
if (node.children)
node.children = node.children.filter(child => !child.text || !child.text.includes('-inline- = %%'));
};
for (const node of body)
visit(node, null);
return body;
}
/**
* @param {MarkdownHeaderNode} item
* @returns {MarkdownNode[]}
*/
function extractComments(item) {
return childrenWithoutProperties(item).filter(c => {
if (c.type.startsWith('h'))
return false;
if (c.type === 'li' && c.liType === 'default')
return false;
return true;
});
}
/**
* @param {string} apiDir
* @param {string=} paramsPath
*/
function parseApi(apiDir, paramsPath) {
return new ApiParser(apiDir, paramsPath).documentation;
}
/**
* @param {MarkdownHeaderNode} spec
* @returns {import('./documentation').Metainfo}
*/
function extractMetainfo(spec) {
return {
langs: extractLangs(spec),
since: extractSince(spec),
experimental: extractExperimental(spec),
deprecated: extractAttribute(spec, 'deprecated'),
discouraged: extractAttribute(spec, 'discouraged'),
};
}
/**
* @param {MarkdownNode} spec
* @returns {import('./documentation').Langs}
*/
function extractLangs(spec) {
for (const child of spec.children || []) {
if (child.type !== 'li' || child.liType !== 'bullet' || !child.text.startsWith('langs:'))
continue;
const only = child.text.substring('langs:'.length).trim();
/** @type {Object<string, string>} */
const aliases = {};
for (const p of child.children || []) {
const match = /** @type {string}*/(p.text).match(/alias-(\w+)[\s]*:(.*)/);
if (match)
aliases[match[1].trim()] = match[2].trim();
}
return {
only: only ? only.split(',').map(l => l.trim()) : undefined,
aliases,
types: {},
overrides: {}
};
}
return {};
}
/**
* @param {MarkdownHeaderNode} spec
* @returns {string}
*/
function extractSince(spec) {
for (const child of spec.children) {
if (child.type !== 'li' || child.liType !== 'bullet' || !child.text.startsWith('since:'))
continue;
return child.text.substring(child.text.indexOf(':') + 1).trim();
}
console.error('Missing since: v1.** declaration in node:');
console.error(spec);
process.exit(1);
}
/**
* @param {MarkdownHeaderNode} spec
* @returns {boolean}
*/
function extractExperimental(spec) {
for (const child of spec.children) {
if (child.type === 'li' && child.liType === 'bullet' && child.text === 'experimental')
return true;
}
return false;
}
/**
* @param {MarkdownHeaderNode} spec
* @param {string} name
* @returns {string | undefined}
*/
function extractAttribute(spec, name) {
for (const child of spec.children) {
if (child.type !== 'li' || child.liType !== 'bullet' || !child.text.startsWith(name + ':'))
continue;
return child.text.substring(child.text.indexOf(':') + 1).trim() || undefined;
}
}
/**
* @param {MarkdownHeaderNode} spec
* @returns {MarkdownNode[]}
*/
function childrenWithoutProperties(spec) {
return (spec.children || []).filter(c => {
const isProperty = c.type === 'li' && c.liType === 'bullet' && (c.text.startsWith('langs:') || c.text.startsWith('since:') || c.text.startsWith('deprecated:') || c.text.startsWith('discouraged:') || c.text === 'experimental');
return !isProperty;
});
}
/**
* @param {docs.Member} existingMember
* @param {docs.Member} member
* @returns {boolean}
*/
function isTypeOverride(existingMember, member) {
if (!existingMember.langs.only || !member.langs.only)
return true;
const existingOnly = existingMember.langs.only;
if (member.langs.only.every(l => existingOnly.includes(l))) {
return true;
} else if (member.langs.only.some(l => existingOnly.includes(l))) {
throw new Error(`Ambiguous language override for: ${member.name}`);
}
return false;
}
/**
* @param {MarkdownNode[]=} params
*/
function checkNoDuplicateParamEntries(params) {
if (!params)
return;
const entries = new Set();
for (const node of params) {
if (entries.has(node.text))
throw new Error('Duplicate param entry, for language-specific params use prefix (e.g. js-...): ' + node.text);
entries.add(node.text);
}
}
module.exports = { parseApi };