For syntax highlighting, we want to iterate over all of the captures in order, and don't care about grouping the captures by pattern.
134 lines
3.8 KiB
JavaScript
134 lines
3.8 KiB
JavaScript
const {assert} = require('chai');
|
|
let Parser, JavaScript;
|
|
|
|
describe("Query", () => {
|
|
let parser, tree, query;
|
|
|
|
before(async () =>
|
|
({Parser, JavaScript} = await require('./helper'))
|
|
);
|
|
|
|
beforeEach(() => {
|
|
parser = new Parser().setLanguage(JavaScript);
|
|
});
|
|
|
|
afterEach(() => {
|
|
parser.delete();
|
|
if (tree) tree.delete();
|
|
if (query) query.delete();
|
|
});
|
|
|
|
describe('construction', () => {
|
|
it('throws an error on invalid syntax', () => {
|
|
assert.throws(() => {
|
|
JavaScript.query("(function_declaration wat)")
|
|
}, "Bad syntax at offset 22: \'wat)\'...");
|
|
assert.throws(() => {
|
|
JavaScript.query("(non_existent)")
|
|
}, "Bad node name 'non_existent'");
|
|
assert.throws(() => {
|
|
JavaScript.query("(a)")
|
|
}, "Bad node name 'a'");
|
|
assert.throws(() => {
|
|
JavaScript.query("(function_declaration non_existent:(identifier))")
|
|
}, "Bad field name 'non_existent'");
|
|
});
|
|
});
|
|
|
|
describe('.matches', () => {
|
|
it('returns all of the matches for the given query', () => {
|
|
tree = parser.parse("function one() { two(); function three() {} }");
|
|
query = JavaScript.query(`
|
|
(function_declaration name:(identifier) @fn-def)
|
|
(call_expression function:(identifier) @fn-ref)
|
|
`);
|
|
const matches = query.matches(tree.rootNode);
|
|
assert.deepEqual(
|
|
formatMatches(matches),
|
|
[
|
|
{pattern: 0, captures: [{name: 'fn-def', text: 'one'}]},
|
|
{pattern: 1, captures: [{name: 'fn-ref', text: 'two'}]},
|
|
{pattern: 0, captures: [{name: 'fn-def', text: 'three'}]},
|
|
]
|
|
);
|
|
});
|
|
|
|
it('can search in a specified ranges', () => {
|
|
tree = parser.parse("[a, b,\nc, d,\ne, f,\ng, h]");
|
|
query = JavaScript.query('(identifier) @element');
|
|
const matches = query.matches(
|
|
tree.rootNode,
|
|
{row: 1, column: 1},
|
|
{row: 3, column: 1}
|
|
);
|
|
assert.deepEqual(
|
|
formatMatches(matches),
|
|
[
|
|
{pattern: 0, captures: [{name: 'element', text: 'd'}]},
|
|
{pattern: 0, captures: [{name: 'element', text: 'e'}]},
|
|
{pattern: 0, captures: [{name: 'element', text: 'f'}]},
|
|
{pattern: 0, captures: [{name: 'element', text: 'g'}]},
|
|
]
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('.captures', () => {
|
|
it('returns all of the captures for the given query, in order', () => {
|
|
tree = parser.parse(`
|
|
a({
|
|
bc: function de() {
|
|
const fg = function hi() {}
|
|
},
|
|
jk: function lm() {
|
|
const no = function pq() {}
|
|
},
|
|
});
|
|
`);
|
|
query = JavaScript.query(`
|
|
(pair
|
|
key: * @method.def
|
|
(function
|
|
name: (identifier) @method.alias))
|
|
|
|
(variable_declarator
|
|
name: * @function.def
|
|
value: (function
|
|
name: (identifier) @function.alias))
|
|
|
|
":" @delimiter
|
|
"=" @operator
|
|
`);
|
|
|
|
const captures = query.captures(tree.rootNode);
|
|
assert.deepEqual(
|
|
formatCaptures(captures),
|
|
[
|
|
{name: "method.def", text: "bc"},
|
|
{name: "delimiter", text: ":"},
|
|
{name: "method.alias", text: "de"},
|
|
{name: "function.def", text: "fg"},
|
|
{name: "operator", text: "="},
|
|
{name: "function.alias", text: "hi"},
|
|
{name: "method.def", text: "jk"},
|
|
{name: "delimiter", text: ":"},
|
|
{name: "method.alias", text: "lm"},
|
|
{name: "function.def", text: "no"},
|
|
{name: "operator", text: "="},
|
|
{name: "function.alias", text: "pq"},
|
|
]
|
|
);
|
|
});
|
|
});
|
|
});
|
|
|
|
function formatMatches(matches) {
|
|
return matches.map(({pattern, captures}) => ({
|
|
pattern,
|
|
captures: formatCaptures(captures)
|
|
}))
|
|
}
|
|
|
|
function formatCaptures(captures) {
|
|
return captures.map(({name, node}) => ({ name, text: node.text }))
|
|
}
|