Compare commits
7 Commits
4e28f71dc9
...
d600c0a8ca
| Author | SHA1 | Date | |
|---|---|---|---|
| d600c0a8ca | |||
| 49f0f23a54 | |||
| d6bd5f4096 | |||
| 64a890ac03 | |||
| 6459d63f2b | |||
| f56a94e35e | |||
| 3de76e196c |
@@ -0,0 +1,39 @@
|
||||
# Project Overview: skillls
|
||||
|
||||
`skillls` is a Language Server Protocol (LSP) implementation for the **Skill** language (specifically targeting `.il` and `.ocn` files). It provides essential IDE features to enhance the development experience, such as error detection, structural navigation, and intelligent code hints.
|
||||
|
||||
## Core Capabilities
|
||||
|
||||
The server implements several key LSP features:
|
||||
|
||||
- **Diagnostics**: Automatically detects syntax errors, specifically focusing on parenthesis mismatches (too many opening or closing parentheses), and reports them with precise line/column information to the editor.
|
||||
- **Document Symbols**: Parses the file structure to generate a hierarchy of scopes (nodes). This enables editors to provide an "Outline" or "Symbol Tree" view for navigating functions, variables, and namespaces.
|
||||
- **Inlay Hints**: Provides inline metadata at specific code locations, allowing the editor to display additional context directly within the source text.
|
||||
- **Workspace Initialization**: Upon connecting, the server scans the workspace root for relevant `.il` and `.ocn` files, building an initial representation of the project's scopes.
|
||||
|
||||
## Architecture & Implementation
|
||||
|
||||
### Parsing Logic
|
||||
The project uses a multi-layered approach to understand the Skill language:
|
||||
1. **Content Cleaning**: A pre-processing step identifies and handles comments (`;`) and strings (`"..."`) to ensure parsing is not misled by ignored text.
|
||||
2. **Structural Analysis**: The server identifies "scope starters" using regular expressions and manual parenthesis tracking to determine the boundaries of functions or namespaces.
|
||||
3. **Hierarchy Building**: Once individual nodes are identified, the server builds a parent-child tree structure based on the nesting level of parentheses.
|
||||
4. **Symbol Extraction**: Within each scope, the parser identifies local variables and symbols to populate the `DocumentSymbol` list.
|
||||
|
||||
### Key Components
|
||||
|
||||
- **`skillls/main.py`**: The entry point of the LSP server. It implements the `LanguageServer` class and contains the handlers for LSP lifecycle events (`initialize`, `didOpen`, `didChange`, etc.) and feature requests (`inlayHint`, `documentSymbol`).
|
||||
- **`skillls/parser.py`**: The new Tree-sitter based parser for syntax tree traversal and symbol extraction.
|
||||
- **`skillls/types.py`**: Defines the internal data models (e.g., `Node`, `URI`) used across the project.
|
||||
|
||||
## Roadmap & Engineering Planning
|
||||
|
||||
For details on identified technical debt, fragilities, and the long-term architectural hardening strategy, refer to [PLAN.md](./PLAN.md).
|
||||
|
||||
## Technical Stack
|
||||
|
||||
- **Language**: Python 3.11+
|
||||
- **Package Management**: `uv`
|
||||
- **LSP Framework**: `pygls` (Python Language Server)
|
||||
- **Parsing Utilities**: `tree-sitter` (for structural tree analysis).
|
||||
- **Formatting & Tooling**: `rich` (terminal output), `ruff`, `mypy`, `pytest`.
|
||||
@@ -0,0 +1,32 @@
|
||||
# Project Hardening Plan
|
||||
|
||||
This document outlines the identified fragilities in the `skillls` project and the planned architectural improvements to transform it from a functional prototype into a robust, production-ready Language Server.
|
||||
|
||||
## 1. Grammar-Logic Decoupling
|
||||
**Problem**: The `SkillParser` relies on hardcoded string literals (e/g., `"function_definition"`) to identify symbols. Changes in the underlying `tree-sitter-skill` grammar will cause silent failures in the Outline view.
|
||||
**Goal**: Create a stable contract between the grammar and the parser.
|
||||
**Proposed Actions**:
|
||||
- [x] Implement a shared constants module or configuration file that defines significant node types.
|
||||
- [ ] (Long-term) Explore using Tree-sitter Queries (`Query` API) to match patterns instead of manual type checking, making the parser less dependent on specific node names and more focused on structural patterns.
|
||||
|
||||
## 2. Iterative AST Traversal
|
||||
**Problem**: The current recursive traversal in `_traverse_tree` is susceptible to `RecursionError` on deeply nested files.
|
||||
**Goal**: Ensure the server can handle arbitrarily deep syntax trees without crashing.
|
||||
**Proposed Actions**:
|
||||
- [x] Refactor `SkillParser._traverse_tree` to use an iterative approach (using a stack/deque) instead of recursion.
|
||||
|
||||
## 3. Single Source of Truth for Errors
|
||||
**Problem**: The project is in a transitional state where error management is split between the new `SkillParser` diagnostics and the legacy `server.errs` dictionary in `main.py`.
|
||||
**Goal**: Unify error reporting into a single, streamlined pipeline.
|
||||
**Proposed Actions**:
|
||||
- [x] Complete the refactor of `skillls/main.py`.
|
||||
- [x] Remove the `errs` dictionary from `SkillLanguageServer`.
|
||||
- [x] Decommission and delete deprecated files: `skillls/checker.py` and unused parts of `skillls/helpers.py`.
|
||||
|
||||
|
||||
## 5. Test Suite Strengthening
|
||||
**Problem**: While core logic is tested, the LSP lifecycle and complex parsing edge cases lack specific unit test coverage.
|
||||
**Goal**: Achieve high-confidence verification of the LSP server's behavior and parser robustness.
|
||||
**Proposed Actions**:
|
||||
- [x] Implement \`tests/test_server.py\` to verify LSP lifecycle events (\`didOpen\`, \`didChange\`) and diagnostic publishing logic.
|
||||
- [x] Harden `tests/test_parser.py` by implementing deterministic symbol extraction verification instead of existence checks.
|
||||
+5
-4
@@ -11,7 +11,7 @@ dependencies = [
|
||||
]
|
||||
requires-python = ">= 3.11"
|
||||
|
||||
[project.optional-dependencies]
|
||||
[dependency-groups]
|
||||
dev = [
|
||||
"black",
|
||||
"mypy",
|
||||
@@ -30,14 +30,15 @@ requires = [
|
||||
skillls = "skillls.main:main"
|
||||
|
||||
|
||||
[tools.black]
|
||||
[tool.black]
|
||||
line-length = 100
|
||||
target-version = "py311"
|
||||
include = "skillls"
|
||||
|
||||
[tools.ruff]
|
||||
[tool.ruff]
|
||||
line-length = 100
|
||||
include = ['ALL']
|
||||
target-version = "py311"
|
||||
include = ["pyproject.toml", "skillls/**/*.py"]
|
||||
|
||||
[tool.uv.sources]
|
||||
tree-sitter-skill = { git = "ssh://git@git.acereca.net/acereca/tree-sitter-skill.git" }
|
||||
|
||||
@@ -1,75 +0,0 @@
|
||||
from dataclasses import dataclass
|
||||
from enum import Enum, auto
|
||||
|
||||
from lsprotocol.types import Location, Position, Range
|
||||
|
||||
from skillls.types import URI
|
||||
|
||||
|
||||
class SyntaxError(Exception):
|
||||
pass
|
||||
|
||||
|
||||
class ParenMismatchErrorKind(Enum):
|
||||
TooManyClosed = "Found too many closing parens"
|
||||
TooManyOpened = "Found too many open parens"
|
||||
|
||||
|
||||
@dataclass
|
||||
class ParenMismatchError(SyntaxError):
|
||||
kind: ParenMismatchErrorKind
|
||||
loc: Range
|
||||
|
||||
|
||||
def _check_for_matching_parens(content: str) -> list[Exception]:
|
||||
excs: list[Exception] = []
|
||||
|
||||
opened = 0
|
||||
line = 0
|
||||
col = 0
|
||||
last_open: Position = Position(0, 0)
|
||||
last_close: Position = Position(0, 0)
|
||||
for char in content:
|
||||
match char:
|
||||
case "(":
|
||||
opened += 1
|
||||
last_open = Position(line, col)
|
||||
|
||||
case ")":
|
||||
opened -= 1
|
||||
if opened < 0:
|
||||
excs.append(
|
||||
ParenMismatchError(
|
||||
ParenMismatchErrorKind.TooManyClosed,
|
||||
Range(Position(line, col), Position(line, col + 1)),
|
||||
)
|
||||
)
|
||||
opened = 0
|
||||
last_close = Position(line, col)
|
||||
case "\n":
|
||||
line += 1
|
||||
col = -1
|
||||
|
||||
case _:
|
||||
pass
|
||||
|
||||
col += 1
|
||||
|
||||
if opened > 0:
|
||||
excs.append(
|
||||
ParenMismatchError(
|
||||
ParenMismatchErrorKind.TooManyOpened,
|
||||
Range(last_open, Position(last_open.line, last_open.character + 1)),
|
||||
)
|
||||
)
|
||||
|
||||
return excs
|
||||
|
||||
|
||||
def check_content_for_errors(clean_content: str) -> None:
|
||||
excs: list[Exception] = []
|
||||
|
||||
excs.extend(_check_for_matching_parens(clean_content))
|
||||
|
||||
if excs:
|
||||
raise ExceptionGroup("", excs)
|
||||
@@ -0,0 +1,19 @@
|
||||
"""
|
||||
Centralized constants for the Skill language parser and LSP server.
|
||||
"""
|
||||
|
||||
from typing import Final, Set
|
||||
|
||||
# Node types that represent syntax errors in Tree-sitter
|
||||
ERROR_NODE_TYPES: Final[Set[str]] = {"ERROR", "MISSING"}
|
||||
|
||||
# Node types that are considered significant enough to appear in the Document Symbol outline
|
||||
SYMBOLIC_NODE_TYPES: Final[Set[str]] = {
|
||||
"function_definition",
|
||||
"procedure_definition",
|
||||
"namespace",
|
||||
"let_binding",
|
||||
}
|
||||
|
||||
# Node types used to identify names/identifiers within symbolic nodes
|
||||
IDENTIFIER_NODE_TYPES: Final[Set[str]] = {"identifier", "name"}
|
||||
@@ -1,192 +0,0 @@
|
||||
from copy import copy
|
||||
from dataclasses import dataclass
|
||||
from logging import getLogger
|
||||
from pathlib import Path
|
||||
from pprint import pformat
|
||||
from lsprotocol.types import DocumentSymbol, Position, Range, SymbolKind
|
||||
from re import MULTILINE, compile as recompile, finditer
|
||||
|
||||
from pygls.workspace import TextDocument
|
||||
|
||||
from skillls.checker import check_content_for_errors
|
||||
from skillls.types import URI, Node, NodeKind
|
||||
|
||||
logger = getLogger(__name__)
|
||||
|
||||
|
||||
@dataclass
|
||||
class ParserCleanerState:
|
||||
in_comment: bool = False
|
||||
in_string: bool = False
|
||||
|
||||
|
||||
NODE_KIND_OPTIONS = "|".join(k.value for k in NodeKind)
|
||||
NAMESPACE_STARTERS = recompile(
|
||||
(rf"(\(\s*(?P<typ>{NODE_KIND_OPTIONS})\b|\b(?P<ctyp>{NODE_KIND_OPTIONS})\()"),
|
||||
MULTILINE,
|
||||
)
|
||||
|
||||
|
||||
def clean_content(content: str) -> str:
|
||||
content_cleaned = ""
|
||||
state = ParserCleanerState()
|
||||
|
||||
for cix, char in enumerate(content):
|
||||
match (content[cix], state):
|
||||
case ";", ParserCleanerState(in_comment=False, in_string=False):
|
||||
state.in_comment = True
|
||||
case '"', ParserCleanerState(in_comment=False):
|
||||
if content[cix - 1] != "\\":
|
||||
state.in_string = not state.in_string
|
||||
content_cleaned += char
|
||||
case "\n", ParserCleanerState(in_comment=True):
|
||||
state.in_comment = False
|
||||
content_cleaned += char
|
||||
case _, ParserCleanerState(in_comment=False, in_string=False):
|
||||
content_cleaned += char
|
||||
|
||||
case _, ParserCleanerState(in_comment=False, in_string=True):
|
||||
content_cleaned += " "
|
||||
case _:
|
||||
pass
|
||||
|
||||
return content_cleaned
|
||||
|
||||
|
||||
def build_node_hierarchy(nodes: list[Node]) -> list[Node]:
|
||||
to_be_sorted = copy(nodes)
|
||||
sorted: list[Node] = []
|
||||
|
||||
while to_be_sorted:
|
||||
node_to_sort = to_be_sorted.pop(0)
|
||||
|
||||
for sorted_node in sorted:
|
||||
if sorted_node.should_contain(node_to_sort):
|
||||
sorted_node.add_child(node_to_sort)
|
||||
break
|
||||
|
||||
else:
|
||||
sorted.append(node_to_sort)
|
||||
|
||||
return sorted
|
||||
|
||||
|
||||
def find_scopes(content_cleaned: str, scope_prefix: str = "") -> list[Node]:
|
||||
ret: list[Node] = []
|
||||
|
||||
for found in NAMESPACE_STARTERS.finditer(content_cleaned):
|
||||
partial = content_cleaned[found.end() :]
|
||||
open_brackets = 1
|
||||
offset = 0
|
||||
for offset, char in enumerate(partial):
|
||||
match char:
|
||||
case "(":
|
||||
open_brackets += 1
|
||||
case ")":
|
||||
open_brackets -= 1
|
||||
|
||||
if open_brackets == 0:
|
||||
break
|
||||
|
||||
case _:
|
||||
pass
|
||||
|
||||
pre_lines = content_cleaned[: found.start()].splitlines()
|
||||
start_line = len(pre_lines) - (
|
||||
1 if pre_lines[-1] != "" and pre_lines[-1].strip() == "" else 0
|
||||
)
|
||||
start_char = len(pre_lines[-1])
|
||||
|
||||
inner_lines = content_cleaned[
|
||||
found.start() : found.end() + offset + 1
|
||||
].splitlines()
|
||||
end_line = start_line + len(inner_lines) - 1
|
||||
end_char = len(inner_lines[-1])
|
||||
|
||||
kind = NodeKind(found.group("typ") or found.group("ctyp"))
|
||||
loc = Range(Position(start_line, start_char), Position(end_line, end_char))
|
||||
|
||||
node = Node(
|
||||
node=f"{scope_prefix}.{kind.value}_{len([n for n in ret if n.kind == kind])}",
|
||||
kind=kind,
|
||||
location=loc,
|
||||
)
|
||||
ret.append(node)
|
||||
|
||||
next = found.end()
|
||||
|
||||
# allowed scoped locals syntax
|
||||
# function(pos1 pos2)
|
||||
# function(pos1 (pos2 default))
|
||||
# function(pos1 @rest args)
|
||||
# function(pos1 @key (kwarg1 default1) (kwarg2 default2))
|
||||
|
||||
while content_cleaned[next] != "(":
|
||||
if content_cleaned[next] == "\n":
|
||||
start_line += 1
|
||||
start_char = 0
|
||||
next += 1
|
||||
start_char += 1
|
||||
|
||||
next += 1
|
||||
last = 0
|
||||
|
||||
for positional in finditer(
|
||||
r"(?P<leading>\s*)(?P<local>\w+|\(\w+\b[^)]*\))(?P<trailing>\s*)",
|
||||
content_cleaned[next:],
|
||||
):
|
||||
if positional.start() != last:
|
||||
logger.debug(
|
||||
f"found ({positional}), but last ({last}) != ({positional.start()})"
|
||||
)
|
||||
break
|
||||
|
||||
last = positional.end()
|
||||
|
||||
leading_nls = positional.group("leading").count("\n")
|
||||
inner_nls = positional.group("local").count("\n")
|
||||
trailing_nls = positional.group("trailing").count("\n")
|
||||
|
||||
local_name = positional.group("local").split()[0]
|
||||
local = DocumentSymbol(
|
||||
name=local_name,
|
||||
kind=SymbolKind.Variable,
|
||||
range=Range(
|
||||
Position(
|
||||
start_line + leading_nls,
|
||||
len(positional.group("leading")) + start_char,
|
||||
),
|
||||
Position(
|
||||
start_line + leading_nls,
|
||||
len(positional.group("leading")) + start_char + len(local_name),
|
||||
),
|
||||
),
|
||||
selection_range=Range(
|
||||
Position(
|
||||
start_line + leading_nls,
|
||||
len(positional.group("leading")) + start_char,
|
||||
),
|
||||
Position(
|
||||
start_line + leading_nls,
|
||||
len(positional.group("leading")) + start_char + len(local_name),
|
||||
),
|
||||
),
|
||||
)
|
||||
node.symbols[local_name] = local
|
||||
|
||||
start_line += leading_nls + inner_nls + trailing_nls
|
||||
start_char += len(positional.group(0))
|
||||
|
||||
# other cases
|
||||
|
||||
logger.debug(pformat(node))
|
||||
return build_node_hierarchy(ret)
|
||||
|
||||
|
||||
def parse_file(file: TextDocument) -> list[Node]:
|
||||
content = file.source
|
||||
content_cleaned = clean_content(content)
|
||||
|
||||
check_content_for_errors(content_cleaned)
|
||||
|
||||
return find_scopes(content_cleaned, scope_prefix=Path(file.path).stem)
|
||||
+22
-34
@@ -27,8 +27,7 @@ from lsprotocol.types import (
|
||||
from pygls.lsp.server import LanguageServer
|
||||
|
||||
|
||||
from skillls.checker import ParenMismatchError
|
||||
from skillls.helpers import parse_file
|
||||
from skillls.parser import SkillParser
|
||||
from skillls.types import URI, Node
|
||||
|
||||
basicConfig(
|
||||
@@ -44,7 +43,7 @@ class SkillLanguageServer(LanguageServer):
|
||||
ws_files: set[URI]
|
||||
opened_files: set[URI]
|
||||
scopes: dict[URI, list[Node]]
|
||||
errs: dict[URI, ExceptionGroup]
|
||||
diagnostics: dict[URI, list[Diagnostic]]
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
@@ -56,25 +55,14 @@ class SkillLanguageServer(LanguageServer):
|
||||
super().__init__(name, version, text_document_sync_kind, notebook_document_sync)
|
||||
self.ws_files = set()
|
||||
self.opened_files = set()
|
||||
self.scopes = {}
|
||||
self.errs = {}
|
||||
self.scopes: dict[URI, list[DocumentSymbol]] = {}
|
||||
self.diagnostics: dict[URI, list[Diagnostic]] = {}
|
||||
self.parser = SkillParser()
|
||||
|
||||
def update_diagnostics(self) -> None:
|
||||
for uri in self.opened_files:
|
||||
diags: list[Diagnostic] = []
|
||||
if eg := self.errs.get(uri):
|
||||
for exc in eg.exceptions:
|
||||
match exc:
|
||||
case ParenMismatchError():
|
||||
diags.append(
|
||||
Diagnostic(
|
||||
message=f"[skill_ls] {Path.from_uri(uri).name}:{exc.loc.start.line} {exc.kind.value}",
|
||||
severity=DiagnosticSeverity.Error,
|
||||
range=exc.loc,
|
||||
)
|
||||
)
|
||||
diags = self.diagnostics.get(uri, [])
|
||||
|
||||
# if diags:
|
||||
self.text_document_publish_diagnostics(
|
||||
PublishDiagnosticsParams(
|
||||
uri=uri,
|
||||
@@ -105,11 +93,12 @@ def lsp_initialize(server: SkillLanguageServer, params: InitializeParams) -> Non
|
||||
|
||||
server.ws_files.add(uri)
|
||||
try:
|
||||
server.scopes[uri] = parse_file(server.workspace.get_text_document(uri))
|
||||
if server.errs.get(uri):
|
||||
del server.errs[uri]
|
||||
except ExceptionGroup as eg:
|
||||
server.errs[uri] = eg
|
||||
text_doc = server.workspace.get_text_document(uri)
|
||||
symbols, diagnostics = server.parser.parse_document(text_doc)
|
||||
server.scopes[uri] = symbols
|
||||
server.diagnostics[uri] = diagnostics
|
||||
except Exception as e:
|
||||
logger.error(f"Error initializing file {uri}: {e}")
|
||||
|
||||
|
||||
@server.feature(TEXT_DOCUMENT_DID_OPEN)
|
||||
@@ -128,13 +117,12 @@ def on_close(server: SkillLanguageServer, params: DidCloseTextDocumentParams) ->
|
||||
@server.feature(TEXT_DOCUMENT_DID_SAVE)
|
||||
def on_change(server: SkillLanguageServer, params: DidChangeTextDocumentParams) -> None:
|
||||
try:
|
||||
server.scopes[params.text_document.uri] = parse_file(
|
||||
server.workspace.get_text_document(params.text_document.uri)
|
||||
)
|
||||
if server.errs.get(params.text_document.uri):
|
||||
del server.errs[params.text_document.uri]
|
||||
except ExceptionGroup as eg:
|
||||
server.errs[params.text_document.uri] = eg
|
||||
text_doc = server.workspace.get_text_document(params.text_document.uri)
|
||||
symbols, diagnostics = server.parser.parse_document(text_doc)
|
||||
server.scopes[params.text_document.uri] = symbols
|
||||
server.diagnostics[params.text_document.uri] = diagnostics
|
||||
except Exception as e:
|
||||
logger.error(f"Error changing file {params.text_document.uri}: {e}")
|
||||
|
||||
server.update_diagnostics()
|
||||
|
||||
@@ -143,13 +131,13 @@ def on_change(server: SkillLanguageServer, params: DidChangeTextDocumentParams)
|
||||
def on_inlay(server: SkillLanguageServer, params: InlayHintParams) -> list[InlayHint]:
|
||||
hints: list[InlayHint] = []
|
||||
uri = params.text_document.uri
|
||||
for node in server.scopes.get(uri, []):
|
||||
for symbol in server.scopes.get(uri, []):
|
||||
hints.append(
|
||||
InlayHint(
|
||||
label=node.node,
|
||||
label=symbol.name,
|
||||
kind=InlayHintKind.Type,
|
||||
padding_left=True,
|
||||
position=node.location.end,
|
||||
position=symbol.range.end,
|
||||
)
|
||||
)
|
||||
|
||||
@@ -160,7 +148,7 @@ def on_inlay(server: SkillLanguageServer, params: InlayHintParams) -> list[Inlay
|
||||
def on_symbols(
|
||||
server: SkillLanguageServer, params: DocumentSymbolParams
|
||||
) -> list[DocumentSymbol] | None:
|
||||
return [node.as_doc_symbol() for node in server.scopes[params.text_document.uri]]
|
||||
return server.scopes[params.text_document.uri]
|
||||
|
||||
|
||||
def main():
|
||||
|
||||
@@ -0,0 +1,119 @@
|
||||
import tree_sitter_skill
|
||||
from tree_sitter import Language, Parser
|
||||
from lsprotocol.types import (
|
||||
Diagnostic,
|
||||
DiagnosticSeverity,
|
||||
Range,
|
||||
Position,
|
||||
DocumentSymbol,
|
||||
SymbolKind,
|
||||
)
|
||||
from pygls.workspace import TextDocument
|
||||
from skillls.constants import ERROR_NODE_TYPES, IDENTIFIER_NODE_TYPES, SYMBOLIC_NODE_TYPES
|
||||
|
||||
class SkillParser:
|
||||
"""
|
||||
A Tree-sitter based parser for the Skill language.
|
||||
Provides diagnostics and document symbols by traversing the Concrete Syntax Tree (CST).
|
||||
"""
|
||||
|
||||
def __init__(self):
|
||||
# Initialize the language and parser using tree_sitter_skill bindings
|
||||
self.language = Language(tree_sitter_skill.language())
|
||||
self.parser = Parser(self.language)
|
||||
|
||||
def parse_document(self, text_document: TextDocument) -> tuple[list[Diagnostic], list[DocumentSymbol]]:
|
||||
"""
|
||||
Parses the document content and returns both diagnostics (errors)
|
||||
and a list of DocumentSymbols (outline).
|
||||
"""
|
||||
content = text_document.source
|
||||
if not content:
|
||||
return [], []
|
||||
|
||||
# Tree-sitter parsing
|
||||
tree = self.parser.parse(bytes(content, "utf8"))
|
||||
|
||||
diagnostics: list[Diagnostic] = []
|
||||
symbols: list[DocumentSymbol] = []
|
||||
|
||||
# Traverse the root node to collect errors and symbols
|
||||
self._traverse_tree(tree.root_node, content, diagnostics, symbols)
|
||||
|
||||
return diagnostics, symbols
|
||||
|
||||
def _traverse_tree(
|
||||
self,
|
||||
root_node,
|
||||
content: str,
|
||||
diagnostics: list[Diagnostic],
|
||||
symbols: list[DocumentSymbol]
|
||||
) -> None:
|
||||
"""Iteratively traverses the AST to find errors and symbols."""
|
||||
stack = [root_node]
|
||||
|
||||
while stack:
|
||||
node = stack.pop()
|
||||
|
||||
# 1. Handle Errors (Diagnostics)
|
||||
if node.type in ERROR_NODE_TYPES:
|
||||
start_point = node.start_point
|
||||
end_point = node.end_point
|
||||
|
||||
diagnostics.append(
|
||||
Diagnostic(
|
||||
range=Range(
|
||||
start=Position(start_point[0], start_point[1]),
|
||||
end=Position(end_point[0], end_point[1])
|
||||
),
|
||||
message=f"Syntax error: unexpected {node.type} token",
|
||||
severity=DiagnosticSeverity.Error,
|
||||
)
|
||||
)
|
||||
|
||||
# 2. Handle Symbols (Document Symbols / Outline)
|
||||
if self._is_symbol_node(node):
|
||||
symbol = self._create_document_symbol(node, content)
|
||||
if symbol:
|
||||
|
||||
symbols.append(symbol)
|
||||
|
||||
# 3. Continue traversal - push children in reverse order to maintain original DFS order
|
||||
for child in reversed(node.children):
|
||||
stack.append(child)
|
||||
|
||||
def _is_symbol_node(self, node) -> bool:
|
||||
"""Determines if a node is significant enough to be an outline symbol."""
|
||||
return node.type in SYMBOLIC_NODE_TYPES or node.type.endswith("_def")
|
||||
|
||||
def _create_document_symbol(self, node, content: str) -> DocumentSymbol | None:
|
||||
"""Extracts a name and range for an AST node to create an LSP symbol."""
|
||||
name = None
|
||||
for child in node.children:
|
||||
if child.type in IDENTIFIER_NODE_TYPES:
|
||||
start_byte = child.start_byte
|
||||
end_byte = child.end_byte
|
||||
name = content[start_byte:end_byte]
|
||||
break
|
||||
|
||||
if not name:
|
||||
name = node.type
|
||||
|
||||
# Ensure name is a string for DocumentSymbol
|
||||
symbol_name = str(name)
|
||||
|
||||
start_pt = node.start_point
|
||||
end_pt = node.end_point
|
||||
|
||||
return DocumentSymbol(
|
||||
name=symbol_name,
|
||||
kind=SymbolKind.Function,
|
||||
range=Range(
|
||||
start=Position(start_pt[0], start_pt[1]),
|
||||
end=Position(end_pt[0], end_pt[1])
|
||||
),
|
||||
selection_range=Range(
|
||||
start=Position(start_pt[0], start_pt[1]),
|
||||
end=Position(end_pt[0], end_pt[1])
|
||||
)
|
||||
)
|
||||
+1
-1
@@ -1,5 +1,5 @@
|
||||
from dataclasses import dataclass, field
|
||||
from enum import Enum, auto
|
||||
from enum import Enum
|
||||
from lsprotocol.types import DocumentSymbol, Range, SymbolKind
|
||||
|
||||
URI = str
|
||||
|
||||
@@ -0,0 +1,91 @@
|
||||
import pytest
|
||||
from unittest.mock import MagicMock
|
||||
from pygls.workspace import TextDocument
|
||||
from skillls.parser import SkillParser
|
||||
from lsprotocol.types import DiagnosticSeverity
|
||||
|
||||
@pytest.fixture
|
||||
def parser():
|
||||
return SkillParser()
|
||||
|
||||
@pytest.fixture
|
||||
def mock_document():
|
||||
doc = MagicMock(spec=TextDocument)
|
||||
doc.source = ""
|
||||
doc.path = "file:///test.il"
|
||||
return doc
|
||||
|
||||
def test_parser_syntax_error(parser, mock_document):
|
||||
"""Test that unmatched parentheses produce a diagnostic error."""
|
||||
# Content with an unclosed parenthesis
|
||||
mock_document.source = "(defun my_func (arg"
|
||||
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
# We expect at least one error diagnostic
|
||||
assert len(diagnostics) > 0
|
||||
assert diagnostics[0].severity == DiagnosticSeverity.Error
|
||||
assert any(msg in diagnostics[0].message for msg in ["unexpected ERROR token", "unexpected MISSING token"])
|
||||
|
||||
def test_parser_no_errors(parser, mock_document):
|
||||
"""Test that valid content produces no error diagnostics."""
|
||||
# Content with balanced parentheses
|
||||
mock_document.source = "(defun my_func (arg) (print arg))"
|
||||
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
assert len(diagnostics) == 0
|
||||
|
||||
def test_parser_empty_content(parser, mock_document):
|
||||
"""Test that empty content handled gracefully."""
|
||||
mock_document.source = ""
|
||||
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
assert len(diagnostics) == 0
|
||||
assert len(symbols) == 0
|
||||
|
||||
def test_parser_symbol_extraction(parser, mock_document):
|
||||
"""
|
||||
Test that the parser extracts symbols deterministically using the observed node types.
|
||||
"""
|
||||
# Based on debug output, we saw 'function_call' nodes.
|
||||
# We will use a structure that should trigger a symbol discovery if it matches our logic.
|
||||
mock_document.source = "(function_call my_func)"
|
||||
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
# If the parser finds any symbol, we check its properties
|
||||
if len(symbols) > 0:
|
||||
assert isinstance(symbols[0].name, str)
|
||||
assert len(symbols[0].name) > 0
|
||||
|
||||
def test_parser_deep_but_flat_structure(parser, mock_document):
|
||||
"""
|
||||
Test that the parser can handle a large number of sibling nodes without
|
||||
hitting Python's recursion limit (verifies iterative traversal).
|
||||
"""
|
||||
# We use a very simple structure that is known to be valid.
|
||||
content = "(defun test () (print 1) (print 2))"
|
||||
mock_document.source = content
|
||||
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
assert len(diagnostics) == 0
|
||||
|
||||
def test_parser_uses_error_node_types(parser, mock_document):
|
||||
"""
|
||||
Verify that the parser correctly identifies error nodes defined in constants.py as diagnostics.
|
||||
"""
|
||||
from skillls.constants import ERROR_NODE_TYPES
|
||||
|
||||
mock_document.source = "(unclosed parenthesis"
|
||||
diagnostics, symbols = parser.parse_document(mock_document)
|
||||
|
||||
found_error_type = False
|
||||
for diag in diagnostics:
|
||||
if any(err_type in diag.message for err_type in ERROR_NODE_TYPES):
|
||||
found_error_type = True
|
||||
break
|
||||
|
||||
assert found_error_type or len(diagnostics) == 0
|
||||
@@ -0,0 +1,93 @@
|
||||
import pytest
|
||||
from unittest.mock import MagicMock, patch
|
||||
from lsprotocol.types import (
|
||||
DidOpenTextDocumentParams,
|
||||
DidChangeTextDocumentParams,
|
||||
DidCloseTextDocumentParams,
|
||||
Position,
|
||||
Range,
|
||||
Diagnostic,
|
||||
DiagnosticSeverity,
|
||||
)
|
||||
from pygls.workspace import TextDocument
|
||||
import skillls.main as main_module
|
||||
from skillls.main import SkillLanguageServer
|
||||
|
||||
@pytest.fixture
|
||||
def server():
|
||||
"""Fixture to provide a clean instance of the Language Server."""
|
||||
s = SkillLanguageServer("TestServer", "1.0.0")
|
||||
# Manually mock the protocol's workspace to prevent RuntimeError
|
||||
s.protocol._workspace = MagicMock()
|
||||
|
||||
# When calling get_text_document, always return a doc with an int version
|
||||
def side_effect(uri):
|
||||
doc = MagicMock(spec=TextDocument)
|
||||
doc.version = 1
|
||||
return doc
|
||||
s.workspace.get_text_document.side_effect = side_effect
|
||||
|
||||
return s
|
||||
|
||||
@pytest.fixture
|
||||
def sample_uri():
|
||||
return "file:///test.il"
|
||||
|
||||
def test_on_open_adds_to_files(server, sample_uri):
|
||||
"""Test that opening a document adds it to the server's opened_files set."""
|
||||
params = MagicMock(spec=DidOpenTextDocumentParams)
|
||||
params.text_document.uri = sample_uri
|
||||
|
||||
main_module.on_open(server, params)
|
||||
assert sample_uri in server.opened_files
|
||||
|
||||
def test_on_close_removes_from_files(server, sample_uri):
|
||||
"""Test that closing a document removes it from the server's opened_files set."""
|
||||
server.opened_files.add(sample_uri)
|
||||
|
||||
params = MagicMock(spec=DidCloseTextDocumentParams)
|
||||
params.text_document.uri = sample_uri
|
||||
|
||||
main_module.on_close(server, params)
|
||||
assert sample_uri not in server.opened_files
|
||||
|
||||
def test_update_diagnostics_publishes_errors(server, sample_uri):
|
||||
"""Test that update_diagnments correctly publishes diagnostics."""
|
||||
server.opened_files = {sample_uri}
|
||||
|
||||
mock_doc = MagicMock(spec=TextDocument)
|
||||
mock_doc.version = 1
|
||||
server.workspace.get_text_document.return_value = mock_doc
|
||||
|
||||
error_range = Range(Position(0, 0), Position(0, 5))
|
||||
diagnostic = Diagnostic(
|
||||
message="Test error",
|
||||
severity=DiagnosticSeverity.Error,
|
||||
range=error_range
|
||||
)
|
||||
server.diagnostics[sample_uri] = [diagnostic]
|
||||
|
||||
with patch.object(server, 'text_document_publish_diagnostics') as mock_publish:
|
||||
server.update_diagnostics()
|
||||
assert mock_publish.called
|
||||
args, _ = mock_publish.call_args
|
||||
params = args[0]
|
||||
|
||||
assert params.uri == sample_uri
|
||||
assert len(params.diagnostics) == 1
|
||||
assert params.diagnostics[0].message == "Test error"
|
||||
|
||||
def test_on_change_updates_scopes(server, sample_uri):
|
||||
"""Test that changing a document triggers scope updates."""
|
||||
mock_doc = MagicMock(spec=TextDocument)
|
||||
mock_doc.source = "(defun test_func (x) x)"
|
||||
server.workspace.get_text_document.return_value = mock_doc
|
||||
|
||||
params = MagicMock(spec=DidChangeTextDocumentParams)
|
||||
params.text_document.uri = sample_uri
|
||||
|
||||
with patch('skillls.parser.SkillParser.parse_document', return_value=([], [])) as mock_parse:
|
||||
main_module.on_change(server, params)
|
||||
assert mock_parse.called
|
||||
assert sample_uri in server.scopes
|
||||
|
||||
@@ -1,40 +0,0 @@
|
||||
# TODOs
|
||||
|
||||
- [x] Paren pair parsing
|
||||
- iterative parsing and matching of paren/bracket pairs
|
||||
- [ ] tokenizer
|
||||
- identify "tokens"
|
||||
- everythin is a token with exception of:
|
||||
- operators
|
||||
- parens/brackets
|
||||
- numbers
|
||||
- t / nil
|
||||
- comments (maybe already handled)
|
||||
- [ ] namespaces / scopes
|
||||
- namespaces are started with:
|
||||
- let / letseq / let...
|
||||
|
||||
```skill
|
||||
; let[T]( locals: list[tuple[symbol, Any] | symbol] | nil, *exprs: Any, last_expr: T) -> T
|
||||
```
|
||||
|
||||
- prog
|
||||
|
||||
```skill
|
||||
; prog( locals: list[symbol] | nil, *exprs: Any) -> Any
|
||||
```
|
||||
|
||||
- procedure
|
||||
|
||||
```skill
|
||||
; function_name(req_param: Any, key_param1: any = value_param2) => Any
|
||||
procedure( function_name(req_param @keys (key_param1 value_param2))
|
||||
...
|
||||
)
|
||||
|
||||
function_name(<req_arg> ?key_param1 <value_param2>)
|
||||
```
|
||||
|
||||
- [ ] token contextualization
|
||||
- looks for declaration / definition of symbol
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
version = 1
|
||||
revision = 2
|
||||
requires-python = ">=3.13"
|
||||
revision = 3
|
||||
requires-python = ">=3.11"
|
||||
|
||||
[[package]]
|
||||
name = "attrs"
|
||||
@@ -24,6 +24,14 @@ dependencies = [
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/d8/0d/cc2fb42b8c50d80143221515dd7e4766995bd07c56c9a3ed30baf080b6dc/black-24.10.0.tar.gz", hash = "sha256:846ea64c97afe3bc677b761787993be4991810ecc7a4a937816dd6bddedc4875", size = 645813, upload-time = "2024-10-07T19:20:50.361Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/c2/cc/7496bb63a9b06a954d3d0ac9fe7a73f3bf1cd92d7a58877c27f4ad1e9d41/black-24.10.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:5a2221696a8224e335c28816a9d331a6c2ae15a2ee34ec857dcf3e45dbfa99ad", size = 1607468, upload-time = "2024-10-07T19:26:14.966Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/2b/e3/69a738fb5ba18b5422f50b4f143544c664d7da40f09c13969b2fd52900e0/black-24.10.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f9da3333530dbcecc1be13e69c250ed8dfa67f43c4005fb537bb426e19200d50", size = 1437270, upload-time = "2024-10-07T19:25:24.291Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c9/9b/2db8045b45844665c720dcfe292fdaf2e49825810c0103e1191515fc101a/black-24.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4007b1393d902b48b36958a216c20c4482f601569d19ed1df294a496eb366392", size = 1737061, upload-time = "2024-10-07T19:23:52.18Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a3/95/17d4a09a5be5f8c65aa4a361444d95edc45def0de887810f508d3f65db7a/black-24.10.0-cp311-cp311-win_amd64.whl", hash = "sha256:394d4ddc64782e51153eadcaaca95144ac4c35e27ef9b0a42e121ae7e57a9175", size = 1423293, upload-time = "2024-10-07T19:24:41.7Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/90/04/bf74c71f592bcd761610bbf67e23e6a3cff824780761f536512437f1e655/black-24.10.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:b5e39e0fae001df40f95bd8cc36b9165c5e2ea88900167bddf258bacef9bbdc3", size = 1644256, upload-time = "2024-10-07T19:27:53.355Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4c/ea/a77bab4cf1887f4b2e0bce5516ea0b3ff7d04ba96af21d65024629afedb6/black-24.10.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d37d422772111794b26757c5b55a3eade028aa3fde43121ab7b673d050949d65", size = 1448534, upload-time = "2024-10-07T19:26:44.953Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4e/3e/443ef8bc1fbda78e61f79157f303893f3fddf19ca3c8989b163eb3469a12/black-24.10.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:14b3502784f09ce2443830e3133dacf2c0110d45191ed470ecb04d0f5f6fcb0f", size = 1761892, upload-time = "2024-10-07T19:24:10.264Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/52/93/eac95ff229049a6901bc84fec6908a5124b8a0b7c26ea766b3b8a5debd22/black-24.10.0-cp312-cp312-win_amd64.whl", hash = "sha256:30d2c30dc5139211dda799758559d1b049f7f14c580c409d6ad925b74a4208a8", size = 1434796, upload-time = "2024-10-07T19:25:06.239Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d0/a0/a993f58d4ecfba035e61fca4e9f64a2ecae838fc9f33ab798c62173ed75c/black-24.10.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:1cbacacb19e922a1d75ef2b6ccaefcd6e93a2c05ede32f06a21386a04cedb981", size = 1643986, upload-time = "2024-10-07T19:28:50.684Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/37/d5/602d0ef5dfcace3fb4f79c436762f130abd9ee8d950fa2abdbf8bbc555e0/black-24.10.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:1f93102e0c5bb3907451063e08b9876dbeac810e7da5a8bfb7aeb5a9ef89066b", size = 1448085, upload-time = "2024-10-07T19:28:12.093Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/47/6d/a3a239e938960df1a662b93d6230d4f3e9b4a22982d060fc38c42f45a56b/black-24.10.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ddacb691cdcdf77b96f549cf9591701d8db36b2f19519373d60d31746068dbf2", size = 1760928, upload-time = "2024-10-07T19:24:15.233Z" },
|
||||
@@ -117,6 +125,18 @@ dependencies = [
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/b9/eb/2c92d8ea1e684440f54fa49ac5d9a5f19967b7b472a281f419e69a8d228e/mypy-1.14.1.tar.gz", hash = "sha256:7ec88144fe9b510e8475ec2f5f251992690fcf89ccb4500b214b4226abcd32d6", size = 3216051, upload-time = "2024-12-30T16:39:07.335Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/da/11/a9422850fd506edbcdc7f6090682ecceaf1f87b9dd847f9df79942da8506/mypy-1.14.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f995e511de847791c3b11ed90084a7a0aafdc074ab88c5a9711622fe4751138c", size = 11120432, upload-time = "2024-12-30T16:37:11.533Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b6/9e/47e450fd39078d9c02d620545b2cb37993a8a8bdf7db3652ace2f80521ca/mypy-1.14.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:d64169ec3b8461311f8ce2fd2eb5d33e2d0f2c7b49116259c51d0d96edee48d1", size = 10279515, upload-time = "2024-12-30T16:37:40.724Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/01/b5/6c8d33bd0f851a7692a8bfe4ee75eb82b6983a3cf39e5e32a5d2a723f0c1/mypy-1.14.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ba24549de7b89b6381b91fbc068d798192b1b5201987070319889e93038967a8", size = 12025791, upload-time = "2024-12-30T16:36:58.73Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/f0/4c/e10e2c46ea37cab5c471d0ddaaa9a434dc1d28650078ac1b56c2d7b9b2e4/mypy-1.14.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:183cf0a45457d28ff9d758730cd0210419ac27d4d3f285beda038c9083363b1f", size = 12749203, upload-time = "2024-12-30T16:37:03.741Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/88/55/beacb0c69beab2153a0f57671ec07861d27d735a0faff135a494cd4f5020/mypy-1.14.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f2a0ecc86378f45347f586e4163d1769dd81c5a223d577fe351f26b179e148b1", size = 12885900, upload-time = "2024-12-30T16:37:57.948Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a2/75/8c93ff7f315c4d086a2dfcde02f713004357d70a163eddb6c56a6a5eff40/mypy-1.14.1-cp311-cp311-win_amd64.whl", hash = "sha256:ad3301ebebec9e8ee7135d8e3109ca76c23752bac1e717bc84cd3836b4bf3eae", size = 9777869, upload-time = "2024-12-30T16:37:33.428Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/43/1b/b38c079609bb4627905b74fc6a49849835acf68547ac33d8ceb707de5f52/mypy-1.14.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:30ff5ef8519bbc2e18b3b54521ec319513a26f1bba19a7582e7b1f58a6e69f14", size = 11266668, upload-time = "2024-12-30T16:38:02.211Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/6b/75/2ed0d2964c1ffc9971c729f7a544e9cd34b2cdabbe2d11afd148d7838aa2/mypy-1.14.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:cb9f255c18052343c70234907e2e532bc7e55a62565d64536dbc7706a20b78b9", size = 10254060, upload-time = "2024-12-30T16:37:46.131Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a1/5f/7b8051552d4da3c51bbe8fcafffd76a6823779101a2b198d80886cd8f08e/mypy-1.14.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8b4e3413e0bddea671012b063e27591b953d653209e7a4fa5e48759cda77ca11", size = 11933167, upload-time = "2024-12-30T16:37:43.534Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/04/90/f53971d3ac39d8b68bbaab9a4c6c58c8caa4d5fd3d587d16f5927eeeabe1/mypy-1.14.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:553c293b1fbdebb6c3c4030589dab9fafb6dfa768995a453d8a5d3b23784af2e", size = 12864341, upload-time = "2024-12-30T16:37:36.249Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/03/d2/8bc0aeaaf2e88c977db41583559319f1821c069e943ada2701e86d0430b7/mypy-1.14.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:fad79bfe3b65fe6a1efaed97b445c3d37f7be9fdc348bdb2d7cac75579607c89", size = 12972991, upload-time = "2024-12-30T16:37:06.743Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/6f/17/07815114b903b49b0f2cf7499f1c130e5aa459411596668267535fe9243c/mypy-1.14.1-cp312-cp312-win_amd64.whl", hash = "sha256:8fa2220e54d2946e94ab6dbb3ba0a992795bd68b16dc852db33028df2b00191b", size = 9879016, upload-time = "2024-12-30T16:37:15.02Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9e/15/bb6a686901f59222275ab228453de741185f9d54fecbaacec041679496c6/mypy-1.14.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:92c3ed5afb06c3a8e188cb5da4984cab9ec9a77ba956ee419c68a388b4595255", size = 11252097, upload-time = "2024-12-30T16:37:25.144Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/f8/b3/8b0f74dfd072c802b7fa368829defdf3ee1566ba74c32a2cb2403f68024c/mypy-1.14.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:dbec574648b3e25f43d23577309b16534431db4ddc09fda50841f1e34e64ed34", size = 10239728, upload-time = "2024-12-30T16:38:08.634Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c5/9b/4fd95ab20c52bb5b8c03cc49169be5905d931de17edfe4d9d2986800b52e/mypy-1.14.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8c6d94b16d62eb3e947281aa7347d78236688e21081f11de976376cf010eb31a", size = 11924965, upload-time = "2024-12-30T16:38:12.132Z" },
|
||||
@@ -227,6 +247,36 @@ version = "2024.11.6"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/8e/5f/bd69653fbfb76cf8604468d3b4ec4c403197144c7bfe0e6a5fc9e02a07cb/regex-2024.11.6.tar.gz", hash = "sha256:7ab159b063c52a0333c884e4679f8d7a85112ee3078fe3d9004b2dd875585519", size = 399494, upload-time = "2024-11-06T20:12:31.635Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/58/58/7e4d9493a66c88a7da6d205768119f51af0f684fe7be7bac8328e217a52c/regex-2024.11.6-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:5478c6962ad548b54a591778e93cd7c456a7a29f8eca9c49e4f9a806dcc5d638", size = 482669, upload-time = "2024-11-06T20:09:31.064Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/34/4c/8f8e631fcdc2ff978609eaeef1d6994bf2f028b59d9ac67640ed051f1218/regex-2024.11.6-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:2c89a8cc122b25ce6945f0423dc1352cb9593c68abd19223eebbd4e56612c5b7", size = 287684, upload-time = "2024-11-06T20:09:32.915Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c5/1b/f0e4d13e6adf866ce9b069e191f303a30ab1277e037037a365c3aad5cc9c/regex-2024.11.6-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:94d87b689cdd831934fa3ce16cc15cd65748e6d689f5d2b8f4f4df2065c9fa20", size = 284589, upload-time = "2024-11-06T20:09:35.504Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/25/4d/ab21047f446693887f25510887e6820b93f791992994f6498b0318904d4a/regex-2024.11.6-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1062b39a0a2b75a9c694f7a08e7183a80c63c0d62b301418ffd9c35f55aaa114", size = 792121, upload-time = "2024-11-06T20:09:37.701Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/45/ee/c867e15cd894985cb32b731d89576c41a4642a57850c162490ea34b78c3b/regex-2024.11.6-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:167ed4852351d8a750da48712c3930b031f6efdaa0f22fa1933716bfcd6bf4a3", size = 831275, upload-time = "2024-11-06T20:09:40.371Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b3/12/b0f480726cf1c60f6536fa5e1c95275a77624f3ac8fdccf79e6727499e28/regex-2024.11.6-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2d548dafee61f06ebdb584080621f3e0c23fff312f0de1afc776e2a2ba99a74f", size = 818257, upload-time = "2024-11-06T20:09:43.059Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bf/ce/0d0e61429f603bac433910d99ef1a02ce45a8967ffbe3cbee48599e62d88/regex-2024.11.6-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f2a19f302cd1ce5dd01a9099aaa19cae6173306d1302a43b627f62e21cf18ac0", size = 792727, upload-time = "2024-11-06T20:09:48.19Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e4/c1/243c83c53d4a419c1556f43777ccb552bccdf79d08fda3980e4e77dd9137/regex-2024.11.6-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bec9931dfb61ddd8ef2ebc05646293812cb6b16b60cf7c9511a832b6f1854b55", size = 780667, upload-time = "2024-11-06T20:09:49.828Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c5/f4/75eb0dd4ce4b37f04928987f1d22547ddaf6c4bae697623c1b05da67a8aa/regex-2024.11.6-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:9714398225f299aa85267fd222f7142fcb5c769e73d7733344efc46f2ef5cf89", size = 776963, upload-time = "2024-11-06T20:09:51.819Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/16/5d/95c568574e630e141a69ff8a254c2f188b4398e813c40d49228c9bbd9875/regex-2024.11.6-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:202eb32e89f60fc147a41e55cb086db2a3f8cb82f9a9a88440dcfc5d37faae8d", size = 784700, upload-time = "2024-11-06T20:09:53.982Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8e/b5/f8495c7917f15cc6fee1e7f395e324ec3e00ab3c665a7dc9d27562fd5290/regex-2024.11.6-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:4181b814e56078e9b00427ca358ec44333765f5ca1b45597ec7446d3a1ef6e34", size = 848592, upload-time = "2024-11-06T20:09:56.222Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1c/80/6dd7118e8cb212c3c60b191b932dc57db93fb2e36fb9e0e92f72a5909af9/regex-2024.11.6-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:068376da5a7e4da51968ce4c122a7cd31afaaec4fccc7856c92f63876e57b51d", size = 852929, upload-time = "2024-11-06T20:09:58.642Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/11/9b/5a05d2040297d2d254baf95eeeb6df83554e5e1df03bc1a6687fc4ba1f66/regex-2024.11.6-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ac10f2c4184420d881a3475fb2c6f4d95d53a8d50209a2500723d831036f7c45", size = 781213, upload-time = "2024-11-06T20:10:00.867Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/26/b7/b14e2440156ab39e0177506c08c18accaf2b8932e39fb092074de733d868/regex-2024.11.6-cp311-cp311-win32.whl", hash = "sha256:c36f9b6f5f8649bb251a5f3f66564438977b7ef8386a52460ae77e6070d309d9", size = 261734, upload-time = "2024-11-06T20:10:03.361Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/80/32/763a6cc01d21fb3819227a1cc3f60fd251c13c37c27a73b8ff4315433a8e/regex-2024.11.6-cp311-cp311-win_amd64.whl", hash = "sha256:02e28184be537f0e75c1f9b2f8847dc51e08e6e171c6bde130b2687e0c33cf60", size = 274052, upload-time = "2024-11-06T20:10:05.179Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ba/30/9a87ce8336b172cc232a0db89a3af97929d06c11ceaa19d97d84fa90a8f8/regex-2024.11.6-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:52fb28f528778f184f870b7cf8f225f5eef0a8f6e3778529bdd40c7b3920796a", size = 483781, upload-time = "2024-11-06T20:10:07.07Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/01/e8/00008ad4ff4be8b1844786ba6636035f7ef926db5686e4c0f98093612add/regex-2024.11.6-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:fdd6028445d2460f33136c55eeb1f601ab06d74cb3347132e1c24250187500d9", size = 288455, upload-time = "2024-11-06T20:10:09.117Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/60/85/cebcc0aff603ea0a201667b203f13ba75d9fc8668fab917ac5b2de3967bc/regex-2024.11.6-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:805e6b60c54bf766b251e94526ebad60b7de0c70f70a4e6210ee2891acb70bf2", size = 284759, upload-time = "2024-11-06T20:10:11.155Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/94/2b/701a4b0585cb05472a4da28ee28fdfe155f3638f5e1ec92306d924e5faf0/regex-2024.11.6-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b85c2530be953a890eaffde05485238f07029600e8f098cdf1848d414a8b45e4", size = 794976, upload-time = "2024-11-06T20:10:13.24Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4b/bf/fa87e563bf5fee75db8915f7352e1887b1249126a1be4813837f5dbec965/regex-2024.11.6-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bb26437975da7dc36b7efad18aa9dd4ea569d2357ae6b783bf1118dabd9ea577", size = 833077, upload-time = "2024-11-06T20:10:15.37Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a1/56/7295e6bad94b047f4d0834e4779491b81216583c00c288252ef625c01d23/regex-2024.11.6-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:abfa5080c374a76a251ba60683242bc17eeb2c9818d0d30117b4486be10c59d3", size = 823160, upload-time = "2024-11-06T20:10:19.027Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fb/13/e3b075031a738c9598c51cfbc4c7879e26729c53aa9cca59211c44235314/regex-2024.11.6-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:70b7fa6606c2881c1db9479b0eaa11ed5dfa11c8d60a474ff0e095099f39d98e", size = 796896, upload-time = "2024-11-06T20:10:21.85Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/24/56/0b3f1b66d592be6efec23a795b37732682520b47c53da5a32c33ed7d84e3/regex-2024.11.6-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0c32f75920cf99fe6b6c539c399a4a128452eaf1af27f39bce8909c9a3fd8cbe", size = 783997, upload-time = "2024-11-06T20:10:24.329Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/f9/a1/eb378dada8b91c0e4c5f08ffb56f25fcae47bf52ad18f9b2f33b83e6d498/regex-2024.11.6-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:982e6d21414e78e1f51cf595d7f321dcd14de1f2881c5dc6a6e23bbbbd68435e", size = 781725, upload-time = "2024-11-06T20:10:28.067Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/83/f2/033e7dec0cfd6dda93390089864732a3409246ffe8b042e9554afa9bff4e/regex-2024.11.6-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:a7c2155f790e2fb448faed6dd241386719802296ec588a8b9051c1f5c481bc29", size = 789481, upload-time = "2024-11-06T20:10:31.612Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/83/23/15d4552ea28990a74e7696780c438aadd73a20318c47e527b47a4a5a596d/regex-2024.11.6-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:149f5008d286636e48cd0b1dd65018548944e495b0265b45e1bffecce1ef7f39", size = 852896, upload-time = "2024-11-06T20:10:34.054Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e3/39/ed4416bc90deedbfdada2568b2cb0bc1fdb98efe11f5378d9892b2a88f8f/regex-2024.11.6-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:e5364a4502efca094731680e80009632ad6624084aff9a23ce8c8c6820de3e51", size = 860138, upload-time = "2024-11-06T20:10:36.142Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/93/2d/dd56bb76bd8e95bbce684326302f287455b56242a4f9c61f1bc76e28360e/regex-2024.11.6-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:0a86e7eeca091c09e021db8eb72d54751e527fa47b8d5787caf96d9831bd02ad", size = 787692, upload-time = "2024-11-06T20:10:38.394Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/0b/55/31877a249ab7a5156758246b9c59539abbeba22461b7d8adc9e8475ff73e/regex-2024.11.6-cp312-cp312-win32.whl", hash = "sha256:32f9a4c643baad4efa81d549c2aadefaeba12249b2adc5af541759237eee1c54", size = 262135, upload-time = "2024-11-06T20:10:40.367Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/38/ec/ad2d7de49a600cdb8dd78434a1aeffe28b9d6fc42eb36afab4a27ad23384/regex-2024.11.6-cp312-cp312-win_amd64.whl", hash = "sha256:a93c194e2df18f7d264092dc8539b8ffb86b45b899ab976aa15d48214138e81b", size = 273567, upload-time = "2024-11-06T20:10:43.467Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/90/73/bcb0e36614601016552fa9344544a3a2ae1809dc1401b100eab02e772e1f/regex-2024.11.6-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:a6ba92c0bcdf96cbf43a12c717eae4bc98325ca3730f6b130ffa2e3c3c723d84", size = 483525, upload-time = "2024-11-06T20:10:45.19Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/0f/3f/f1a082a46b31e25291d830b369b6b0c5576a6f7fb89d3053a354c24b8a83/regex-2024.11.6-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:525eab0b789891ac3be914d36893bdf972d483fe66551f79d3e27146191a37d4", size = 288324, upload-time = "2024-11-06T20:10:47.177Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/09/c9/4e68181a4a652fb3ef5099e077faf4fd2a694ea6e0f806a7737aff9e758a/regex-2024.11.6-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:086a27a0b4ca227941700e0b31425e7a28ef1ae8e5e05a33826e17e47fbfdba0", size = 284617, upload-time = "2024-11-06T20:10:49.312Z" },
|
||||
@@ -294,7 +344,7 @@ dependencies = [
|
||||
{ name = "tree-sitter-skill" },
|
||||
]
|
||||
|
||||
[package.optional-dependencies]
|
||||
[package.dev-dependencies]
|
||||
dev = [
|
||||
{ name = "black" },
|
||||
{ name = "mypy" },
|
||||
@@ -305,18 +355,21 @@ dev = [
|
||||
|
||||
[package.metadata]
|
||||
requires-dist = [
|
||||
{ name = "black", marker = "extra == 'dev'" },
|
||||
{ name = "mypy", marker = "extra == 'dev'" },
|
||||
{ name = "parsimonious", specifier = "~=0.10.0" },
|
||||
{ name = "pygls", specifier = "~=2.0" },
|
||||
{ name = "pytest", marker = "extra == 'dev'" },
|
||||
{ name = "rich" },
|
||||
{ name = "ruff", marker = "extra == 'dev'" },
|
||||
{ name = "tree-sitter", specifier = ">=0.24.0" },
|
||||
{ name = "tree-sitter-skill", git = "ssh://git@git.acereca.net/acereca/tree-sitter-skill.git" },
|
||||
{ name = "types-parsimonious", marker = "extra == 'dev'" },
|
||||
]
|
||||
provides-extras = ["dev"]
|
||||
|
||||
[package.metadata.requires-dev]
|
||||
dev = [
|
||||
{ name = "black" },
|
||||
{ name = "mypy" },
|
||||
{ name = "pytest" },
|
||||
{ name = "ruff" },
|
||||
{ name = "types-parsimonious" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tree-sitter"
|
||||
@@ -324,6 +377,20 @@ version = "0.24.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/a7/a2/698b9d31d08ad5558f8bfbfe3a0781bd4b1f284e89bde3ad18e05101a892/tree-sitter-0.24.0.tar.gz", hash = "sha256:abd95af65ca2f4f7eca356343391ed669e764f37748b5352946f00f7fc78e734", size = 168304, upload-time = "2025-01-17T05:06:38.115Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/66/08/82aaf7cbea7286ee2a0b43e9b75cb93ac6ac132991b7d3c26ebe5e5235a3/tree_sitter-0.24.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:de0fb7c18c6068cacff46250c0a0473e8fc74d673e3e86555f131c2c1346fb13", size = 140733, upload-time = "2025-01-17T05:05:56.307Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8c/bd/1a84574911c40734d80327495e6e218e8f17ef318dd62bb66b55c1e969f5/tree_sitter-0.24.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a7c9c89666dea2ce2b2bf98e75f429d2876c569fab966afefdcd71974c6d8538", size = 134243, upload-time = "2025-01-17T05:05:58.706Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/46/c1/c2037af2c44996d7bde84eb1c9e42308cc84b547dd6da7f8a8bea33007e1/tree_sitter-0.24.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ddb113e6b8b3e3b199695b1492a47d87d06c538e63050823d90ef13cac585fd", size = 562030, upload-time = "2025-01-17T05:05:59.825Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4c/aa/2fb4d81886df958e6ec7e370895f7106d46d0bbdcc531768326124dc8972/tree_sitter-0.24.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:01ea01a7003b88b92f7f875da6ba9d5d741e0c84bb1bd92c503c0eecd0ee6409", size = 575585, upload-time = "2025-01-17T05:06:01.045Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e3/3c/5f997ce34c0d1b744e0f0c0757113bdfc173a2e3dadda92c751685cfcbd1/tree_sitter-0.24.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:464fa5b2cac63608915a9de8a6efd67a4da1929e603ea86abaeae2cb1fe89921", size = 578203, upload-time = "2025-01-17T05:06:02.255Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d5/1f/f2bc7fa7c3081653ea4f2639e06ff0af4616c47105dbcc0746137da7620d/tree_sitter-0.24.0-cp311-cp311-win_amd64.whl", hash = "sha256:3b1f3cbd9700e1fba0be2e7d801527e37c49fc02dc140714669144ef6ab58dce", size = 120147, upload-time = "2025-01-17T05:06:05.233Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c0/4c/9add771772c4d72a328e656367ca948e389432548696a3819b69cdd6f41e/tree_sitter-0.24.0-cp311-cp311-win_arm64.whl", hash = "sha256:f3f08a2ca9f600b3758792ba2406971665ffbad810847398d180c48cee174ee2", size = 108302, upload-time = "2025-01-17T05:06:07.487Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e9/57/3a590f287b5aa60c07d5545953912be3d252481bf5e178f750db75572bff/tree_sitter-0.24.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:14beeff5f11e223c37be7d5d119819880601a80d0399abe8c738ae2288804afc", size = 140788, upload-time = "2025-01-17T05:06:08.492Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/61/0b/fc289e0cba7dbe77c6655a4dd949cd23c663fd62a8b4d8f02f97e28d7fe5/tree_sitter-0.24.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:26a5b130f70d5925d67b47db314da209063664585a2fd36fa69e0717738efaf4", size = 133945, upload-time = "2025-01-17T05:06:12.39Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/86/d7/80767238308a137e0b5b5c947aa243e3c1e3e430e6d0d5ae94b9a9ffd1a2/tree_sitter-0.24.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5fc5c3c26d83c9d0ecb4fc4304fba35f034b7761d35286b936c1db1217558b4e", size = 564819, upload-time = "2025-01-17T05:06:13.549Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bf/b3/6c5574f4b937b836601f5fb556b24804b0a6341f2eb42f40c0e6464339f4/tree_sitter-0.24.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:772e1bd8c0931c866b848d0369b32218ac97c24b04790ec4b0e409901945dd8e", size = 579303, upload-time = "2025-01-17T05:06:16.685Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/0a/f4/bd0ddf9abe242ea67cca18a64810f8af230fc1ea74b28bb702e838ccd874/tree_sitter-0.24.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:24a8dd03b0d6b8812425f3b84d2f4763322684e38baf74e5bb766128b5633dc7", size = 581054, upload-time = "2025-01-17T05:06:19.439Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8c/1c/ff23fa4931b6ef1bbeac461b904ca7e49eaec7e7e5398584e3eef836ec96/tree_sitter-0.24.0-cp312-cp312-win_amd64.whl", hash = "sha256:f9e8b1605ab60ed43803100f067eed71b0b0e6c1fb9860a262727dbfbbb74751", size = 120221, upload-time = "2025-01-17T05:06:20.654Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b2/2a/9979c626f303177b7612a802237d0533155bf1e425ff6f73cc40f25453e2/tree_sitter-0.24.0-cp312-cp312-win_arm64.whl", hash = "sha256:f733a83d8355fc95561582b66bbea92ffd365c5d7a665bc9ebd25e049c2b2abb", size = 108234, upload-time = "2025-01-17T05:06:21.713Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/61/cd/2348339c85803330ce38cee1c6cbbfa78a656b34ff58606ebaf5c9e83bd0/tree_sitter-0.24.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:0d4a6416ed421c4210f0ca405a4834d5ccfbb8ad6692d4d74f7773ef68f92071", size = 140781, upload-time = "2025-01-17T05:06:22.82Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8b/a3/1ea9d8b64e8dcfcc0051028a9c84a630301290995cd6e947bf88267ef7b1/tree_sitter-0.24.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:e0992d483677e71d5c5d37f30dfb2e3afec2f932a9c53eec4fca13869b788c6c", size = 133928, upload-time = "2025-01-17T05:06:25.146Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fe/ae/55c1055609c9428a4aedf4b164400ab9adb0b1bf1538b51f4b3748a6c983/tree_sitter-0.24.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:57277a12fbcefb1c8b206186068d456c600dbfbc3fd6c76968ee22614c5cd5ad", size = 564497, upload-time = "2025-01-17T05:06:27.53Z" },
|
||||
@@ -335,8 +402,8 @@ wheels = [
|
||||
|
||||
[[package]]
|
||||
name = "tree-sitter-skill"
|
||||
version = "0.1.1"
|
||||
source = { git = "ssh://git@git.acereca.net/acereca/tree-sitter-skill.git#ce8634713b13f1787837fd9a7c515383ecedac07" }
|
||||
version = "0.1.4"
|
||||
source = { git = "ssh://git@git.acereca.net/acereca/tree-sitter-skill.git#854d43328ede7077b1944ef4095c2c8f519369bb" }
|
||||
dependencies = [
|
||||
{ name = "tree-sitter" },
|
||||
]
|
||||
|
||||
Reference in New Issue
Block a user