mirror of https://github.com/Wilfred/difftastic/
Update Nix parser and use the crate from crates.io
parent
bd9c98050e
commit
d42797fa7e
@ -1 +0,0 @@
|
||||
../tree-sitter-nix/queries/highlights.scm
|
||||
@ -1 +0,0 @@
|
||||
tree-sitter-nix/src
|
||||
@ -1,18 +0,0 @@
|
||||
root = true
|
||||
|
||||
[*]
|
||||
end_of_line = lf
|
||||
insert_final_newline = true
|
||||
trim_trailing_whitespace = false
|
||||
|
||||
# for testing purposes, the corpus may have trailing whitespace
|
||||
# and may have mixed EOL.
|
||||
# Still want a final newline though, as that makes no semantic difference.
|
||||
[corpus/*]
|
||||
trim_trailing_whitespace = false
|
||||
end_of_line = unset
|
||||
|
||||
[**.{js,json,cc,css}]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
|
||||
@ -1,2 +0,0 @@
|
||||
/src/** linguist-vendored
|
||||
/src/scanner.cc linguist-vendored=false
|
||||
@ -1,33 +0,0 @@
|
||||
name: build
|
||||
|
||||
on:
|
||||
workflow_dispatch:
|
||||
pull_request:
|
||||
push:
|
||||
|
||||
jobs:
|
||||
build:
|
||||
name: build
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Checkout code
|
||||
uses: actions/checkout@v2
|
||||
|
||||
- name: Setup Node
|
||||
uses: actions/setup-node@v2-beta
|
||||
with:
|
||||
node-version: '16'
|
||||
|
||||
- name: Display Node versions
|
||||
run: |
|
||||
node --version
|
||||
npm --version
|
||||
|
||||
- name: Install dependencies
|
||||
run: npm install
|
||||
|
||||
- name: Check generated parser is checked in
|
||||
run: npm run generate && git diff --exit-code src
|
||||
|
||||
- name: Test corpus
|
||||
run: npm test
|
||||
@ -1,3 +0,0 @@
|
||||
build
|
||||
node_modules
|
||||
Cargo.lock
|
||||
@ -1,21 +0,0 @@
|
||||
language: node_js
|
||||
|
||||
sudo: false
|
||||
|
||||
node_js:
|
||||
- node
|
||||
|
||||
os:
|
||||
- linux
|
||||
|
||||
matrix:
|
||||
include:
|
||||
- os: osx
|
||||
osx_image: xcode9.2
|
||||
- os: linux
|
||||
env: CXX=clang++
|
||||
|
||||
branches:
|
||||
only:
|
||||
- master
|
||||
- /^v.*$/
|
||||
@ -1,26 +0,0 @@
|
||||
[package]
|
||||
name = "tree-sitter-nix"
|
||||
description = "nix grammar for the tree-sitter parsing library"
|
||||
version = "0.0.1"
|
||||
keywords = ["incremental", "parsing", "nix"]
|
||||
categories = ["parsing", "text-editors"]
|
||||
repository = "https://github.com/cstrahan/tree-sitter-nix"
|
||||
edition = "2018"
|
||||
license = "MIT"
|
||||
|
||||
build = "bindings/rust/build.rs"
|
||||
include = [
|
||||
"bindings/rust/*",
|
||||
"grammar.js",
|
||||
"queries/*",
|
||||
"src/*",
|
||||
]
|
||||
|
||||
[lib]
|
||||
path = "bindings/rust/lib.rs"
|
||||
|
||||
[dependencies]
|
||||
tree-sitter = "0.20"
|
||||
|
||||
[build-dependencies]
|
||||
cc = "1.0"
|
||||
@ -1,21 +0,0 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2019 Charles Strahan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
@ -1,7 +0,0 @@
|
||||
tree-sitter-nix
|
||||
================
|
||||
|
||||
[](https://travis-ci.org/cstrahan/tree-sitter-nix)
|
||||
[](https://github.com/cstrahan/tree-sitter-nix/actions?query=workflow%3Abuild)
|
||||
|
||||
Nix grammar for [tree-sitter](https://github.com/tree-sitter/tree-sitter).
|
||||
@ -1,19 +0,0 @@
|
||||
{
|
||||
"targets": [
|
||||
{
|
||||
"target_name": "tree_sitter_nix_binding",
|
||||
"include_dirs": [
|
||||
"<!(node -e \"require('nan')\")",
|
||||
"src"
|
||||
],
|
||||
"sources": [
|
||||
"src/parser.c",
|
||||
"bindings/node/binding.cc",
|
||||
"src/scanner.c"
|
||||
],
|
||||
"cflags_c": [
|
||||
"-std=c99",
|
||||
]
|
||||
}
|
||||
]
|
||||
}
|
||||
@ -1,28 +0,0 @@
|
||||
#include "tree_sitter/parser.h"
|
||||
#include <node.h>
|
||||
#include "nan.h"
|
||||
|
||||
using namespace v8;
|
||||
|
||||
extern "C" TSLanguage * tree_sitter_nix();
|
||||
|
||||
namespace {
|
||||
|
||||
NAN_METHOD(New) {}
|
||||
|
||||
void Init(Local<Object> exports, Local<Object> module) {
|
||||
Local<FunctionTemplate> tpl = Nan::New<FunctionTemplate>(New);
|
||||
tpl->SetClassName(Nan::New("Language").ToLocalChecked());
|
||||
tpl->InstanceTemplate()->SetInternalFieldCount(1);
|
||||
|
||||
Local<Function> constructor = Nan::GetFunction(tpl).ToLocalChecked();
|
||||
Local<Object> instance = constructor->NewInstance(Nan::GetCurrentContext()).ToLocalChecked();
|
||||
Nan::SetInternalFieldPointer(instance, 0, tree_sitter_nix());
|
||||
|
||||
Nan::Set(instance, Nan::New("name").ToLocalChecked(), Nan::New("nix").ToLocalChecked());
|
||||
Nan::Set(module, Nan::New("exports").ToLocalChecked(), instance);
|
||||
}
|
||||
|
||||
NODE_MODULE(tree_sitter_nix_binding, Init)
|
||||
|
||||
} // namespace
|
||||
@ -1,19 +0,0 @@
|
||||
try {
|
||||
module.exports = require("../../build/Release/tree_sitter_nix_binding");
|
||||
} catch (error1) {
|
||||
if (error1.code !== 'MODULE_NOT_FOUND') {
|
||||
throw error1;
|
||||
}
|
||||
try {
|
||||
module.exports = require("../../build/Debug/tree_sitter_nix_binding");
|
||||
} catch (error2) {
|
||||
if (error2.code !== 'MODULE_NOT_FOUND') {
|
||||
throw error2;
|
||||
}
|
||||
throw error1
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
module.exports.nodeTypeInfo = require("../../src/node-types.json");
|
||||
} catch (_) {}
|
||||
@ -1,38 +0,0 @@
|
||||
fn main() {
|
||||
let src_dir = std::path::Path::new("src");
|
||||
|
||||
let mut c_config = cc::Build::new();
|
||||
c_config.include(&src_dir);
|
||||
c_config
|
||||
.flag_if_supported("-Wno-unused-parameter")
|
||||
.flag_if_supported("-Wno-unused-but-set-variable")
|
||||
.flag_if_supported("-Wno-trigraphs");
|
||||
let parser_path = src_dir.join("parser.c");
|
||||
c_config.file(&parser_path);
|
||||
|
||||
// If your language uses an external scanner written in C,
|
||||
// then include this block of code:
|
||||
|
||||
let scanner_path = src_dir.join("scanner.c");
|
||||
c_config.file(&scanner_path);
|
||||
println!("cargo:rerun-if-changed={}", scanner_path.to_str().unwrap());
|
||||
|
||||
c_config.compile("parser");
|
||||
println!("cargo:rerun-if-changed={}", parser_path.to_str().unwrap());
|
||||
|
||||
// If your language uses an external scanner written in C++,
|
||||
// then include this block of code:
|
||||
|
||||
/*
|
||||
let mut cpp_config = cc::Build::new();
|
||||
cpp_config.cpp(true);
|
||||
cpp_config.include(&src_dir);
|
||||
cpp_config
|
||||
.flag_if_supported("-Wno-unused-parameter")
|
||||
.flag_if_supported("-Wno-unused-but-set-variable");
|
||||
let scanner_path = src_dir.join("scanner.cc");
|
||||
cpp_config.file(&scanner_path);
|
||||
cpp_config.compile("scanner");
|
||||
println!("cargo:rerun-if-changed={}", scanner_path.to_str().unwrap());
|
||||
*/
|
||||
}
|
||||
@ -1,52 +0,0 @@
|
||||
//! This crate provides nix language support for the [tree-sitter][] parsing library.
|
||||
//!
|
||||
//! Typically, you will use the [language][language func] function to add this language to a
|
||||
//! tree-sitter [Parser][], and then use the parser to parse some code:
|
||||
//!
|
||||
//! ```
|
||||
//! let code = "";
|
||||
//! let mut parser = tree_sitter::Parser::new();
|
||||
//! parser.set_language(tree_sitter_nix::language()).expect("Error loading nix grammar");
|
||||
//! let tree = parser.parse(code, None).unwrap();
|
||||
//! ```
|
||||
//!
|
||||
//! [Language]: https://docs.rs/tree-sitter/*/tree_sitter/struct.Language.html
|
||||
//! [language func]: fn.language.html
|
||||
//! [Parser]: https://docs.rs/tree-sitter/*/tree_sitter/struct.Parser.html
|
||||
//! [tree-sitter]: https://tree-sitter.github.io/
|
||||
|
||||
use tree_sitter::Language;
|
||||
|
||||
extern "C" {
|
||||
fn tree_sitter_nix() -> Language;
|
||||
}
|
||||
|
||||
/// Get the tree-sitter [Language][] for this grammar.
|
||||
///
|
||||
/// [Language]: https://docs.rs/tree-sitter/*/tree_sitter/struct.Language.html
|
||||
pub fn language() -> Language {
|
||||
unsafe { tree_sitter_nix() }
|
||||
}
|
||||
|
||||
/// The content of the [`node-types.json`][] file for this grammar.
|
||||
///
|
||||
/// [`node-types.json`]: https://tree-sitter.github.io/tree-sitter/using-parsers#static-node-types
|
||||
pub const NODE_TYPES: &'static str = include_str!("../../src/node-types.json");
|
||||
|
||||
// Uncomment these to include any queries that this grammar contains
|
||||
|
||||
// pub const HIGHLIGHTS_QUERY: &'static str = include_str!("../../queries/highlights.scm");
|
||||
// pub const INJECTIONS_QUERY: &'static str = include_str!("../../queries/injections.scm");
|
||||
// pub const LOCALS_QUERY: &'static str = include_str!("../../queries/locals.scm");
|
||||
// pub const TAGS_QUERY: &'static str = include_str!("../../queries/tags.scm");
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
#[test]
|
||||
fn test_can_load_grammar() {
|
||||
let mut parser = tree_sitter::Parser::new();
|
||||
parser
|
||||
.set_language(super::language())
|
||||
.expect("Error loading nix language");
|
||||
}
|
||||
}
|
||||
@ -1,528 +0,0 @@
|
||||
====================
|
||||
empty source
|
||||
====================
|
||||
|
||||
---
|
||||
|
||||
(source_code)
|
||||
|
||||
====================
|
||||
comments
|
||||
====================
|
||||
|
||||
# This is a comment.
|
||||
|
||||
/*
|
||||
This is also a comment.
|
||||
*/
|
||||
|
||||
/*This as well, even with two asterisks.**/
|
||||
|
||||
"This is a string."
|
||||
|
||||
---
|
||||
|
||||
(source_code (comment) (comment) (comment) (string_expression (string_fragment)))
|
||||
|
||||
====================
|
||||
identifier
|
||||
====================
|
||||
|
||||
abc
|
||||
|
||||
---
|
||||
|
||||
(source_code (variable_expression (identifier)))
|
||||
|
||||
====================
|
||||
integer_expression
|
||||
====================
|
||||
|
||||
123
|
||||
|
||||
---
|
||||
|
||||
(source_code (integer_expression))
|
||||
|
||||
====================
|
||||
float_expression
|
||||
====================
|
||||
|
||||
123.456
|
||||
|
||||
---
|
||||
|
||||
(source_code (float_expression))
|
||||
|
||||
====================
|
||||
uri
|
||||
====================
|
||||
|
||||
http://foobar.com:80?baz=quux&blarg=etc
|
||||
|
||||
---
|
||||
|
||||
(source_code (uri_expression))
|
||||
|
||||
====================
|
||||
list (empty)
|
||||
====================
|
||||
|
||||
[ ]
|
||||
|
||||
---
|
||||
|
||||
(source_code (list_expression))
|
||||
|
||||
====================
|
||||
list
|
||||
====================
|
||||
|
||||
[ a 1 2.0 ]
|
||||
|
||||
---
|
||||
|
||||
(source_code (list_expression (variable_expression (identifier)) (integer_expression) (float_expression)))
|
||||
|
||||
====================
|
||||
if
|
||||
====================
|
||||
|
||||
if a then b else c
|
||||
|
||||
---
|
||||
|
||||
(source_code (if_expression (variable_expression (identifier)) (variable_expression (identifier)) (variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
assert
|
||||
====================
|
||||
|
||||
assert a; b
|
||||
|
||||
---
|
||||
|
||||
(source_code (assert_expression (variable_expression (identifier)) (variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
with
|
||||
====================
|
||||
|
||||
with a; b
|
||||
|
||||
---
|
||||
|
||||
(source_code (with_expression (variable_expression (identifier)) (variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
let (empty)
|
||||
====================
|
||||
|
||||
let in a
|
||||
|
||||
---
|
||||
|
||||
(source_code (let_expression (variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
let (binding)
|
||||
====================
|
||||
|
||||
let a = b; in c
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath (identifier))
|
||||
(variable_expression (identifier))))
|
||||
(variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
let (binding, comments)
|
||||
====================
|
||||
|
||||
let
|
||||
# foo
|
||||
a = b;
|
||||
# bar
|
||||
x = y;
|
||||
in
|
||||
# baz
|
||||
c
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(comment)
|
||||
(binding_set
|
||||
(binding (attrpath (identifier)) (variable_expression (identifier)))
|
||||
(comment)
|
||||
(binding (attrpath (identifier)) (variable_expression (identifier))))
|
||||
(comment)
|
||||
(variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
let (inherit)
|
||||
====================
|
||||
|
||||
let inherit a; in c
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(binding_set
|
||||
(inherit
|
||||
(inherited_attrs
|
||||
(identifier))))
|
||||
(variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
let (inherit from)
|
||||
====================
|
||||
|
||||
let inherit (a) b "c" ${d}; in 123
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(binding_set
|
||||
(inherit_from
|
||||
(variable_expression (identifier))
|
||||
(inherited_attrs
|
||||
(identifier) (string_expression (string_fragment)) (interpolation (variable_expression (identifier))))))
|
||||
(integer_expression)))
|
||||
|
||||
====================
|
||||
function_expression
|
||||
====================
|
||||
|
||||
a: b
|
||||
|
||||
---
|
||||
|
||||
(source_code (function_expression (identifier) (variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
function_expression (with_expression formals)
|
||||
====================
|
||||
|
||||
a@{ /*1*/ b, /*2*/ c ? 123, /*3*/ ... }: 1.234
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(function_expression
|
||||
(identifier)
|
||||
(formals
|
||||
(comment)
|
||||
(formal (identifier))
|
||||
(comment)
|
||||
(formal (identifier) (integer_expression))
|
||||
(comment)
|
||||
(ellipses))
|
||||
(float_expression)))
|
||||
|
||||
====================
|
||||
rec. attrset (empty)
|
||||
====================
|
||||
|
||||
rec { }
|
||||
|
||||
---
|
||||
|
||||
(source_code (rec_attrset_expression))
|
||||
|
||||
====================
|
||||
let attrset (empty)
|
||||
====================
|
||||
|
||||
let { }
|
||||
|
||||
---
|
||||
|
||||
(source_code (let_attrset_expression))
|
||||
|
||||
====================
|
||||
attrset (empty)
|
||||
====================
|
||||
|
||||
{ }
|
||||
|
||||
---
|
||||
|
||||
(source_code (attrset_expression))
|
||||
|
||||
====================
|
||||
attr set
|
||||
====================
|
||||
|
||||
{ a = 1; ${b} = "quux"; "c" = 3.14; x.y.z = <foo>; }
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(binding_set
|
||||
(binding (attrpath (identifier)) (integer_expression))
|
||||
(binding (attrpath (interpolation (variable_expression (identifier)))) (string_expression (string_fragment)))
|
||||
(binding (attrpath (string_expression (string_fragment))) (float_expression))
|
||||
(binding (attrpath (identifier) (identifier) (identifier)) (spath_expression)))))
|
||||
|
||||
====================
|
||||
select_expression
|
||||
====================
|
||||
|
||||
u.v.${w}."x${y}z"
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(select_expression
|
||||
(variable_expression (identifier))
|
||||
(attrpath
|
||||
(identifier)
|
||||
(interpolation (variable_expression (identifier)))
|
||||
(string_expression
|
||||
(string_fragment)
|
||||
(interpolation (variable_expression (identifier)))
|
||||
(string_fragment)))))
|
||||
|
||||
|
||||
|
||||
====================
|
||||
hasAttr operator
|
||||
====================
|
||||
|
||||
a ? ${b}.c."d"
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(has_attr_expression
|
||||
(variable_expression (identifier))
|
||||
(attrpath
|
||||
(interpolation (variable_expression (identifier)))
|
||||
(identifier)
|
||||
(string_expression (string_fragment)))))
|
||||
|
||||
====================
|
||||
apply_expressionlication
|
||||
====================
|
||||
|
||||
a 1 2.0
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(apply_expression
|
||||
(apply_expression
|
||||
(variable_expression (identifier))
|
||||
(integer_expression))
|
||||
(float_expression)))
|
||||
|
||||
====================
|
||||
path_expression test
|
||||
====================
|
||||
|
||||
a/c${x}c
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(path_fragment)))
|
||||
|
||||
|
||||
====================
|
||||
path_expression1
|
||||
====================
|
||||
|
||||
x/a${x}a.a${123}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(integer_expression))))
|
||||
|
||||
====================
|
||||
path_expression2
|
||||
====================
|
||||
|
||||
x/a${x}a/a${123}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(integer_expression))))
|
||||
|
||||
|
||||
====================
|
||||
path_expression3
|
||||
====================
|
||||
|
||||
/abc
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)))
|
||||
|
||||
====================
|
||||
path_expression4
|
||||
====================
|
||||
|
||||
-123/abc
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)))
|
||||
|
||||
====================
|
||||
path_expression5
|
||||
====================
|
||||
|
||||
dir/${name}.${extension}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))))
|
||||
|
||||
====================
|
||||
path_expression6
|
||||
====================
|
||||
|
||||
|
||||
a/b${c} d/e${f}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(apply_expression
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier))))
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier))))))
|
||||
|
||||
====================
|
||||
hpath_expression1
|
||||
====================
|
||||
|
||||
~/.
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(hpath_expression
|
||||
(path_fragment)))
|
||||
|
||||
|
||||
====================
|
||||
hpath_expression2
|
||||
====================
|
||||
|
||||
~/a${x}a/a${123}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(hpath_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(integer_expression))))
|
||||
|
||||
====================
|
||||
division not mistaken for path_expressions (1)
|
||||
====================
|
||||
|
||||
a/ b
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(binary_expression
|
||||
(variable_expression (identifier))
|
||||
(variable_expression (identifier))))
|
||||
|
||||
====================
|
||||
path_expression craziness - see https://github.com/NixOS/nix/pull/5066#issuecomment-1071918251
|
||||
====================
|
||||
|
||||
a.${foo}/b.${bar}
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(apply_expression
|
||||
(select_expression
|
||||
(variable_expression (identifier))
|
||||
(attrpath
|
||||
(interpolation
|
||||
(variable_expression (identifier)))))
|
||||
(path_expression
|
||||
(path_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier))))))
|
||||
|
||||
====================
|
||||
operators
|
||||
====================
|
||||
|
||||
square 2 + -pi - 42.0
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(binary_expression
|
||||
(binary_expression
|
||||
(apply_expression (variable_expression (identifier)) (integer_expression))
|
||||
(unary_expression (variable_expression (identifier))))
|
||||
(float_expression)))
|
||||
|
||||
====================
|
||||
parens
|
||||
====================
|
||||
|
||||
(123)
|
||||
|
||||
---
|
||||
|
||||
(source_code
|
||||
(parenthesized_expression (integer_expression)))
|
||||
@ -1,268 +0,0 @@
|
||||
================================================================================
|
||||
attrset typing field
|
||||
================================================================================
|
||||
|
||||
{
|
||||
a = "foo";
|
||||
b = 42;
|
||||
typing
|
||||
c = {};
|
||||
d = x: x;
|
||||
}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(integer_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(ERROR
|
||||
(identifier))
|
||||
(attrset_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))))
|
||||
|
||||
================================================================================
|
||||
attrset typing field following string
|
||||
================================================================================
|
||||
|
||||
{
|
||||
typing
|
||||
inputs.nixpkgs.url = "github:nixos/nixpkgs";
|
||||
inputs.nixpkgs-21-05.url = "github:nixos/nixpkgs/nixos-21.05";
|
||||
}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(ERROR
|
||||
(identifier))
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier)
|
||||
(identifier)
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier)
|
||||
(identifier)
|
||||
(identifier))
|
||||
(string_expression (string_fragment))))))
|
||||
|
||||
================================================================================
|
||||
attrset typing attrpath
|
||||
================================================================================
|
||||
|
||||
{
|
||||
a = "foo";
|
||||
b = 42;
|
||||
typing.path_expression
|
||||
c = {};
|
||||
d = x: x;
|
||||
}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(integer_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier)
|
||||
(identifier))
|
||||
(ERROR
|
||||
(identifier))
|
||||
(attrset_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))))
|
||||
|
||||
================================================================================
|
||||
attrset missing value
|
||||
================================================================================
|
||||
|
||||
{
|
||||
a = "foo";
|
||||
b = 42;
|
||||
typing =
|
||||
c = {};
|
||||
d = x: x;
|
||||
}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(integer_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(apply_expression
|
||||
(variable_expression (identifier))
|
||||
(ERROR)
|
||||
(attrset_expression)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))))
|
||||
|
||||
================================================================================
|
||||
bind typing parenthesis
|
||||
================================================================================
|
||||
|
||||
{
|
||||
a = "foo";
|
||||
b = a: 42;
|
||||
typing = (a:
|
||||
c = {};
|
||||
d = x: x;
|
||||
}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(attrset_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(integer_expression)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(ERROR
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier))))
|
||||
(attrset_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))))
|
||||
|
||||
================================================================================
|
||||
let typing field
|
||||
================================================================================
|
||||
|
||||
let
|
||||
a = "foo";
|
||||
b = 42;
|
||||
typing
|
||||
c = {};
|
||||
d = x: x;
|
||||
in {}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(integer_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(ERROR
|
||||
(identifier))
|
||||
(attrset_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))
|
||||
(attrset_expression)))
|
||||
|
||||
================================================================================
|
||||
let missing value
|
||||
================================================================================
|
||||
|
||||
let
|
||||
a = "foo";
|
||||
b = 42;
|
||||
typing =
|
||||
c = {};
|
||||
d = x: x;
|
||||
in
|
||||
a
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(let_expression
|
||||
(binding_set
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(string_expression (string_fragment)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(integer_expression))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(apply_expression
|
||||
(variable_expression (identifier))
|
||||
(ERROR)
|
||||
(attrset_expression)))
|
||||
(binding
|
||||
(attrpath
|
||||
(identifier))
|
||||
(function_expression
|
||||
(identifier)
|
||||
(variable_expression (identifier)))))
|
||||
(variable_expression (identifier))))
|
||||
|
||||
@ -1,168 +0,0 @@
|
||||
================================================================================
|
||||
string
|
||||
================================================================================
|
||||
|
||||
"abcdef"
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(string_expression (string_fragment)))
|
||||
|
||||
================================================================================
|
||||
string (complex)
|
||||
================================================================================
|
||||
|
||||
"
|
||||
${bob} likes crisp $ bills. escape newline \
|
||||
\${don't interpolate here!}"
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(string_expression
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(escape_sequence)
|
||||
(string_fragment)))
|
||||
|
||||
================================================================================
|
||||
indented string
|
||||
================================================================================
|
||||
|
||||
''
|
||||
abc
|
||||
def
|
||||
ghi
|
||||
''
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(indented_string_expression (string_fragment)))
|
||||
|
||||
================================================================================
|
||||
indented string (complex)
|
||||
================================================================================
|
||||
|
||||
''
|
||||
This is just a couple of quotes: '''
|
||||
A lone $ doesn't throw things off.
|
||||
And of course, ''\${this shouldn't be an interpolation}.
|
||||
But ${this} is.
|
||||
This works, too: ''$
|
||||
''
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(indented_string_expression
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(string_fragment)))
|
||||
|
||||
================================================================================
|
||||
string ($)
|
||||
================================================================================
|
||||
|
||||
[
|
||||
"$"
|
||||
"$\n"
|
||||
"${x}"
|
||||
"$${x}"
|
||||
"$$${x}"
|
||||
"$$$${x}"
|
||||
]
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(list_expression
|
||||
(string_expression (string_fragment))
|
||||
(string_expression
|
||||
(string_fragment)
|
||||
(escape_sequence))
|
||||
(string_expression
|
||||
(interpolation
|
||||
(variable_expression (identifier))))
|
||||
(string_expression (string_fragment))
|
||||
(string_expression
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier))))
|
||||
(string_expression (string_fragment))))
|
||||
|
||||
================================================================================
|
||||
indented string ($)
|
||||
================================================================================
|
||||
|
||||
[
|
||||
''$''
|
||||
''$''\n''
|
||||
''${x}''
|
||||
''$${x}''
|
||||
''$$${x}''
|
||||
''$$$${x}''
|
||||
]
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(list_expression
|
||||
(indented_string_expression (string_fragment))
|
||||
(indented_string_expression
|
||||
(string_fragment)
|
||||
(escape_sequence))
|
||||
(indented_string_expression
|
||||
(interpolation
|
||||
(variable_expression (identifier))))
|
||||
(indented_string_expression (string_fragment))
|
||||
(indented_string_expression
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier))))
|
||||
(indented_string_expression (string_fragment))))
|
||||
|
||||
================================================================================
|
||||
unterminated string
|
||||
================================================================================
|
||||
|
||||
"foo ${bar} \n
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(string_expression
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(MISSING """)))
|
||||
|
||||
================================================================================
|
||||
unterminated string escape
|
||||
================================================================================
|
||||
|
||||
"foo ${bar} \n \
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
(source_code
|
||||
(ERROR
|
||||
(string_fragment)
|
||||
(interpolation
|
||||
(variable_expression (identifier)))
|
||||
(string_fragment)
|
||||
(escape_sequence)
|
||||
(string_fragment)
|
||||
(UNEXPECTED '\0')))
|
||||
@ -1,299 +0,0 @@
|
||||
const PREC = {
|
||||
impl: 1,
|
||||
or: 2,
|
||||
and: 3,
|
||||
eq: 4,
|
||||
neq: 4,
|
||||
'<': 5,
|
||||
'>': 5,
|
||||
leq: 5,
|
||||
geq: 5,
|
||||
update: 6,
|
||||
not: 7,
|
||||
'+': 8,
|
||||
'-': 8,
|
||||
'*': 9,
|
||||
'/': 9,
|
||||
concat: 10,
|
||||
'?': 11,
|
||||
negate: 12
|
||||
}
|
||||
|
||||
module.exports = grammar({
|
||||
name: 'nix',
|
||||
|
||||
extras: $ => [
|
||||
/\s/,
|
||||
$.comment,
|
||||
],
|
||||
|
||||
supertypes: $ => [
|
||||
$._expression
|
||||
],
|
||||
|
||||
inline: $ => [
|
||||
],
|
||||
|
||||
externals: $ => [
|
||||
$.string_fragment,
|
||||
$._indented_string_fragment,
|
||||
$._path_start,
|
||||
$.path_fragment,
|
||||
],
|
||||
|
||||
word: $ => $.keyword,
|
||||
|
||||
conflicts: $ => [
|
||||
],
|
||||
|
||||
rules: {
|
||||
source_code: $ => optional(field('expression', $._expression)),
|
||||
_expression: $ => $._expr_function_expression,
|
||||
|
||||
// Keywords go before identifiers to let them take precedence when both are expected.
|
||||
// Workaround before https://github.com/tree-sitter/tree-sitter/pull/246
|
||||
keyword: $ => /if|then|else|let|inherit|in|rec|with|assert/,
|
||||
identifier: $ => /[a-zA-Z_][a-zA-Z0-9_\'\-]*/,
|
||||
|
||||
variable_expression: $ => field('name', $.identifier),
|
||||
integer_expression: $ => /[0-9]+/,
|
||||
float_expression: $ => /(([1-9][0-9]*\.[0-9]*)|(0?\.[0-9]+))([Ee][+-]?[0-9]+)?/,
|
||||
|
||||
path_expression: $=> seq(
|
||||
alias($._path_start, $.path_fragment),
|
||||
repeat(
|
||||
choice(
|
||||
$.path_fragment,
|
||||
alias($._immediate_interpolation, $.interpolation),
|
||||
)
|
||||
),
|
||||
),
|
||||
|
||||
_hpath_start: $ => /\~\/[a-zA-Z0-9\._\-\+\/]+/,
|
||||
hpath_expression: $=> seq(
|
||||
alias($._hpath_start, $.path_fragment),
|
||||
repeat(
|
||||
choice(
|
||||
$.path_fragment,
|
||||
alias($._immediate_interpolation, $.interpolation),
|
||||
)
|
||||
),
|
||||
),
|
||||
|
||||
spath_expression: $ => /<[a-zA-Z0-9\._\-\+]+(\/[a-zA-Z0-9\._\-\+]+)*>/,
|
||||
uri_expression: $ => /[a-zA-Z][a-zA-Z0-9\+\-\.]*:[a-zA-Z0-9%\/\?:@\&=\+\$,\-_\.\!\~\*\']+/,
|
||||
|
||||
_expr_function_expression: $ => choice(
|
||||
$.function_expression,
|
||||
$.assert_expression,
|
||||
$.with_expression,
|
||||
$.let_expression,
|
||||
$._expr_if
|
||||
),
|
||||
|
||||
function_expression: $ => choice(
|
||||
seq(field('universal', $.identifier), ':', field('body', $._expr_function_expression)),
|
||||
seq(field('formals', $.formals), ":", field('body', $._expr_function_expression)),
|
||||
seq(field('formals', $.formals), '@', field('universal', $.identifier), ':', field('body', $._expr_function_expression)),
|
||||
seq(field('universal', $.identifier), '@', field('formals', $.formals), ':', field('body', $._expr_function_expression)),
|
||||
),
|
||||
|
||||
formals: $ => choice(
|
||||
seq('{', '}'),
|
||||
seq('{', commaSep1(field('formal', $.formal)), '}'),
|
||||
seq('{', commaSep1(field('formal', $.formal)), ',', field('ellipses', $.ellipses), '}'),
|
||||
seq('{', field('ellipses', $.ellipses), '}'),
|
||||
),
|
||||
formal: $ => seq(field("name", $.identifier), optional(seq('?', field('default', $._expression)))),
|
||||
ellipses: $ => '...',
|
||||
|
||||
assert_expression: $ => seq('assert', field('condition', $._expression), ';', field('body', $._expr_function_expression)),
|
||||
with_expression: $ => seq('with', field('environment', $._expression), ';', field('body', $._expr_function_expression)),
|
||||
let_expression: $ => seq('let', optional($.binding_set), 'in', field('body', $._expr_function_expression)),
|
||||
|
||||
_expr_if: $ => choice(
|
||||
$.if_expression,
|
||||
$._expr_op
|
||||
),
|
||||
|
||||
if_expression: $ => seq('if', field('condition', $._expression), 'then', field('consequence', $._expression), 'else', field('alternative', $._expression)),
|
||||
|
||||
_expr_op: $ => choice(
|
||||
$.has_attr_expression,
|
||||
$.unary_expression,
|
||||
$.binary_expression,
|
||||
$._expr_apply_expression
|
||||
),
|
||||
|
||||
// I choose to *not* have this among the binary operators because
|
||||
// this is the sole exception that takes an attrpath (instead of expression)
|
||||
// as its right operand.
|
||||
// My gut feeling is that this is:
|
||||
// 1) better in theory, and
|
||||
// 2) will be easier to work with in practice.
|
||||
has_attr_expression: $ => prec(PREC['?'],
|
||||
seq(
|
||||
field('expression', $._expr_op),
|
||||
field('operator', '?'),
|
||||
field('attrpath', $.attrpath)
|
||||
)
|
||||
),
|
||||
|
||||
unary_expression: $ => choice(
|
||||
...[
|
||||
['!', PREC.not],
|
||||
['-', PREC.negate],
|
||||
].map(([operator, precedence]) =>
|
||||
prec(precedence, seq(
|
||||
field('operator', operator),
|
||||
field('argument', $._expr_op)
|
||||
))
|
||||
)
|
||||
),
|
||||
|
||||
binary_expression: $ => choice(
|
||||
// left assoc.
|
||||
...[
|
||||
['==', PREC.eq],
|
||||
['!=', PREC.neq],
|
||||
['<', PREC['<']],
|
||||
['<=', PREC.leq],
|
||||
['>', PREC['>']],
|
||||
['>=', PREC.geq],
|
||||
['&&', PREC.and],
|
||||
['||', PREC.or],
|
||||
['+', PREC['+']],
|
||||
['-', PREC['-']],
|
||||
['*', PREC['*']],
|
||||
['/', PREC['/']],
|
||||
].map(([operator, precedence]) =>
|
||||
prec.left(precedence, seq(
|
||||
field('left', $._expr_op),
|
||||
field('operator', operator),
|
||||
field('right', $._expr_op)
|
||||
))),
|
||||
// right assoc.
|
||||
...[
|
||||
['->', PREC.impl],
|
||||
['//', PREC.update],
|
||||
['++', PREC.concat],
|
||||
].map(([operator, precedence]) =>
|
||||
prec.right(precedence, seq(
|
||||
field('left', $._expr_op),
|
||||
field('operator', operator),
|
||||
field('right', $._expr_op)
|
||||
)))
|
||||
),
|
||||
|
||||
_expr_apply_expression: $ => choice(
|
||||
$.apply_expression,
|
||||
$._expr_select_expression
|
||||
),
|
||||
|
||||
apply_expression: $ => seq(field('function', $._expr_apply_expression), field('argument', $._expr_select_expression)),
|
||||
|
||||
_expr_select_expression: $ => choice(
|
||||
$.select_expression,
|
||||
$._expr_simple
|
||||
),
|
||||
|
||||
select_expression: $ => choice(
|
||||
seq(field('expression', $._expr_simple), '.', field('attrpath', $.attrpath)),
|
||||
seq(field('expression', $._expr_simple), '.', field('attrpath', $.attrpath), 'or', field('default', $._expr_select_expression)),
|
||||
),
|
||||
|
||||
_expr_simple: $ => choice(
|
||||
$.variable_expression,
|
||||
$.integer_expression,
|
||||
$.float_expression,
|
||||
$.string_expression,
|
||||
$.indented_string_expression,
|
||||
$.path_expression,
|
||||
$.hpath_expression,
|
||||
$.spath_expression,
|
||||
$.uri_expression,
|
||||
$.parenthesized_expression,
|
||||
$.attrset_expression,
|
||||
$.let_attrset_expression,
|
||||
$.rec_attrset_expression,
|
||||
$.list_expression
|
||||
),
|
||||
|
||||
parenthesized_expression: $ => seq('(', field('expression', $._expression), ')'),
|
||||
|
||||
attrset_expression: $ => seq('{', optional($.binding_set), '}'),
|
||||
let_attrset_expression: $ => seq('let', '{', optional($.binding_set), '}'),
|
||||
rec_attrset_expression: $ => seq('rec', '{', optional($.binding_set), '}'),
|
||||
|
||||
string_expression: $ => seq(
|
||||
'"',
|
||||
repeat(choice(
|
||||
$.string_fragment,
|
||||
$.interpolation,
|
||||
$.escape_sequence
|
||||
)),
|
||||
'"'
|
||||
),
|
||||
escape_sequence: $ => token.immediate(/\\(.|\s)/), // Can also escape newline.
|
||||
|
||||
indented_string_expression: $ => seq(
|
||||
"''",
|
||||
repeat(choice(
|
||||
alias($._indented_string_fragment, $.string_fragment),
|
||||
$.interpolation,
|
||||
alias($._indented_escape_sequence, $.escape_sequence),
|
||||
)),
|
||||
"''"
|
||||
),
|
||||
_indented_escape_sequence: $ => token.immediate(/'''|''\$|''\\(.|\s)/), // Can also escape newline.
|
||||
|
||||
binding_set: $ => repeat1(field('binding', choice($.binding, $.inherit, $.inherit_from))),
|
||||
binding: $ => seq(field('attrpath', $.attrpath), '=', field('expression', $._expression), ';'),
|
||||
inherit: $ => seq('inherit', field('attrs', $.inherited_attrs), ';'),
|
||||
inherit_from: $ =>
|
||||
seq('inherit', '(', field('expression', $._expression), ')', field('attrs', $.inherited_attrs), ';'),
|
||||
|
||||
attrpath: $ => sep1(field('attr', choice(
|
||||
$.identifier,
|
||||
$.string_expression,
|
||||
$.interpolation,
|
||||
)), "."),
|
||||
|
||||
inherited_attrs: $ => repeat1(field('attr', choice(
|
||||
$.identifier,
|
||||
$.string_expression,
|
||||
$.interpolation,
|
||||
))),
|
||||
|
||||
_immediate_interpolation: $ => seq(token.immediate('${'), field('expression', $._expression), '}'),
|
||||
interpolation: $ => seq('${', field('expression', $._expression), '}'),
|
||||
|
||||
list_expression: $ => seq('[', repeat(field('element', $._expr_select_expression)), ']'),
|
||||
|
||||
comment: $ => token(choice(
|
||||
seq('#', /.*/),
|
||||
seq(
|
||||
'/*',
|
||||
/[^*]*\*+([^/*][^*]*\*+)*/,
|
||||
'/'
|
||||
)
|
||||
)),
|
||||
},
|
||||
});
|
||||
|
||||
|
||||
function sep(rule, separator) {
|
||||
return optional(sep1(rule, separator));
|
||||
}
|
||||
|
||||
function sep1(rule, separator) {
|
||||
return seq(rule, repeat(seq(separator, rule)));
|
||||
}
|
||||
|
||||
function commaSep1(rule) {
|
||||
return sep1(rule, ',');
|
||||
}
|
||||
|
||||
function commaSep(rule) {
|
||||
return optional(commaSep1(rule));
|
||||
}
|
||||
@ -1,47 +0,0 @@
|
||||
{
|
||||
"name": "tree-sitter-nix",
|
||||
"version": "0.0.2",
|
||||
"lockfileVersion": 2,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "tree-sitter-nix",
|
||||
"version": "0.0.2",
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"nan": "^2.14.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"tree-sitter-cli": "^0.20.0"
|
||||
}
|
||||
},
|
||||
"node_modules/nan": {
|
||||
"version": "2.15.0",
|
||||
"resolved": "https://registry.npmjs.org/nan/-/nan-2.15.0.tgz",
|
||||
"integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ=="
|
||||
},
|
||||
"node_modules/tree-sitter-cli": {
|
||||
"version": "0.20.6",
|
||||
"resolved": "https://registry.npmjs.org/tree-sitter-cli/-/tree-sitter-cli-0.20.6.tgz",
|
||||
"integrity": "sha512-tjbAeuGSMhco/EnsThjWkQbDIYMDmdkWsTPsa/NJAW7bjaki9P7oM9TkLxfdlnm4LXd1wR5wVSM2/RTLtZbm6A==",
|
||||
"dev": true,
|
||||
"hasInstallScript": true,
|
||||
"bin": {
|
||||
"tree-sitter": "cli.js"
|
||||
}
|
||||
}
|
||||
},
|
||||
"dependencies": {
|
||||
"nan": {
|
||||
"version": "2.15.0",
|
||||
"resolved": "https://registry.npmjs.org/nan/-/nan-2.15.0.tgz",
|
||||
"integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ=="
|
||||
},
|
||||
"tree-sitter-cli": {
|
||||
"version": "0.20.6",
|
||||
"resolved": "https://registry.npmjs.org/tree-sitter-cli/-/tree-sitter-cli-0.20.6.tgz",
|
||||
"integrity": "sha512-tjbAeuGSMhco/EnsThjWkQbDIYMDmdkWsTPsa/NJAW7bjaki9P7oM9TkLxfdlnm4LXd1wR5wVSM2/RTLtZbm6A==",
|
||||
"dev": true
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1,41 +0,0 @@
|
||||
{
|
||||
"name": "tree-sitter-nix",
|
||||
"version": "0.0.2",
|
||||
"description": "Tree Sitter grammar for Nix",
|
||||
"main": "bindings/node",
|
||||
"scripts": {
|
||||
"generate": "tree-sitter generate",
|
||||
"test": "tree-sitter test"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/cstrahan/tree-sitter-nix.git"
|
||||
},
|
||||
"author": "Charles Strahan",
|
||||
"license": "MIT",
|
||||
"bugs": {
|
||||
"url": "https://github.com/cstrahan/tree-sitter-nix/issues"
|
||||
},
|
||||
"homepage": "https://github.com/cstrahan/tree-sitter-nix#readme",
|
||||
"dependencies": {
|
||||
"nan": "^2.14.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"tree-sitter-cli": "^0.20.6"
|
||||
},
|
||||
"tree-sitter": [
|
||||
{
|
||||
"file-types": [
|
||||
"nix"
|
||||
],
|
||||
"highlights": [
|
||||
"queries/highlights.scm"
|
||||
],
|
||||
"injection-regex": "^(nix)$",
|
||||
"locals": [
|
||||
"queries/locals.scm"
|
||||
],
|
||||
"scope": "source.nix"
|
||||
}
|
||||
]
|
||||
}
|
||||
@ -1,97 +0,0 @@
|
||||
(comment) @comment
|
||||
|
||||
[
|
||||
"if"
|
||||
"then"
|
||||
"else"
|
||||
"let"
|
||||
"inherit"
|
||||
"in"
|
||||
"rec"
|
||||
"with"
|
||||
"assert"
|
||||
"or"
|
||||
] @keyword
|
||||
|
||||
((identifier) @variable.builtin
|
||||
(#match? @variable.builtin "^(__currentSystem|__currentTime|__nixPath|__nixVersion|__storeDir|builtins|false|null|true)$")
|
||||
(#is-not? local))
|
||||
|
||||
((identifier) @function.builtin
|
||||
(#match? @function.builtin "^(__add|__addErrorContext|__all|__any|__appendContext|__attrNames|__attrValues|__bitAnd|__bitOr|__bitXor|__catAttrs|__compareVersions|__concatLists|__concatMap|__concatStringsSep|__deepSeq|__div|__elem|__elemAt|__fetchurl|__filter|__filterSource|__findFile|__foldl'|__fromJSON|__functionArgs|__genList|__genericClosure|__getAttr|__getContext|__getEnv|__hasAttr|__hasContext|__hashFile|__hashString|__head|__intersectAttrs|__isAttrs|__isBool|__isFloat|__isFunction|__isInt|__isList|__isPath|__isString|__langVersion|__length|__lessThan|__listToAttrs|__mapAttrs|__match|__mul|__parseDrvName|__partition|__path|__pathExists|__readDir|__readFile|__replaceStrings|__seq|__sort|__split|__splitVersion|__storePath|__stringLength|__sub|__substring|__tail|__toFile|__toJSON|__toPath|__toXML|__trace|__tryEval|__typeOf|__unsafeDiscardOutputDependency|__unsafeDiscardStringContext|__unsafeGetAttrPos|__valueSize|abort|baseNameOf|derivation|derivationStrict|dirOf|fetchGit|fetchMercurial|fetchTarball|fromTOML|import|isNull|map|placeholder|removeAttrs|scopedImport|throw|toString)$")
|
||||
(#is-not? local))
|
||||
|
||||
[
|
||||
(string_expression)
|
||||
(indented_string_expression)
|
||||
] @string
|
||||
|
||||
[
|
||||
(path_expression)
|
||||
(hpath_expression)
|
||||
(spath_expression)
|
||||
] @string.special.path
|
||||
|
||||
(uri_expression) @string.special.uri
|
||||
|
||||
[
|
||||
(integer_expression)
|
||||
(float_expression)
|
||||
] @number
|
||||
|
||||
(interpolation
|
||||
"${" @punctuation.special
|
||||
"}" @punctuation.special) @embedded
|
||||
|
||||
(escape_sequence) @escape
|
||||
|
||||
(function_expression
|
||||
universal: (identifier) @variable.parameter
|
||||
)
|
||||
|
||||
(formal
|
||||
name: (identifier) @variable.parameter
|
||||
"?"? @punctuation.delimiter)
|
||||
|
||||
(select_expression
|
||||
attrpath: (attrpath (identifier)) @property)
|
||||
|
||||
(apply_expression
|
||||
function: [
|
||||
(variable_expression (identifier)) @function
|
||||
(select_expression
|
||||
attrpath: (attrpath
|
||||
attr: (identifier) @function .))])
|
||||
|
||||
(unary_expression
|
||||
operator: _ @operator)
|
||||
|
||||
(binary_expression
|
||||
operator: _ @operator)
|
||||
|
||||
(variable_expression (identifier) @variable)
|
||||
|
||||
(binding
|
||||
attrpath: (attrpath (identifier)) @property)
|
||||
|
||||
(identifier) @property
|
||||
|
||||
(inherit_from attrs: (inherited_attrs attr: (identifier) @property) )
|
||||
|
||||
[
|
||||
";"
|
||||
"."
|
||||
","
|
||||
"="
|
||||
] @punctuation.delimiter
|
||||
|
||||
[
|
||||
"("
|
||||
")"
|
||||
"["
|
||||
"]"
|
||||
"{"
|
||||
"}"
|
||||
] @punctuation.bracket
|
||||
|
||||
(identifier) @variable
|
||||
@ -1,33 +0,0 @@
|
||||
;; when using @local.reference, tree-sitter seems to
|
||||
;; apply the scope from the identifier it has looked up,
|
||||
;; which makes sense for most languages.
|
||||
;; however, we want to highlight things as function based on their call-site,
|
||||
;; not their definition; therefore using TS's support for tracking locals
|
||||
;; impedes our ability to get the highlighting we want.
|
||||
;;
|
||||
;; also, TS doesn't seem to support scoping as implemented in languages
|
||||
;; with lazy let bindings, which results in syntax highlighting/goto-reference
|
||||
;; results that depend on the order of definitions, which is counter to the
|
||||
;; semantics of Nix.
|
||||
;;
|
||||
;; so for now we'll opt for not having any locals queries.
|
||||
;;
|
||||
;; see: https://github.com/tree-sitter/tree-sitter/issues/918
|
||||
|
||||
;(function_expression
|
||||
; universal: (identifier)? @local.definition
|
||||
; formals: (formals (formal name: (identifier) @local.definition)*)
|
||||
; universal: (identifier)? @local.definition
|
||||
; ) @local.scope
|
||||
;
|
||||
;(rec_attrset_expression
|
||||
; bind: (binding
|
||||
; attrpath: (attrpath . (attr_identifier) @local.definition))
|
||||
;) @local.scope
|
||||
;
|
||||
;(let_expression
|
||||
; bind: (binding
|
||||
; attrpath: (attrpath . (attr_identifier) @local.definition))
|
||||
;) @local.scope
|
||||
;
|
||||
;(identifier) @local.reference
|
||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,192 +0,0 @@
|
||||
#include <tree_sitter/parser.h>
|
||||
|
||||
enum TokenType {
|
||||
STRING_FRAGMENT,
|
||||
INDENTED_STRING_FRAGMENT,
|
||||
PATH_START,
|
||||
PATH_FRAGMENT,
|
||||
};
|
||||
|
||||
static void advance(TSLexer *lexer) {
|
||||
lexer->advance(lexer, false);
|
||||
}
|
||||
|
||||
static void skip(TSLexer *lexer) {
|
||||
lexer->advance(lexer, true);
|
||||
}
|
||||
|
||||
// Here we only parse literal fragment inside a string.
|
||||
// Delimiter, interpolation and escape sequence are handled by the parser and we simply stop at them.
|
||||
//
|
||||
// The implementation is inspired by tree-sitter-javascript:
|
||||
// https://github.com/tree-sitter/tree-sitter-javascript/blob/fdeb68ac8d2bd5a78b943528bb68ceda3aade2eb/src/scanner.c#L19
|
||||
static bool scan_string_fragment(TSLexer *lexer) {
|
||||
lexer->result_symbol = STRING_FRAGMENT;
|
||||
for (bool has_content = false;; has_content = true) {
|
||||
lexer->mark_end(lexer);
|
||||
switch (lexer->lookahead) {
|
||||
case '"':
|
||||
case '\\':
|
||||
return has_content;
|
||||
case '$':
|
||||
advance(lexer);
|
||||
if (lexer->lookahead == '{') {
|
||||
return has_content;
|
||||
} else if (lexer->lookahead != '"' && lexer->lookahead != '\\') {
|
||||
// Any char following '$' other than '"', '\\' and '{' (which was handled above)
|
||||
// should be consumed as additional string content.
|
||||
// This means `$${` doesn't start an interpolation, but `$$${` does.
|
||||
advance(lexer);
|
||||
}
|
||||
break;
|
||||
// Simply give up on EOF or '\0'.
|
||||
case '\0':
|
||||
return false;
|
||||
default:
|
||||
advance(lexer);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// See comments of scan_string_fragment.
|
||||
static bool scan_indented_string_fragment(TSLexer *lexer) {
|
||||
lexer->result_symbol = INDENTED_STRING_FRAGMENT;
|
||||
for (bool has_content = false;; has_content = true) {
|
||||
lexer->mark_end(lexer);
|
||||
switch (lexer->lookahead) {
|
||||
case '$':
|
||||
advance(lexer);
|
||||
if (lexer->lookahead == '{') {
|
||||
return has_content;
|
||||
} else if (lexer->lookahead != '\'') {
|
||||
// Any char following '$' other than '\'' and '{' (which was handled above)
|
||||
// should be consumed as additional string content.
|
||||
// This means `$${` doesn't start an interpolation, but `$$${` does.
|
||||
advance(lexer);
|
||||
}
|
||||
break;
|
||||
case '\'':
|
||||
advance(lexer);
|
||||
if (lexer->lookahead == '\'') {
|
||||
// Two single quotes always stop current string fragment.
|
||||
// It can be either an end delimiter '', or escape sequences ''', ''$, ''\<any>
|
||||
return has_content;
|
||||
}
|
||||
break;
|
||||
// Simply give up on EOF or '\0'.
|
||||
case '\0':
|
||||
return false;
|
||||
default:
|
||||
advance(lexer);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static bool is_path_char(char c) {
|
||||
return (c >= '0' && c <= '9') || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '-' || c == '+' || c == '_' || c == '.' || c == '/';
|
||||
}
|
||||
|
||||
static bool scan_path_start(TSLexer *lexer) {
|
||||
lexer->result_symbol = PATH_START;
|
||||
|
||||
bool have_sep = false;
|
||||
bool have_after_sep = false;
|
||||
char c = lexer->lookahead;
|
||||
|
||||
// unlike string_fragments which which are preceded by initial token (i.e. '"')
|
||||
// and thus will have all leading external whitespace consumed,
|
||||
// we have no such luxury with the path_start token.
|
||||
//
|
||||
// so we must skip over any leading whitespace here.
|
||||
while (c == ' ' || c == '\n' || c == '\r' || c == '\t') {
|
||||
skip(lexer);
|
||||
c = lexer->lookahead;
|
||||
}
|
||||
|
||||
while (true) {
|
||||
lexer->mark_end(lexer);
|
||||
c = lexer->lookahead;
|
||||
|
||||
if (c == '/') {
|
||||
have_sep = true;
|
||||
} else if (is_path_char(c)) {
|
||||
if (have_sep) {
|
||||
have_after_sep = true;
|
||||
}
|
||||
} else if (c == '$') {
|
||||
// starting a interpolation,
|
||||
// so we have a valid token as long as we've seen a separator.
|
||||
// example: a/${x}
|
||||
return have_sep;
|
||||
} else {
|
||||
// we have a valid token if we've consumed anything after a separator.
|
||||
// example: a/b
|
||||
return have_after_sep;
|
||||
}
|
||||
|
||||
advance(lexer);
|
||||
}
|
||||
}
|
||||
|
||||
static bool scan_path_fragment(TSLexer *lexer) {
|
||||
lexer->result_symbol = PATH_FRAGMENT;
|
||||
|
||||
for (bool has_content = false;; has_content = true) {
|
||||
lexer->mark_end(lexer);
|
||||
if (!is_path_char(lexer->lookahead)) {
|
||||
return has_content;
|
||||
}
|
||||
advance(lexer);
|
||||
}
|
||||
}
|
||||
|
||||
void *tree_sitter_nix_external_scanner_create() {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
bool tree_sitter_nix_external_scanner_scan(void *payload, TSLexer *lexer,
|
||||
const bool *valid_symbols) {
|
||||
// This never happens in valid grammar. Only during error recovery, everything becomes valid.
|
||||
// See: https://github.com/tree-sitter/tree-sitter/issues/1259
|
||||
//
|
||||
// We should not consume any content as string fragment during error recovery, or we'll break
|
||||
// more valid grammar below.
|
||||
// The test 'attrset typing field following string' covers this.
|
||||
if (valid_symbols[STRING_FRAGMENT] && valid_symbols[INDENTED_STRING_FRAGMENT] && valid_symbols[PATH_START] && valid_symbols[PATH_FRAGMENT] ) {
|
||||
return false;
|
||||
} else if (valid_symbols[STRING_FRAGMENT]) {
|
||||
return scan_string_fragment(lexer);
|
||||
} else if (valid_symbols[INDENTED_STRING_FRAGMENT]) {
|
||||
return scan_indented_string_fragment(lexer);
|
||||
} else if (valid_symbols[PATH_FRAGMENT] && is_path_char(lexer->lookahead)) {
|
||||
// path_fragments should be scanned as immediate tokens, with no preceding extras.
|
||||
// so we assert that the very first token is a path character,
|
||||
// and otherwise we fall through to the case below.
|
||||
// example:
|
||||
// a/b${c} d/e${f}
|
||||
// ^--- note that scanning for the path_fragment will start here.
|
||||
// this *should* be parsed as a function application.
|
||||
// so we want to fall through to the path_start case below,
|
||||
// which will skip the whitespace and correctly scan the following path_start.
|
||||
//
|
||||
// also, we want this above path_start, because wherever there's ambiguity we want to parse another fragment
|
||||
// instead of starting a new path.
|
||||
// example:
|
||||
// a/b${c}d/e${f}
|
||||
// if we swap the precedence, we'd effectively parse the above as the following function application:
|
||||
// (a/b${c}) (d/e${f})
|
||||
return scan_path_fragment(lexer);
|
||||
} else if (valid_symbols[PATH_START]) {
|
||||
return scan_path_start(lexer);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
unsigned tree_sitter_nix_external_scanner_serialize(void *payload, char *buffer) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
void tree_sitter_nix_external_scanner_deserialize(void *payload, const char *buffer, unsigned length) { }
|
||||
|
||||
void tree_sitter_nix_external_scanner_destroy(void *payload) { }
|
||||
@ -1,224 +0,0 @@
|
||||
#ifndef TREE_SITTER_PARSER_H_
|
||||
#define TREE_SITTER_PARSER_H_
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#define ts_builtin_sym_error ((TSSymbol)-1)
|
||||
#define ts_builtin_sym_end 0
|
||||
#define TREE_SITTER_SERIALIZATION_BUFFER_SIZE 1024
|
||||
|
||||
typedef uint16_t TSStateId;
|
||||
|
||||
#ifndef TREE_SITTER_API_H_
|
||||
typedef uint16_t TSSymbol;
|
||||
typedef uint16_t TSFieldId;
|
||||
typedef struct TSLanguage TSLanguage;
|
||||
#endif
|
||||
|
||||
typedef struct {
|
||||
TSFieldId field_id;
|
||||
uint8_t child_index;
|
||||
bool inherited;
|
||||
} TSFieldMapEntry;
|
||||
|
||||
typedef struct {
|
||||
uint16_t index;
|
||||
uint16_t length;
|
||||
} TSFieldMapSlice;
|
||||
|
||||
typedef struct {
|
||||
bool visible;
|
||||
bool named;
|
||||
bool supertype;
|
||||
} TSSymbolMetadata;
|
||||
|
||||
typedef struct TSLexer TSLexer;
|
||||
|
||||
struct TSLexer {
|
||||
int32_t lookahead;
|
||||
TSSymbol result_symbol;
|
||||
void (*advance)(TSLexer *, bool);
|
||||
void (*mark_end)(TSLexer *);
|
||||
uint32_t (*get_column)(TSLexer *);
|
||||
bool (*is_at_included_range_start)(const TSLexer *);
|
||||
bool (*eof)(const TSLexer *);
|
||||
};
|
||||
|
||||
typedef enum {
|
||||
TSParseActionTypeShift,
|
||||
TSParseActionTypeReduce,
|
||||
TSParseActionTypeAccept,
|
||||
TSParseActionTypeRecover,
|
||||
} TSParseActionType;
|
||||
|
||||
typedef union {
|
||||
struct {
|
||||
uint8_t type;
|
||||
TSStateId state;
|
||||
bool extra;
|
||||
bool repetition;
|
||||
} shift;
|
||||
struct {
|
||||
uint8_t type;
|
||||
uint8_t child_count;
|
||||
TSSymbol symbol;
|
||||
int16_t dynamic_precedence;
|
||||
uint16_t production_id;
|
||||
} reduce;
|
||||
uint8_t type;
|
||||
} TSParseAction;
|
||||
|
||||
typedef struct {
|
||||
uint16_t lex_state;
|
||||
uint16_t external_lex_state;
|
||||
} TSLexMode;
|
||||
|
||||
typedef union {
|
||||
TSParseAction action;
|
||||
struct {
|
||||
uint8_t count;
|
||||
bool reusable;
|
||||
} entry;
|
||||
} TSParseActionEntry;
|
||||
|
||||
struct TSLanguage {
|
||||
uint32_t version;
|
||||
uint32_t symbol_count;
|
||||
uint32_t alias_count;
|
||||
uint32_t token_count;
|
||||
uint32_t external_token_count;
|
||||
uint32_t state_count;
|
||||
uint32_t large_state_count;
|
||||
uint32_t production_id_count;
|
||||
uint32_t field_count;
|
||||
uint16_t max_alias_sequence_length;
|
||||
const uint16_t *parse_table;
|
||||
const uint16_t *small_parse_table;
|
||||
const uint32_t *small_parse_table_map;
|
||||
const TSParseActionEntry *parse_actions;
|
||||
const char * const *symbol_names;
|
||||
const char * const *field_names;
|
||||
const TSFieldMapSlice *field_map_slices;
|
||||
const TSFieldMapEntry *field_map_entries;
|
||||
const TSSymbolMetadata *symbol_metadata;
|
||||
const TSSymbol *public_symbol_map;
|
||||
const uint16_t *alias_map;
|
||||
const TSSymbol *alias_sequences;
|
||||
const TSLexMode *lex_modes;
|
||||
bool (*lex_fn)(TSLexer *, TSStateId);
|
||||
bool (*keyword_lex_fn)(TSLexer *, TSStateId);
|
||||
TSSymbol keyword_capture_token;
|
||||
struct {
|
||||
const bool *states;
|
||||
const TSSymbol *symbol_map;
|
||||
void *(*create)(void);
|
||||
void (*destroy)(void *);
|
||||
bool (*scan)(void *, TSLexer *, const bool *symbol_whitelist);
|
||||
unsigned (*serialize)(void *, char *);
|
||||
void (*deserialize)(void *, const char *, unsigned);
|
||||
} external_scanner;
|
||||
const TSStateId *primary_state_ids;
|
||||
};
|
||||
|
||||
/*
|
||||
* Lexer Macros
|
||||
*/
|
||||
|
||||
#define START_LEXER() \
|
||||
bool result = false; \
|
||||
bool skip = false; \
|
||||
bool eof = false; \
|
||||
int32_t lookahead; \
|
||||
goto start; \
|
||||
next_state: \
|
||||
lexer->advance(lexer, skip); \
|
||||
start: \
|
||||
skip = false; \
|
||||
lookahead = lexer->lookahead;
|
||||
|
||||
#define ADVANCE(state_value) \
|
||||
{ \
|
||||
state = state_value; \
|
||||
goto next_state; \
|
||||
}
|
||||
|
||||
#define SKIP(state_value) \
|
||||
{ \
|
||||
skip = true; \
|
||||
state = state_value; \
|
||||
goto next_state; \
|
||||
}
|
||||
|
||||
#define ACCEPT_TOKEN(symbol_value) \
|
||||
result = true; \
|
||||
lexer->result_symbol = symbol_value; \
|
||||
lexer->mark_end(lexer);
|
||||
|
||||
#define END_STATE() return result;
|
||||
|
||||
/*
|
||||
* Parse Table Macros
|
||||
*/
|
||||
|
||||
#define SMALL_STATE(id) id - LARGE_STATE_COUNT
|
||||
|
||||
#define STATE(id) id
|
||||
|
||||
#define ACTIONS(id) id
|
||||
|
||||
#define SHIFT(state_value) \
|
||||
{{ \
|
||||
.shift = { \
|
||||
.type = TSParseActionTypeShift, \
|
||||
.state = state_value \
|
||||
} \
|
||||
}}
|
||||
|
||||
#define SHIFT_REPEAT(state_value) \
|
||||
{{ \
|
||||
.shift = { \
|
||||
.type = TSParseActionTypeShift, \
|
||||
.state = state_value, \
|
||||
.repetition = true \
|
||||
} \
|
||||
}}
|
||||
|
||||
#define SHIFT_EXTRA() \
|
||||
{{ \
|
||||
.shift = { \
|
||||
.type = TSParseActionTypeShift, \
|
||||
.extra = true \
|
||||
} \
|
||||
}}
|
||||
|
||||
#define REDUCE(symbol_val, child_count_val, ...) \
|
||||
{{ \
|
||||
.reduce = { \
|
||||
.type = TSParseActionTypeReduce, \
|
||||
.symbol = symbol_val, \
|
||||
.child_count = child_count_val, \
|
||||
__VA_ARGS__ \
|
||||
}, \
|
||||
}}
|
||||
|
||||
#define RECOVER() \
|
||||
{{ \
|
||||
.type = TSParseActionTypeRecover \
|
||||
}}
|
||||
|
||||
#define ACCEPT_INPUT() \
|
||||
{{ \
|
||||
.type = TSParseActionTypeAccept \
|
||||
}}
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif // TREE_SITTER_PARSER_H_
|
||||
@ -1,80 +0,0 @@
|
||||
{
|
||||
or = { or = 1; }.or or 42;
|
||||
# <- property
|
||||
# ^ punctuation.delimiter
|
||||
# ^ property
|
||||
# ^ property
|
||||
# ^ keyword
|
||||
the-question = if builtins.true then "to be" else "not to be";
|
||||
# <- property
|
||||
# ^ property
|
||||
# ^ property
|
||||
# ^ keyword
|
||||
# ^ variable.builtin
|
||||
# ^ property
|
||||
# ^ keyword
|
||||
# ^ string
|
||||
# ^ keyword
|
||||
# ^ string
|
||||
null = if null then true else false;
|
||||
# <- property
|
||||
# ^ variable.builtin
|
||||
# ^ variable.builtin
|
||||
# ^ variable.builtin
|
||||
pkgs' = { inherit (pkgs) stdenv lib; };
|
||||
# <- property
|
||||
# ^ property
|
||||
# ^ keyword
|
||||
# ^ variable
|
||||
# ^ property
|
||||
# ^ property
|
||||
thing' =
|
||||
# <- property
|
||||
let inherit (pkgs) stdenv lib;
|
||||
# <- keyword
|
||||
# ^ keyword
|
||||
# ^ variable
|
||||
# ^ property
|
||||
# ^ property
|
||||
in derivation rec {
|
||||
# <- keyword
|
||||
# ^ function.builtin
|
||||
# ^ keyword
|
||||
pname = "thing";
|
||||
# <- property
|
||||
# ^ string
|
||||
version = "v1.2.3";
|
||||
name = "${pname}-${version}";
|
||||
# <- property
|
||||
# ^ string
|
||||
# ^ punctuation.special
|
||||
# ^ variable
|
||||
# ^ punctuation.special
|
||||
# ^ string
|
||||
# ^ variable
|
||||
# ^ string
|
||||
buildInputs = with pkgs; [ thing_a thing_b ];
|
||||
# <- property
|
||||
# ^ keyword
|
||||
# ^ variable
|
||||
# ^ variable
|
||||
# ^ variable
|
||||
};
|
||||
assert_bool = bool: assert lib.isBool bool; bool;
|
||||
# <- property
|
||||
# ^ variable.parameter
|
||||
# ^ keyword
|
||||
# ^ variable
|
||||
# ^ function
|
||||
# ^ variable
|
||||
# ^ variable
|
||||
import = import ./overlays.nix { inherit pkgs; };
|
||||
# <- property
|
||||
# ^ function.builtin
|
||||
# ^ string.special.path
|
||||
# ^ keyword
|
||||
# ^ property
|
||||
uri = https://github.com;
|
||||
# ^ string.special.uri
|
||||
# ^ string.special.uri
|
||||
}
|
||||
Loading…
Reference in New Issue