128 lines
3.3 KiB
JavaScript
128 lines
3.3 KiB
JavaScript
|
const createCustomError = require('../utils/createCustomError');
|
||
|
const generate = require('../definition-syntax/generate');
|
||
|
const defaultLoc = { offset: 0, line: 1, column: 1 };
|
||
|
|
||
|
function locateMismatch(matchResult, node) {
|
||
|
const tokens = matchResult.tokens;
|
||
|
const longestMatch = matchResult.longestMatch;
|
||
|
const mismatchNode = longestMatch < tokens.length ? tokens[longestMatch].node || null : null;
|
||
|
const badNode = mismatchNode !== node ? mismatchNode : null;
|
||
|
let mismatchOffset = 0;
|
||
|
let mismatchLength = 0;
|
||
|
let entries = 0;
|
||
|
let css = '';
|
||
|
let start;
|
||
|
let end;
|
||
|
|
||
|
for (let i = 0; i < tokens.length; i++) {
|
||
|
const token = tokens[i].value;
|
||
|
|
||
|
if (i === longestMatch) {
|
||
|
mismatchLength = token.length;
|
||
|
mismatchOffset = css.length;
|
||
|
}
|
||
|
|
||
|
if (badNode !== null && tokens[i].node === badNode) {
|
||
|
if (i <= longestMatch) {
|
||
|
entries++;
|
||
|
} else {
|
||
|
entries = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
css += token;
|
||
|
}
|
||
|
|
||
|
if (longestMatch === tokens.length || entries > 1) { // last
|
||
|
start = fromLoc(badNode || node, 'end') || buildLoc(defaultLoc, css);
|
||
|
end = buildLoc(start);
|
||
|
} else {
|
||
|
start = fromLoc(badNode, 'start') ||
|
||
|
buildLoc(fromLoc(node, 'start') || defaultLoc, css.slice(0, mismatchOffset));
|
||
|
end = fromLoc(badNode, 'end') ||
|
||
|
buildLoc(start, css.substr(mismatchOffset, mismatchLength));
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
css,
|
||
|
mismatchOffset,
|
||
|
mismatchLength,
|
||
|
start,
|
||
|
end
|
||
|
};
|
||
|
}
|
||
|
|
||
|
function fromLoc(node, point) {
|
||
|
const value = node && node.loc && node.loc[point];
|
||
|
|
||
|
if (value) {
|
||
|
return 'line' in value ? buildLoc(value) : value;
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
function buildLoc({ offset, line, column }, extra) {
|
||
|
const loc = {
|
||
|
offset,
|
||
|
line,
|
||
|
column
|
||
|
};
|
||
|
|
||
|
if (extra) {
|
||
|
const lines = extra.split(/\n|\r\n?|\f/);
|
||
|
|
||
|
loc.offset += extra.length;
|
||
|
loc.line += lines.length - 1;
|
||
|
loc.column = lines.length === 1 ? loc.column + extra.length : lines.pop().length + 1;
|
||
|
}
|
||
|
|
||
|
return loc;
|
||
|
}
|
||
|
|
||
|
const SyntaxReferenceError = function(type, referenceName) {
|
||
|
const error = createCustomError(
|
||
|
'SyntaxReferenceError',
|
||
|
type + (referenceName ? ' `' + referenceName + '`' : '')
|
||
|
);
|
||
|
|
||
|
error.reference = referenceName;
|
||
|
|
||
|
return error;
|
||
|
};
|
||
|
|
||
|
const SyntaxMatchError = function(message, syntax, node, matchResult) {
|
||
|
const error = createCustomError('SyntaxMatchError', message);
|
||
|
const {
|
||
|
css,
|
||
|
mismatchOffset,
|
||
|
mismatchLength,
|
||
|
start,
|
||
|
end
|
||
|
} = locateMismatch(matchResult, node);
|
||
|
|
||
|
error.rawMessage = message;
|
||
|
error.syntax = syntax ? generate(syntax) : '<generic>';
|
||
|
error.css = css;
|
||
|
error.mismatchOffset = mismatchOffset;
|
||
|
error.mismatchLength = mismatchLength;
|
||
|
error.message = message + '\n' +
|
||
|
' syntax: ' + error.syntax + '\n' +
|
||
|
' value: ' + (css || '<empty string>') + '\n' +
|
||
|
' --------' + new Array(error.mismatchOffset + 1).join('-') + '^';
|
||
|
|
||
|
Object.assign(error, start);
|
||
|
error.loc = {
|
||
|
source: (node && node.loc && node.loc.source) || '<unknown>',
|
||
|
start,
|
||
|
end
|
||
|
};
|
||
|
|
||
|
return error;
|
||
|
};
|
||
|
|
||
|
module.exports = {
|
||
|
SyntaxReferenceError,
|
||
|
SyntaxMatchError
|
||
|
};
|