/*! * @license * Copyright © 2005-2024 Hyland Software, Inc. and its affiliates. 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. */ import * as fs from 'fs'; import * as path from 'path'; import * as replaceSection from 'mdast-util-heading-range'; import * as remark from 'remark'; import * as ejs from 'ejs'; import { MDNav } from '../mdNav'; import { ngNameToClassName } from '../ngHelpers'; import { ComponentInfo } from '../sourceInfoClasses'; const templateFolder = path.resolve('tools', 'doc', 'templates'); let nameExceptions; export function processDocs(mdCache, aggData) { nameExceptions = aggData.config.typeNameExceptions; const pathNames = Object.keys(mdCache); let internalErrors; pathNames.forEach((pathname) => { internalErrors = []; updateFile(mdCache[pathname].mdOutTree, pathname, aggData, internalErrors); if (internalErrors.length > 0) { showErrors(pathname, internalErrors); } }); } function showErrors(filename, errorMessages) { console.log(filename); errorMessages.forEach((message) => { console.log(' ' + message); }); console.log(''); } function updateFile(tree, pathname, aggData, errorMessages) { const className = ngNameToClassName(path.basename(pathname, '.md'), nameExceptions); const classTypeMatch = className.match(/component|directive|service/i); const compData = aggData.classInfo[className]; if (classTypeMatch && compData) { const classType = classTypeMatch[0].toLowerCase(); // Copy docs back from the .md file when the JSDocs are empty. const inputMD = getPropDocsFromMD(tree, 'Properties', 3); const outputMD = getPropDocsFromMD(tree, 'Events', 2); updatePropDocsFromMD(compData, inputMD, outputMD, errorMessages); if (classType === 'service') { const methodMD = getMethodDocsFromMD(tree); updateMethodDocsFromMD(compData, methodMD, errorMessages); } const templateName = path.resolve(templateFolder, classType + '.ejs'); const templateSource = fs.readFileSync(templateName, 'utf8'); const template = ejs.compile(templateSource); let mdText = template(compData); mdText = mdText.replace(/^ +\|/gm, '|'); const newSection = remark().parse(mdText.trim()).children; replaceSection(tree, 'Class members', (before, section, after) => { newSection.unshift(before); newSection.push(after); return newSection; }); compData.errors.forEach((err) => { errorMessages.push(err); }); } return true; } function getPropDocsFromMD(tree, sectionHeading, docsColumn) { const result = {}; const nav = new MDNav(tree); const classMemHeading = nav.heading((h) => { return h.children[0].type === 'text' && h.children[0].value === 'Class members'; }); const propsTable = classMemHeading .heading((h) => { return h.children[0].type === 'text' && h.children[0].value === sectionHeading; }) .table(); let propTableRow = propsTable.childNav.tableRow(() => true, 1).childNav; let i = 1; while (!propTableRow.empty) { const propName = propTableRow.tableCell().childNav.text().item.value; const propDocText = propTableRow.tableCell(() => true, docsColumn).childNav.text().item; if (propDocText) { result[propName] = propDocText.value; } i++; propTableRow = propsTable.childNav.tableRow(() => true, i).childNav; } return result; } function getMethodDocsFromMD(tree) { const result = {}; const nav = new MDNav(tree); const classMemHeading = nav.heading((h) => { return h.children[0].type === 'text' && h.children[0].value === 'Class members'; }); const methListItems = classMemHeading .heading((h) => { return h.children[0].type === 'text' && h.children[0].value === 'Methods'; }) .list().childNav; let methItem = methListItems.listItem(); let i = 0; while (!methItem.empty) { const methNameSection = methItem.childNav.paragraph().childNav.strong().childNav; let methName = ''; // Method docs must be in "new" format with names and types styled separately. if (!methNameSection.empty) { methName = methNameSection.text().item.value; const methDoc = methItem.childNav.paragraph().childNav.html().text().value; const params = getMDMethodParams(methItem); result[methName] = { docText: methDoc.replace(/^\n/, ''), params: params }; } i++; methItem = methListItems.listItem((l) => true, i); } return result; } function getMDMethodParams(methItem: MDNav) { const result = {}; const paramList = methItem.childNav.list().childNav; const paramListItems = paramList.listItems(); paramListItems.forEach((paramListItem) => { const paramNameNode = paramListItem.childNav.paragraph().childNav.emph().childNav; let paramName; if (!paramNameNode.empty) { paramName = paramNameNode.text().item.value.replace(/:/, ''); } else { let item = paramListItem.childNav.paragraph().childNav.strong().childNav.text(); if (paramName) { paramName = item.value; } } const paramDoc = paramListItem.childNav.paragraph().childNav.text((t) => true, 1).value; // item.value; result[paramName] = paramDoc.replace(/^[ -]+/, ''); }); return result; } function updatePropDocsFromMD(comp: ComponentInfo, inputDocs, outputDocs, errorMessages) { comp.properties.forEach((prop) => { let propMDDoc: string; if (prop.isInput) { propMDDoc = inputDocs[prop.name]; } else if (prop.isOutput) { propMDDoc = outputDocs[prop.name]; } // If JSDocs are empty but MD docs aren't then the Markdown is presumably more up-to-date. if (!prop.docText && propMDDoc) { prop.docText = propMDDoc; errorMessages.push(`Warning: empty JSDocs for property "${prop.name}" may need sync with the .md file.`); } }); } function updateMethodDocsFromMD(comp: ComponentInfo, methodDocs, errorMessages) { comp.methods.forEach((meth) => { const currMethMD = methodDocs[meth.name]; // If JSDocs are empty but MD docs aren't then the Markdown is presumably more up-to-date. if (!meth.docText && currMethMD && currMethMD.docText) { meth.docText = currMethMD.docText; errorMessages.push(`Warning: empty JSDocs for method sig "${meth.name}" may need sync with the .md file.`); } meth.params.forEach((param) => { if (!param.docText && currMethMD && currMethMD.params[param.name]) { param.docText = currMethMD.params[param.name]; errorMessages.push(`Warning: empty JSDocs for parameter "${param.name} (${meth.name})" may need sync with the .md file.`); } }); }); }