refactor(parser): remove inheritance from NodeVisitor

remove the parent NodeVisitor class from PythonParser and implement all custom recursive methods instead
This commit is contained in:
2026-05-25 21:42:04 +02:00
parent ecab1b74a4
commit c64d626d1c

View File

@@ -1,14 +1,20 @@
import ast import ast
from typing import Any, Optional from typing import Optional
from midas.ast.location import Location from midas.ast.location import Location
from midas.ast.python import ( from midas.ast.python import (
AssignExpr,
BaseType, BaseType,
ConstraintType, ConstraintType,
Expr,
ExpressionStmt,
FrameColumn, FrameColumn,
FrameType, FrameType,
Function, Function,
MidasType, MidasType,
Stmt,
TypeAssign,
) )
@@ -23,33 +29,66 @@ class UnsupportedSyntaxError(Exception):
) )
class PythonParser(ast.NodeVisitor): class PythonParser:
def __init__(self) -> None: def parse_module(self, node: ast.Module) -> list[Stmt]:
super().__init__() statements: list[Stmt] = []
for stmt in node.body:
parsed: None | Stmt | list[Stmt] = self.parse_stmt(stmt)
if isinstance(parsed, Stmt):
statements.append(parsed)
elif parsed is not None:
statements.extend(parsed)
return statements
self.annotations: list[tuple[str, Optional[MidasType]]] = [] def parse_stmt(self, node: ast.stmt) -> None | Stmt | list[Stmt]:
self.functions: list[Function] = []
def visit_AnnAssign(self, node: ast.AnnAssign) -> Any:
match node: match node:
case ast.AnnAssign( case ast.AnnAssign():
target=ast.Name(id=target), annotation=annotation, simple=1 return self.parse_annotation_assign(node)
):
self.annotations.append( case ast.FunctionDef():
(target, self._parse_type(annotation, root=True)) return self.parse_function(node)
)
case _:
print(f"Unsupported assignment: {ast.unparse(node)}")
return None
def parse_annotation_assign(self, node: ast.AnnAssign) -> list[Stmt]:
statements: list[Stmt] = []
loc: Location = Location.from_ast(node)
match node:
case ast.AnnAssign(
target=ast.Name(id=target),
annotation=annotation,
value=value,
simple=1,
):
type = self._parse_type(annotation, root=True)
if type is not None:
statements.append(
TypeAssign(
location=loc,
name=target,
type=type,
)
)
if value is not None:
parsed_value: Expr = self.parse_expr(value)
statements.append(
ExpressionStmt(
location=loc,
expr=AssignExpr(
location=loc,
name=target,
value=parsed_value,
),
)
)
case _: case _:
print(f"Unsupported annotation: {ast.unparse(node)}") print(f"Unsupported annotation: {ast.unparse(node)}")
return statements
def visit_FunctionDef(self, node: ast.FunctionDef) -> Any: def parse_function(self, node: ast.FunctionDef) -> Function:
self.functions.append(self._parse_function(node))
# Call visit on children to process body
# TODO: scope the resulting nodes to the function
self.generic_visit(node)
def _parse_function(self, node: ast.FunctionDef) -> Function:
loc: Location = Location.from_ast(node) loc: Location = Location.from_ast(node)
match node: match node:
case ast.FunctionDef( case ast.FunctionDef(
@@ -73,6 +112,8 @@ class PythonParser(ast.NodeVisitor):
kwonlyargs=parse_args(kwonlyargs), kwonlyargs=parse_args(kwonlyargs),
returns=self._parse_type(returns) if returns is not None else None, returns=self._parse_type(returns) if returns is not None else None,
) )
case _:
print(f"Unsupported function definition: {ast.unparse(node)}")
def _parse_function_argument(self, arg: ast.arg) -> Function.Argument: def _parse_function_argument(self, arg: ast.arg) -> Function.Argument:
loc: Location = Location.from_ast(arg) loc: Location = Location.from_ast(arg)
@@ -185,3 +226,6 @@ class PythonParser(ast.NodeVisitor):
case _: case _:
raise UnsupportedSyntaxError(column) raise UnsupportedSyntaxError(column)
def parse_expr(self, node: ast.expr) -> Expr:
raise NotImplementedError()