2023-07-24 23:27:44 +03:00
|
|
|
#!/usr/bin/env node
|
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* 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 debug = require('debug')
|
|
|
|
const fs = require('fs');
|
|
|
|
const path = require('path');
|
|
|
|
const { parseApi } = require('../api_parser');
|
|
|
|
const md = require('../../markdown');
|
|
|
|
const { ESLint } = require('eslint')
|
|
|
|
const child_process = require('child_process');
|
|
|
|
const os = require('os');
|
|
|
|
const actions = require('@actions/core')
|
2023-10-05 05:48:26 +03:00
|
|
|
const { codeFrameColumns } = require('@babel/code-frame');
|
2023-07-24 23:27:44 +03:00
|
|
|
|
|
|
|
/** @typedef {import('../documentation').Type} Type */
|
|
|
|
/** @typedef {import('../../markdown').MarkdownNode} MarkdownNode */
|
|
|
|
|
|
|
|
const PROJECT_DIR = path.join(__dirname, '..', '..', '..');
|
|
|
|
|
|
|
|
function getAllMarkdownFiles(dirPath, filePaths = []) {
|
|
|
|
for (const entry of fs.readdirSync(dirPath, { withFileTypes: true })) {
|
|
|
|
if (entry.isFile() && entry.name.toLowerCase().endsWith('.md'))
|
|
|
|
filePaths.push(path.join(dirPath, entry.name));
|
|
|
|
else if (entry.isDirectory())
|
|
|
|
getAllMarkdownFiles(path.join(dirPath, entry.name), filePaths);
|
|
|
|
}
|
|
|
|
return filePaths;
|
|
|
|
}
|
|
|
|
|
|
|
|
const run = async () => {
|
|
|
|
const documentationRoot = path.join(PROJECT_DIR, 'docs', 'src');
|
|
|
|
const lintingServiceFactory = new LintingServiceFactory();
|
|
|
|
let documentation = parseApi(path.join(documentationRoot, 'api'));
|
|
|
|
|
|
|
|
/** @type {CodeSnippet[]} */
|
|
|
|
const codeSnippets = [];
|
|
|
|
for (const filePath of getAllMarkdownFiles(documentationRoot)) {
|
|
|
|
const data = fs.readFileSync(filePath, 'utf-8');
|
|
|
|
let rootNode = md.parse(data);
|
|
|
|
// Renders links.
|
|
|
|
documentation.renderLinksInNodes(rootNode);
|
|
|
|
documentation.generateSourceCodeComments();
|
|
|
|
md.visitAll(rootNode, node => {
|
|
|
|
if (node.type !== 'code')
|
|
|
|
return;
|
|
|
|
const codeLang = node.codeLang.split(' ')[0];
|
|
|
|
const code = node.lines.join('\n');
|
|
|
|
codeSnippets.push({
|
|
|
|
filePath,
|
|
|
|
codeLang,
|
|
|
|
code,
|
|
|
|
})
|
|
|
|
});
|
|
|
|
}
|
|
|
|
await lintingServiceFactory.lintAndReport(codeSnippets);
|
|
|
|
const { hasErrors } = lintingServiceFactory.reportMetrics();
|
|
|
|
if (hasErrors)
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @typedef {{ codeLang: string, code: string, filePath: string }} CodeSnippet */
|
|
|
|
/** @typedef {{ status: 'ok' | 'updated' | 'error' | 'unsupported', error?: string }} LintResult */
|
|
|
|
|
|
|
|
class LintingService {
|
|
|
|
/**
|
|
|
|
* @param {string} codeLang
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
supports(codeLang) {
|
|
|
|
throw new Error('supports() is not implemented');
|
|
|
|
}
|
|
|
|
|
|
|
|
async _writeTempSnippetsFile(snippets) {
|
|
|
|
const tempFile = path.join(os.tmpdir(), `snippet-${Date.now()}.json`);
|
|
|
|
await fs.promises.writeFile(tempFile, JSON.stringify(snippets, undefined, 2));
|
|
|
|
return tempFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} command
|
|
|
|
* @param {string[]} args
|
|
|
|
* @param {CodeSnippet[]} snippets
|
|
|
|
* @param {string} cwd
|
|
|
|
* @returns {Promise<LintResult[]>}
|
|
|
|
*/
|
|
|
|
async spawnAsync(command, args, snippets, cwd) {
|
|
|
|
const tempFile = await this._writeTempSnippetsFile(snippets);
|
|
|
|
return await new Promise((fulfill, reject) => {
|
|
|
|
const child = child_process.spawn(command, [...args, tempFile], { cwd });
|
|
|
|
let stdout = '';
|
|
|
|
child.on('error', reject);
|
|
|
|
child.stdout.on('data', data => stdout += data.toString());
|
|
|
|
child.stderr.pipe(process.stderr);
|
|
|
|
child.on('exit', code => {
|
|
|
|
if (code)
|
|
|
|
reject(new Error(`${command} exited with code ${code}`));
|
|
|
|
else
|
|
|
|
fulfill(JSON.parse(stdout));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {CodeSnippet[]} snippets
|
|
|
|
* @returns {Promise<LintResult[]>}
|
|
|
|
*/
|
|
|
|
async lint(snippets) {
|
|
|
|
throw new Error('lint() is not implemented');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class JSLintingService extends LintingService {
|
|
|
|
_knownBadSnippets = [
|
|
|
|
'mount(',
|
|
|
|
'render(',
|
|
|
|
'vue-router',
|
|
|
|
'experimental-ct',
|
|
|
|
];
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.eslint = new ESLint({
|
|
|
|
overrideConfigFile: path.join(PROJECT_DIR, '.eslintrc.js'),
|
|
|
|
useEslintrc: false,
|
|
|
|
overrideConfig: {
|
2023-10-05 05:48:26 +03:00
|
|
|
plugins: ['react'],
|
|
|
|
settings: {
|
|
|
|
react: { version: 'detect', }
|
|
|
|
},
|
|
|
|
extends: [
|
|
|
|
'plugin:react/recommended',
|
|
|
|
],
|
2023-07-24 23:27:44 +03:00
|
|
|
rules: {
|
|
|
|
'notice/notice': 'off',
|
|
|
|
'@typescript-eslint/no-unused-vars': 'off',
|
2023-08-02 12:23:47 +03:00
|
|
|
'max-len': ['error', { code: 100 }],
|
2023-07-24 23:27:44 +03:00
|
|
|
},
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
supports(codeLang) {
|
|
|
|
return codeLang === 'js' || codeLang === 'ts';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {CodeSnippet} snippet
|
|
|
|
* @returns {Promise<LintResult>}
|
|
|
|
*/
|
|
|
|
async _lintSnippet(snippet) {
|
|
|
|
if (this._knownBadSnippets.some(s => snippet.code.includes(s)))
|
|
|
|
return { status: 'ok' };
|
|
|
|
const results = await this.eslint.lintText(snippet.code);
|
|
|
|
if (!results || !results.length || !results[0].messages.length)
|
|
|
|
return { status: 'ok' };
|
2023-10-05 05:48:26 +03:00
|
|
|
const result = results[0];
|
|
|
|
const error = result.source ? results[0].messages[0].message + '\n\n' + codeFrameColumns(result.source, { start: result.messages[0] }, { highlightCode: true }) : results[0].messages[0].message;
|
|
|
|
return { status: 'error', error };
|
2023-07-24 23:27:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {CodeSnippet[]} snippets
|
|
|
|
* @returns {Promise<LintResult[]>}
|
|
|
|
*/
|
|
|
|
async lint(snippets) {
|
|
|
|
return Promise.all(snippets.map(async snippet => this._lintSnippet(snippet)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class PythonLintingService extends LintingService {
|
|
|
|
supports(codeLang) {
|
|
|
|
return codeLang === 'python' || codeLang === 'py';
|
|
|
|
}
|
|
|
|
|
|
|
|
async lint(snippets) {
|
|
|
|
const result = await this.spawnAsync('python', [path.join(__dirname, 'python', 'main.py')], snippets, path.join(__dirname, 'python'))
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class CSharpLintingService extends LintingService {
|
|
|
|
supports(codeLang) {
|
|
|
|
return codeLang === 'csharp';
|
|
|
|
}
|
|
|
|
|
|
|
|
async lint(snippets) {
|
|
|
|
return await this.spawnAsync('dotnet', ['run', '--project', path.join(__dirname, 'csharp')], snippets, path.join(__dirname, 'csharp'))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class LintingServiceFactory {
|
|
|
|
constructor() {
|
|
|
|
/** @type {LintingService[]} */
|
|
|
|
this.services = [
|
|
|
|
new JSLintingService(),
|
|
|
|
]
|
|
|
|
if (!process.env.NO_EXTERNAL_DEPS) {
|
|
|
|
this.services.push(
|
|
|
|
new PythonLintingService(),
|
|
|
|
new CSharpLintingService(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
this._metrics = {};
|
|
|
|
this._log = debug('linting-service');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {CodeSnippet[]} allSnippets
|
|
|
|
*/
|
|
|
|
async lintAndReport(allSnippets) {
|
|
|
|
/** @type {Record<string, CodeSnippet[]>} */
|
|
|
|
const groupedByLanguage = allSnippets.reduce((acc, snippet) => {
|
|
|
|
if (!acc[snippet.codeLang])
|
|
|
|
acc[snippet.codeLang] = [];
|
|
|
|
acc[snippet.codeLang].push(snippet);
|
|
|
|
return acc;
|
|
|
|
}, {});
|
|
|
|
for (const language in groupedByLanguage) {
|
|
|
|
const service = this.services.find(service => service.supports(language));
|
|
|
|
if (!service) {
|
|
|
|
this._collectMetrics(language, {
|
|
|
|
status: 'unsupported',
|
|
|
|
});
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const languageSnippets = groupedByLanguage[language];
|
|
|
|
const results = await service.lint(languageSnippets);
|
|
|
|
if (results.length !== languageSnippets.length)
|
|
|
|
throw new Error('Linting service returned wrong number of results');
|
|
|
|
|
|
|
|
for (const [{ code, codeLang, filePath }, result] of /** @type {[[CodeSnippet, LintResult]]} */ (results.map((result, index) => [languageSnippets[index], result]))) {
|
|
|
|
const { status, error } = result;
|
|
|
|
this._collectMetrics(codeLang, result);
|
|
|
|
if (status === 'error') {
|
|
|
|
console.log(`${codeLang} linting error!`);
|
|
|
|
console.log(`ERROR: ${error}`);
|
|
|
|
console.log(`File: ${filePath}`);
|
|
|
|
console.log(code);
|
|
|
|
console.log('-'.repeat(80));
|
|
|
|
if (process.env.GITHUB_ACTION)
|
|
|
|
actions.warning(`Error: ${error}\nUnable to lint:\n${code}`, {
|
|
|
|
title: `${codeLang} linting error`,
|
|
|
|
file: filePath,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns {{ hasErrors: boolean }}
|
|
|
|
*/
|
|
|
|
reportMetrics() {
|
|
|
|
console.log('Metrics:');
|
|
|
|
const renderMetric = (metric, name) => {
|
|
|
|
if (!metric[name])
|
|
|
|
return '';
|
|
|
|
return `${name}: ${metric[name]}`;
|
|
|
|
}
|
|
|
|
let hasErrors = false;
|
|
|
|
const languagesOrderedByOk = Object.entries(this._metrics).sort(([langA], [langB]) => {
|
|
|
|
return this._metrics[langB].ok - this._metrics[langA].ok
|
|
|
|
})
|
|
|
|
for (const [language, metrics] of languagesOrderedByOk) {
|
|
|
|
if (metrics.error)
|
|
|
|
hasErrors = true;
|
|
|
|
console.log(` ${language}: ${['ok', 'updated', 'error', 'unsupported'].map(name => renderMetric(metrics, name)).filter(Boolean).join(', ')}`)
|
|
|
|
}
|
|
|
|
return { hasErrors }
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} language
|
|
|
|
* @param {LintResult} result
|
|
|
|
*/
|
|
|
|
_collectMetrics(language, result) {
|
|
|
|
if (!this._metrics[language])
|
|
|
|
this._metrics[language] = { ok: 0, updated: 0, error: 0, unsupported: 0 };
|
|
|
|
this._metrics[language][result.status]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
run().catch(e => {
|
|
|
|
console.error(e);
|
|
|
|
process.exit(1);
|
|
|
|
});
|