-
Notifications
You must be signed in to change notification settings - Fork 30.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
test: add test for error codes doc and impl #21470
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,158 @@ | ||
'use strict'; | ||
|
||
// Flags: --expose-internals | ||
|
||
// Note: this script reads Node.js sources and docs, and expects to be run on | ||
// the very same Node.js version as the source and doc tree is. | ||
|
||
require('../common'); | ||
const assert = require('assert'); | ||
const errors = require('internal/errors').codes; | ||
const fs = require('fs'); | ||
const path = require('path'); | ||
const natives = process.binding('natives'); | ||
|
||
// --report prints only the list of failed checks and does not assert | ||
const report = process.argv[2] === '--report'; | ||
const results = []; | ||
function check(ok, type, name, reason) { | ||
const label = `${type}: ${name} - ${reason}`; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Since the last three argument combined become the label, wouldn't it be best to pass through the label directly? |
||
if (report) { | ||
if (!ok) results.push(label); | ||
} else { | ||
assert.ok(ok, label); | ||
} | ||
} | ||
|
||
const set = { | ||
all: new Set(), | ||
js: new Set(), // Error codes found in natives. | ||
doc: new Set(), // Error codes mentioned in `doc/api/*.md`. | ||
cpp: new Set(), // Error codes found in `src/node_errors.h`. | ||
noargs: new Set(), // Subset of `js` which is constructed without arguments. | ||
// `set.manual` contains errors manually created from js and not defined in | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It is not immediately clear if this comment is a continuation of the previous line or a new section. Maybe uppercase the previous (if not all) line and add a period or insert a line break before? |
||
// internals/errors. That is not scanned now, update this list if required. | ||
manual: new Set(['ERR_HTTP2_ERROR', 'ERR_UNKNOWN_BUILTIN_MODULE']), | ||
// `set.examples` contains errors mentioned in documentation purely for | ||
// demonstration purposes. These errors are not required to be present. | ||
examples: new Set(['ERR_ERROR_1']) | ||
}; | ||
|
||
const root = path.join(__dirname, '..', '..'); | ||
|
||
// File containing error definitions | ||
const jsdata = natives['internal/errors']; | ||
// File containing error documentation | ||
let docdata = fs.readFileSync(path.join(root, 'doc/api/errors.md'), 'utf-8') | ||
docdata = docdata.replace(/## Legacy [\s\S]*/, ''); | ||
// File containing cpp-side errors | ||
const cppdata = fs.readFileSync(path.join(root, 'src/node_errors.h'), 'utf-8'); | ||
// Directory containing documentation | ||
const docdir = path.join(root, 'doc/api/'); | ||
|
||
function addSource(source, type) { | ||
// eslint-disable-next-line node-core/no-unescaped-regexp-dot | ||
const re = /(.)?(ERR_[A-Z0-9_]+)(..)?/g; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If I understand the regexp correct the last part is there to potentially match a JS error called without arguments. If that's the case, could we just change this to: |
||
let match; | ||
while (match = re.exec(source)) { | ||
if (match[1] === '_') continue; // does not start with ERR_ | ||
const name = match[2]; | ||
if (type === 'doc' && set.examples.has(name)) continue; // is an example | ||
set.all.add(name); | ||
set[type].add(name); | ||
if (type === 'js' && match[3] === '()') { | ||
// Is directly called without arguments from js, we will try that | ||
set.noargs.add(name); | ||
} | ||
} | ||
} | ||
|
||
// Add all errors from JS natives | ||
for (const file of Object.keys(natives)) addSource(natives[file], 'js'); | ||
// Add all errors from src/node_errors.h | ||
addSource(cppdata, 'cpp'); | ||
// Add all errors from doc/api/*.md files | ||
for (const file of fs.readdirSync(docdir)) { | ||
if (!file.endsWith('.md')) continue; | ||
let data = fs.readFileSync(path.join(docdir, file), 'utf-8'); | ||
if (file === 'errors.md') data = data.replace(/## Legacy [\s\S]*/, ''); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should |
||
addSource(data, 'doc'); | ||
} | ||
|
||
// Check that we have all js errors | ||
for (const name of set.js) { | ||
if (set.manual.has(name)) continue; | ||
const defined = jsdata.includes(`E('${name}',`); | ||
check(defined, 'js', name, 'missing JS implementation (source)'); | ||
if (defined) { | ||
check(errors[name], 'js', name, 'missing JS implementation (runtime)'); | ||
} | ||
} | ||
|
||
// Check that we can initialize errors called without args | ||
for (const name of set.noargs) { | ||
if (!errors[name]) continue; // Already catched that above | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. catched -> caught |
||
let ok = true; | ||
try { | ||
new errors[name](); | ||
} catch (e) { | ||
ok = false; | ||
} | ||
check(ok, 'init', name, 'failed init without args, but is called with "()"'); | ||
} | ||
|
||
// Check that we have implementation for all errors mentioned in docs. | ||
// C++ does not need that, and JS is already checked above. | ||
for (const name of set.doc) { | ||
const ok = set.manual.has(name) || | ||
name.startsWith('ERR_NAPI_') || // napi errors are created directly | ||
jsdata.includes(`E('${name}',`) || | ||
cppdata.includes(`V(${name}, `); | ||
const reason = docdata.includes(`### ${name}\n`) ? | ||
'documented' : 'mentioned in doc/api/'; | ||
check(ok, 'impl', name, `missing implementation, ${reason}`); | ||
} | ||
|
||
// Check that we have documentation for all errors | ||
for (const name of set.all) { | ||
const ok = docdata.includes(`### ${name}\n`); | ||
check(ok, 'doc', name, 'missing documentation'); | ||
// Check that documentation references are correctly formatted | ||
if (ok) { | ||
const anchor = docdata.includes(`\n\n<a id="${name}"></a>\n### ${name}\n`); | ||
check(anchor, 'doc', name, 'missing anchor or not properly formatted'); | ||
} | ||
} | ||
|
||
// Check that documentation is sorted, formatted, and does not contain dupes | ||
{ | ||
const compare = (a, b) => { | ||
// HTTP2_ should come after HTTP_ | ||
if (a.startsWith('ERR_HTTP_') && b.startsWith('ERR_HTTP2_')) return -1; | ||
if (a.startsWith('ERR_HTTP2_') && b.startsWith('ERR_HTTP_')) return 1; | ||
if (a < b) return -1; | ||
if (a > b) return 1; | ||
return 0; | ||
}; | ||
const re = /\n### (ERR_[A-Z0-9_]+)\n/g; | ||
let match; | ||
let last; | ||
const documented = new Set(); | ||
while (match = re.exec(docdata)) { | ||
const name = match[1]; | ||
if (documented.has(name)) { | ||
check(false, 'doc', name, 'duplicate documentation entry'); | ||
} else { | ||
const sorted = !last || compare(name, last) === 1; | ||
check(sorted, 'doc', name, 'is out of order'); | ||
documented.add(name); | ||
} | ||
last = name; | ||
} | ||
} | ||
|
||
if (report) { | ||
console.log('Report mode'); | ||
console.log(results.join('\n')); | ||
console.log(`There were ${results.length} problems found`); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I suggest to change that to always asserting but checking all entries first instead of asserting on the first failure.
That way the failures are obvious from the assertion and the user does not have to run the test with this flag (and it's not that obvious that this flag must be used by just taking a glimpse at this).