| 
									
										
										
										
											2019-11-19 10:18:28 +08: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 23:03:09 +08:00
										 |  |  | //@ts-check
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-23 15:03:57 +08:00
										 |  |  | const playwright = require('../../'); | 
					
						
							| 
									
										
										
										
											2020-12-03 05:50:10 +08:00
										 |  |  | const fs = require('fs'); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | const path = require('path'); | 
					
						
							| 
									
										
										
										
											2020-12-29 09:38:00 +08:00
										 |  |  | const md = require('../markdown'); | 
					
						
							|  |  |  | const { MDOutline } = require('./MDBuilder'); | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  | const Documentation = require('./Documentation'); | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  | const missingDocs = require('./missingDocs'); | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 08:19:28 +08:00
										 |  |  | /** @typedef {import('./Documentation').Type} Type */ | 
					
						
							| 
									
										
										
										
											2020-12-29 09:38:00 +08:00
										 |  |  | /** @typedef {import('../markdown').MarkdownNode} MarkdownNode */ | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | const PROJECT_DIR = path.join(__dirname, '..', '..'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | const links = new Map(); | 
					
						
							|  |  |  | const rLinks = new Map(); | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  | const dirtyFiles = new Set(); | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 01:21:56 +08:00
										 |  |  | run().catch(e => { | 
					
						
							|  |  |  |   console.error(e); | 
					
						
							|  |  |  |   process.exit(1); | 
					
						
							|  |  |  | });; | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | async function run() { | 
					
						
							| 
									
										
										
										
											2020-12-29 09:38:00 +08:00
										 |  |  |   const outline = new MDOutline(path.join(PROJECT_DIR, 'docs-src', 'api-body.md'), path.join(PROJECT_DIR, 'docs-src', 'api-params.md')); | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   outline.setLinkRenderer(item => { | 
					
						
							|  |  |  |     const { clazz, member, param, option } = item; | 
					
						
							|  |  |  |     if (param) | 
					
						
							|  |  |  |       return `\`${param}\``; | 
					
						
							|  |  |  |     if (option) | 
					
						
							|  |  |  |       return `\`${option}\``; | 
					
						
							|  |  |  |     if (clazz) | 
					
						
							|  |  |  |       return `[${clazz.name}]`; | 
					
						
							|  |  |  |     return createMemberLink(member); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |   let generatedLinksSuffix; | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   { | 
					
						
							|  |  |  |     const links = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'links.md')).toString(); | 
					
						
							|  |  |  |     const localLinks = []; | 
					
						
							|  |  |  |     for (const clazz of outline.classesArray) | 
					
						
							|  |  |  |       localLinks.push(`[${clazz.name}]: api/class-${clazz.name.toLowerCase()}.md "${clazz.name}"`); | 
					
						
							|  |  |  |     generatedLinksSuffix = localLinks.join('\n') + '\n' + links; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-03 05:50:10 +08:00
										 |  |  |   // Produce api.md
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |     for (const clazz of outline.classesArray) { | 
					
						
							| 
									
										
										
										
											2020-12-29 09:38:00 +08:00
										 |  |  |       /** @type {MarkdownNode[]} */ | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08:00
										 |  |  |       const result = []; | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |       result.push({ | 
					
						
							|  |  |  |         type: 'text', | 
					
						
							|  |  |  |         text: `---
 | 
					
						
							|  |  |  | id: class-${clazz.name.toLowerCase()} | 
					
						
							|  |  |  | title: "class: ${clazz.name}" | 
					
						
							|  |  |  | --- | 
					
						
							|  |  |  | `});
 | 
					
						
							|  |  |  |       result.push(...(clazz.spec || []).map(c => md.clone(c))); | 
					
						
							|  |  |  |       result.push({ | 
					
						
							|  |  |  |         type: 'text', | 
					
						
							|  |  |  |         text: '' | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       result.push(...generateClassToc(clazz)); | 
					
						
							|  |  |  |       if (clazz.extends && clazz.extends !== 'EventEmitter' && clazz.extends !== 'Error') { | 
					
						
							|  |  |  |         const superClass = outline.documentation.classes.get(clazz.extends); | 
					
						
							|  |  |  |         result.push(...generateClassToc(superClass)); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |       for (const member of clazz.membersArray) { | 
					
						
							|  |  |  |         // Iterate members
 | 
					
						
							|  |  |  |         /** @type {MarkdownNode} */ | 
					
						
							|  |  |  |         const memberNode = { type: 'h4', children: [] }; | 
					
						
							|  |  |  |         if (member.kind === 'event') { | 
					
						
							|  |  |  |           memberNode.text = `${clazz.varName}.on('${member.name}')`; | 
					
						
							|  |  |  |         } else if (member.kind === 'property') { | 
					
						
							|  |  |  |           memberNode.text = `${clazz.varName}.${member.name}`; | 
					
						
							|  |  |  |         } else if (member.kind === 'method') { | 
					
						
							|  |  |  |           // Patch method signatures
 | 
					
						
							|  |  |  |           memberNode.text = `${clazz.varName}.${member.name}(${member.signature})`; | 
					
						
							|  |  |  |           for (const arg of member.argsArray) { | 
					
						
							|  |  |  |             if (arg.type) | 
					
						
							|  |  |  |               memberNode.children.push(renderProperty(`\`${arg.name}\``, arg.type, arg.spec)); | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08: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 03:09:20 +08:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |           memberNode.children.push(renderProperty(name, member.type)); | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Append member doc
 | 
					
						
							|  |  |  |         memberNode.children.push(...(member.spec || []).map(c => md.clone(c))); | 
					
						
							|  |  |  |         result.push(memberNode); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:28:11 +08:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |       writeAssumeNoop(path.join(PROJECT_DIR, 'docs', `class-${clazz.name.toLowerCase()}.md`), [md.render(result), generatedLinksSuffix].join('\n'), dirtyFiles); | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Produce other docs
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |     for (const name of fs.readdirSync(path.join(PROJECT_DIR, 'docs-src'))) { | 
					
						
							|  |  |  |       if (name === 'links.md' || name.startsWith('api-')) | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |         continue; | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |       const content = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', name)).toString(); | 
					
						
							|  |  |  |       const nodes = md.parse(content); | 
					
						
							|  |  |  |       outline.renderLinksInText(nodes); | 
					
						
							|  |  |  |       for (const node of nodes) { | 
					
						
							|  |  |  |         if (node.text === '<!-- TOC -->') | 
					
						
							|  |  |  |           node.text = md.generateToc(nodes); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       writeAssumeNoop(path.join(PROJECT_DIR, 'docs', name), [md.render(nodes), generatedLinksSuffix].join('\n'), dirtyFiles); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:28:11 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-03 05:50:10 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   // Patch README.md
 | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |     const versions = await getBrowserVersions(); | 
					
						
							|  |  |  |     const params = new Map(); | 
					
						
							|  |  |  |     const { chromium, firefox, webkit } = versions; | 
					
						
							|  |  |  |     params.set('chromium-version', chromium); | 
					
						
							|  |  |  |     params.set('firefox-version', firefox); | 
					
						
							|  |  |  |     params.set('webkit-version', webkit); | 
					
						
							|  |  |  |     params.set('chromium-version-badge', `[](https://www.chromium.org/Home)`); | 
					
						
							|  |  |  |     params.set('firefox-version-badge', `[](https://www.mozilla.org/en-US/firefox/new/)`); | 
					
						
							|  |  |  |     params.set('webkit-version-badge', `[](https://webkit.org/)`); | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |     let content = fs.readFileSync(path.join(PROJECT_DIR, 'README.md')).toString(); | 
					
						
							|  |  |  |     content = content.replace(/<!-- GEN:([^ ]+) -->([^<]*)<!-- GEN:stop -->/ig, (match, p1) => { | 
					
						
							|  |  |  |       if (!params.has(p1)) { | 
					
						
							|  |  |  |         console.log(`ERROR: Invalid generate parameter "${p1}" in "${match}"`); | 
					
						
							|  |  |  |         process.exit(1); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return `<!-- GEN:${p1} -->${params.get(p1)}<!-- GEN:stop -->`; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     writeAssumeNoop(path.join(PROJECT_DIR, 'README.md'), content, dirtyFiles); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   // Check for missing docs
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     const srcClient = path.join(PROJECT_DIR, 'src', 'client'); | 
					
						
							|  |  |  |     const sources = fs.readdirSync(srcClient).map(n => path.join(srcClient, n)); | 
					
						
							|  |  |  |     const errors = missingDocs(outline, sources, path.join(srcClient, 'api.ts')); | 
					
						
							|  |  |  |     if (errors.length) { | 
					
						
							|  |  |  |       console.log('============================'); | 
					
						
							|  |  |  |       console.log('ERROR: missing documentation:'); | 
					
						
							|  |  |  |       errors.forEach(e => console.log(e)); | 
					
						
							|  |  |  |       console.log('============================') | 
					
						
							|  |  |  |       process.exit(1); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   if (dirtyFiles.size) { | 
					
						
							|  |  |  |     console.log('============================') | 
					
						
							|  |  |  |     console.log('ERROR: generated markdown files have changed, this is only error if happens in CI:'); | 
					
						
							|  |  |  |     [...dirtyFiles].forEach(f => console.log(f)); | 
					
						
							|  |  |  |     console.log('============================') | 
					
						
							|  |  |  |     process.exit(1); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   process.exit(0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @param {string} name | 
					
						
							|  |  |  |  * @param {string} content | 
					
						
							|  |  |  |  * @param {Set<string>} dirtyFiles | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function writeAssumeNoop(name, content, dirtyFiles) { | 
					
						
							|  |  |  |   const oldContent = fs.readFileSync(name).toString(); | 
					
						
							|  |  |  |   if (oldContent !== content) { | 
					
						
							|  |  |  |     fs.writeFileSync(name, content); | 
					
						
							|  |  |  |     dirtyFiles.add(name); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  | async function getBrowserVersions() { | 
					
						
							| 
									
										
										
										
											2020-12-22 10:09:55 +08: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 05:03:01 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-12-22 10:09:55 +08:00
										 |  |  |   await Promise.all(browsers.map(browser => browser.close())); | 
					
						
							|  |  |  |   return result; | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |  * @param {string} file | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  |  * @param {string} text | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | function createLink(file, text) { | 
					
						
							|  |  |  |   const key = file + '#' + text; | 
					
						
							|  |  |  |   if (links.has(key)) | 
					
						
							|  |  |  |     return links.get(key); | 
					
						
							|  |  |  |   const baseLink = file + '#' + text.toLowerCase().split(',').map(c => c.replace(/[^a-z]/g, '')).join('-'); | 
					
						
							|  |  |  |   let link = baseLink; | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  |   let index = 0; | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |   while (rLinks.has(link)) | 
					
						
							|  |  |  |     link = baseLink + '-' + (++index); | 
					
						
							|  |  |  |   const result = `[${text}](${link})`; | 
					
						
							|  |  |  |   links.set(key, result); | 
					
						
							|  |  |  |   rLinks.set(link, text); | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  |   return result; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @param {Documentation.Member} member | 
					
						
							|  |  |  |  * @return {string} | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function createMemberLink(member) { | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  |   const file = `api/class-${member.clazz.name.toLowerCase()}.md`; | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |   if (member.kind === 'property') | 
					
						
							|  |  |  |     return createLink(file, `${member.clazz.varName}.${member.name}`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (member.kind === 'event') | 
					
						
							|  |  |  |     return createLink(file, `${member.clazz.varName}.on('${member.name}')`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (member.kind === 'method') | 
					
						
							|  |  |  |     return createLink(file, `${member.clazz.varName}.${member.name}(${member.signature})`); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @param {Documentation.Class} clazz | 
					
						
							|  |  |  |  * @return {MarkdownNode[]} | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-01-02 07:17:27 +08:00
										 |  |  | function generateClassToc(clazz) { | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  |   /** @type {MarkdownNode[]} */ | 
					
						
							|  |  |  |   const result = []; | 
					
						
							|  |  |  |   for (const member of clazz.membersArray) { | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  |     result.push({ | 
					
						
							|  |  |  |       type: 'li', | 
					
						
							|  |  |  |       liType: 'default', | 
					
						
							|  |  |  |       text: createMemberLink(member) | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-12-30 09:35:01 +08:00
										 |  |  |   } | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-12-31 10:04:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08: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 09:38:00 +08:00
										 |  |  |     children = spec.slice(1).map(s => md.clone(s)); | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 09:38:00 +08:00
										 |  |  |   /** @type {MarkdownNode} */ | 
					
						
							| 
									
										
										
										
											2020-12-28 23:03:09 +08: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, '\\)'); | 
					
						
							|  |  |  | } |