| 
									
										
										
										
											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-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-03 05:50:10 +08:00
										 |  |  | const os = require('os'); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | const Source = require('./Source'); | 
					
						
							| 
									
										
										
										
											2020-04-24 10:52:06 +08:00
										 |  |  | const Message = require('./Message'); | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  | const { parseMd, renderMd, parseArgument } = require('./../parse_md'); | 
					
						
							| 
									
										
										
										
											2020-12-04 01:11:48 +08:00
										 |  |  | const { spawnSync } = require('child_process'); | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  | const preprocessor = require('./preprocessor'); | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08: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-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() { | 
					
						
							|  |  |  |   const startTime = Date.now(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08: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]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |   /** @type {!Array<!Message>} */ | 
					
						
							|  |  |  |   const messages = []; | 
					
						
							|  |  |  |   let changedFiles = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-03 05:50:10 +08:00
										 |  |  |   // Produce api.md
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     const comment = '<!-- THIS FILE IS NOW GENERATED -->'; | 
					
						
							|  |  |  |     const header = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-header.md')).toString(); | 
					
						
							|  |  |  |     const body = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-body.md')).toString(); | 
					
						
							|  |  |  |     const footer = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-footer.md')).toString(); | 
					
						
							| 
									
										
										
										
											2020-12-04 08:02:34 +08:00
										 |  |  |     let params = fs.readFileSync(path.join(PROJECT_DIR, 'docs-src', 'api-params.md')).toString(); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:28:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |     const paramsMap = new Map(); | 
					
						
							|  |  |  |     for (const node of parseMd(params)) { | 
					
						
							|  |  |  |       if (node.h2.endsWith('-list')) { | 
					
						
							|  |  |  |         node.children = node.children.map(child => paramsMap.get(child.li)); | 
					
						
							|  |  |  |         paramsMap.set('%%-' + node.h2 + '-%%', node); | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (node.children[1]) | 
					
						
							|  |  |  |         node.children[0].li += ' ' + node.children[1].text; | 
					
						
							|  |  |  |       paramsMap.set('%%-' + node.h2 + '-%%', node.children[0]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  |     // Generate signatures
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |       const nodes = parseMd(body); | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  |       const signatures = new Map(); | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |       for (const clazz of nodes) { | 
					
						
							|  |  |  |         clazz.h3 = clazz.h1; | 
					
						
							|  |  |  |         clazz.h1 = undefined; | 
					
						
							|  |  |  |         for (const member of clazz.children) { | 
					
						
							|  |  |  |           if (!member.h2) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |           member.h4 = member.h2; | 
					
						
							|  |  |  |           member.h2 = undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           let match = member.h4.match(/(event|method|namespace): (JS|CDP|[A-Z])([^.]+)\.(.*)/); | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |           if (!match) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |           if (match[1] === 'event') { | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |             member.h4 = `${match[2].toLowerCase() + match[3]}.on('${match[4]}')`; | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |             continue; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (match[1] === 'method') { | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |             const args = []; | 
					
						
							|  |  |  |             const argChildren = []; | 
					
						
							|  |  |  |             const nonArgChildren = []; | 
					
						
							|  |  |  |             const optionsContainer = []; | 
					
						
							|  |  |  |             for (const item of member.children) { | 
					
						
							|  |  |  |               if (!item.h3) { | 
					
						
							|  |  |  |                 nonArgChildren.push(item); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               if (item.h3.startsWith('param:')) { | 
					
						
							|  |  |  |                 if (item.h3.includes('=')) { | 
					
						
							|  |  |  |                   const [name, key] = item.h3.split(' = '); | 
					
						
							|  |  |  |                   item.h3 = name; | 
					
						
							|  |  |  |                   const template = paramsMap.get(key); | 
					
						
							|  |  |  |                   if (!template) | 
					
						
							|  |  |  |                     throw new Error('Bad template: ' + kkey); | 
					
						
							|  |  |  |                   args.push(parseArgument(template.li)); | 
					
						
							|  |  |  |                   argChildren.push(template); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                   const param = item.children[0]; | 
					
						
							|  |  |  |                   if (item.children[1]) | 
					
						
							|  |  |  |                     param.li += ' ' + item.children[1].text; | 
					
						
							|  |  |  |                   args.push(parseArgument(param.li)); | 
					
						
							|  |  |  |                   argChildren.push(param); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               if (item.h3.startsWith('option:')) { | 
					
						
							|  |  |  |                 let optionsNode = optionsContainer[0]; | 
					
						
							|  |  |  |                 if (!optionsNode) { | 
					
						
							|  |  |  |                   optionsNode = { | 
					
						
							|  |  |  |                     li: '`options` <[Object]>', | 
					
						
							|  |  |  |                     liType: 'default', | 
					
						
							|  |  |  |                     children: [], | 
					
						
							|  |  |  |                   }; | 
					
						
							|  |  |  |                   optionsContainer.push(optionsNode); | 
					
						
							|  |  |  |                   args.push(parseArgument(optionsNode.li)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (item.h3.includes('=')) { | 
					
						
							|  |  |  |                   const [name, key] = item.h3.split(' = '); | 
					
						
							|  |  |  |                   const template = paramsMap.get(key); | 
					
						
							|  |  |  |                   if (!template) | 
					
						
							|  |  |  |                     throw new Error('Bad template: ' + key); | 
					
						
							|  |  |  |                   if (item.h3.includes('-inline-')) { | 
					
						
							|  |  |  |                     optionsNode.children.push(...template.children); | 
					
						
							|  |  |  |                   } else { | 
					
						
							|  |  |  |                     item.h3 = name; | 
					
						
							|  |  |  |                     optionsNode.children.push(template); | 
					
						
							|  |  |  |                   } | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                   const param = item.children[0]; | 
					
						
							|  |  |  |                   if (item.children[1]) | 
					
						
							|  |  |  |                     param.li += ' ' + item.children[1].text; | 
					
						
							|  |  |  |                   optionsNode.children.push(param); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             member.children = [...argChildren, ...optionsContainer, ...nonArgChildren]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const tokens = []; | 
					
						
							|  |  |  |             let hasOptional = false; | 
					
						
							|  |  |  |             for (const arg of args) { | 
					
						
							|  |  |  |               const optional = arg.name === 'options' || arg.text.includes('Optional'); | 
					
						
							|  |  |  |               if (tokens.length) { | 
					
						
							|  |  |  |                 if (optional && !hasOptional) | 
					
						
							|  |  |  |                   tokens.push(`[, ${arg.name}`); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                   tokens.push(`, ${arg.name}`); | 
					
						
							|  |  |  |               } else { | 
					
						
							|  |  |  |                 if (optional && !hasOptional) | 
					
						
							|  |  |  |                   tokens.push(`[${arg.name}`); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                   tokens.push(`${arg.name}`); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               hasOptional = hasOptional || optional; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (hasOptional) | 
					
						
							|  |  |  |               tokens.push(']'); | 
					
						
							|  |  |  |             const signature = tokens.join(''); | 
					
						
							|  |  |  |             const methodName = `${match[2].toLowerCase() + match[3]}.${match[4]}`; | 
					
						
							|  |  |  |             signatures.set(methodName, signature); | 
					
						
							|  |  |  |             member.h4 = `${methodName}(${signature})`; | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (match[1] === 'namespace') { | 
					
						
							| 
									
										
										
										
											2020-12-05 10:05:35 +08:00
										 |  |  |             member.h4 = `${match[2].toLowerCase() + match[3]}.${match[4]}`; | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08:00
										 |  |  |             continue; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-04 14:28:11 +08:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-12-05 01:03:33 +08:00
										 |  |  |       api.setText([comment, header, renderMd(nodes), footer].join('\n')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Generate links
 | 
					
						
							|  |  |  |       preprocessor.generateLinks(api, signatures, messages); | 
					
						
							| 
									
										
										
										
											2020-12-04 14:28:11 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-03 05:50:10 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  |   // Documentation checks.
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  |     const browserVersions = await getBrowserVersions(); | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  |     messages.push(...(await preprocessor.runCommands(mdSources, { | 
					
						
							|  |  |  |       libversion: VERSION, | 
					
						
							|  |  |  |       chromiumVersion: browserVersions.chromium, | 
					
						
							|  |  |  |       firefoxVersion: browserVersions.firefox, | 
					
						
							|  |  |  |     }))); | 
					
						
							| 
									
										
										
										
											2020-07-23 02:03:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-05 04:19:48 +08:00
										 |  |  |     messages.push(...preprocessor.autocorrectInvalidLinks(PROJECT_DIR, mdSources, getRepositoryFiles())); | 
					
						
							|  |  |  |     for (const source of mdSources.filter(source => source.hasUpdatedText())) | 
					
						
							|  |  |  |       messages.push(Message.warning(`WARN: updated ${source.projectPath()}`)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const browser = await playwright.chromium.launch(); | 
					
						
							|  |  |  |     const page = await browser.newPage(); | 
					
						
							|  |  |  |     const checkPublicAPI = require('./check_public_api'); | 
					
						
							|  |  |  |     const jsSources = await Source.readdir(path.join(PROJECT_DIR, 'src', 'client'), '', []); | 
					
						
							|  |  |  |     messages.push(...await checkPublicAPI(page, [api], jsSources)); | 
					
						
							|  |  |  |     await browser.close(); | 
					
						
							| 
									
										
										
										
											2019-11-19 10:18:28 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (const source of mdSources) { | 
					
						
							|  |  |  |       if (!source.hasUpdatedText()) | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       await source.save(); | 
					
						
							|  |  |  |       changedFiles = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Report results.
 | 
					
						
							|  |  |  |   const errors = messages.filter(message => message.type === 'error'); | 
					
						
							|  |  |  |   if (errors.length) { | 
					
						
							|  |  |  |     console.log('DocLint Failures:'); | 
					
						
							|  |  |  |     for (let i = 0; i < errors.length; ++i) { | 
					
						
							|  |  |  |       let error = errors[i].text; | 
					
						
							|  |  |  |       error = error.split('\n').join('\n      '); | 
					
						
							|  |  |  |       console.log(`  ${i + 1}) ${RED_COLOR}${error}${RESET_COLOR}`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   const warnings = messages.filter(message => message.type === 'warning'); | 
					
						
							|  |  |  |   if (warnings.length) { | 
					
						
							|  |  |  |     console.log('DocLint Warnings:'); | 
					
						
							|  |  |  |     for (let i = 0; i < warnings.length; ++i) { | 
					
						
							|  |  |  |       let warning = warnings[i].text; | 
					
						
							|  |  |  |       warning = warning.split('\n').join('\n      '); | 
					
						
							|  |  |  |       console.log(`  ${i + 1}) ${YELLOW_COLOR}${warning}${RESET_COLOR}`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   let clearExit = messages.length === 0; | 
					
						
							|  |  |  |   if (changedFiles) { | 
					
						
							|  |  |  |     if (clearExit) | 
					
						
							|  |  |  |       console.log(`${YELLOW_COLOR}Some files were updated.${RESET_COLOR}`); | 
					
						
							|  |  |  |     clearExit = false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   console.log(`${errors.length} failures, ${warnings.length} warnings.`); | 
					
						
							|  |  |  |   const runningTime = Date.now() - startTime; | 
					
						
							|  |  |  |   console.log(`DocLint Finished in ${runningTime / 1000} seconds`); | 
					
						
							| 
									
										
										
										
											2020-12-05 04:19:48 +08:00
										 |  |  |   process.exit(clearExit ? 0 : 1); | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							|  |  |  |   const [chromium, firefox] = await Promise.all([ | 
					
						
							|  |  |  |     getChromeVersion(), | 
					
						
							|  |  |  |     getFirefoxVersion(), | 
					
						
							|  |  |  |   ]) | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  |   return { | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  |     chromium, | 
					
						
							|  |  |  |     firefox, | 
					
						
							| 
									
										
										
										
											2020-02-14 10:26:38 +08:00
										 |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | async function getChromeVersion() { | 
					
						
							|  |  |  |   if (os.platform() === 'win32' || os.platform() === 'cygwin') { | 
					
						
							|  |  |  |     const browser = await playwright.chromium.launch(); | 
					
						
							|  |  |  |     const page = await browser.newPage(); | 
					
						
							|  |  |  |     const userAgent = await page.evaluate('navigator.userAgent'); | 
					
						
							|  |  |  |     const [type] = userAgent.split(' ').filter(str => str.includes('Chrome')); | 
					
						
							|  |  |  |     await browser.close(); | 
					
						
							|  |  |  |     return type.split('/')[1]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   const version = spawnSync(playwright.chromium.executablePath(), ['--version'], undefined).stdout.toString(); | 
					
						
							|  |  |  |   return version.trim().split(' ').pop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 10:52:06 +08:00
										 |  |  | function getRepositoryFiles() { | 
					
						
							|  |  |  |   const out = spawnSync('git', ['ls-files'], {cwd: PROJECT_DIR}); | 
					
						
							| 
									
										
										
										
											2020-12-05 03:09:20 +08: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 10:52:06 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 05:03:01 +08:00
										 |  |  | async function getFirefoxVersion() { | 
					
						
							|  |  |  |   const isWin = os.platform() === 'win32' || os.platform() === 'cygwin'; | 
					
						
							|  |  |  |   const out = spawnSync(playwright.firefox.executablePath(), [isWin ? '/version' : '--version'], undefined); | 
					
						
							|  |  |  |   const version = out.stdout.toString(); | 
					
						
							|  |  |  |   return version.trim().split(' ').pop(); | 
					
						
							| 
									
										
										
										
											2020-04-16 02:29:59 +08:00
										 |  |  | } |