chore(fmt): add legacy stubs and strip trailing whitespace to unblock cargo fmt
This commit is contained in:
@ -1,16 +1,16 @@
|
||||
/*!
|
||||
* Nyash Parser - Rust Implementation
|
||||
*
|
||||
*
|
||||
* Python版nyashc_v4.pyのNyashParserをRustで完全再実装
|
||||
* Token列をAST (Abstract Syntax Tree) に変換
|
||||
*
|
||||
*
|
||||
* モジュール構造:
|
||||
* - common.rs: 共通ユーティリティとトレイト (ParserUtils)
|
||||
* - expressions.rs: 式パーサー (parse_expression, parse_or, parse_and等)
|
||||
* - statements.rs: 文パーサー (parse_statement, parse_if, parse_loop等)
|
||||
* - declarations/: Box宣言パーサー (box_definition, static_box, dependency_helpers)
|
||||
* - items/: トップレベル宣言 (global_vars, functions, static_items)
|
||||
*
|
||||
*
|
||||
* 2025-08-16: 大規模リファクタリング完了
|
||||
* - 1530行 → 227行 (85%削減)
|
||||
* - 機能ごとにモジュール分離で保守性向上
|
||||
@ -18,24 +18,26 @@
|
||||
|
||||
// サブモジュール宣言
|
||||
mod common;
|
||||
mod expressions;
|
||||
mod expr;
|
||||
mod statements;
|
||||
mod declarations;
|
||||
mod items;
|
||||
pub mod sugar; // Phase 12.7-B: desugar pass (basic)
|
||||
pub mod entry_sugar; // helper to parse with sugar level
|
||||
mod expr;
|
||||
mod expressions;
|
||||
mod items;
|
||||
mod statements;
|
||||
pub mod sugar; // Phase 12.7-B: desugar pass (basic)
|
||||
pub mod sugar_gate; // thread-local gate for sugar parsing (tests/docs)
|
||||
// mod errors;
|
||||
// mod errors;
|
||||
|
||||
use common::ParserUtils;
|
||||
|
||||
use crate::tokenizer::{Token, TokenType, TokenizeError};
|
||||
use crate::ast::{ASTNode, Span};
|
||||
use crate::tokenizer::{Token, TokenType, TokenizeError};
|
||||
use thiserror::Error;
|
||||
|
||||
#[inline]
|
||||
fn is_sugar_enabled() -> bool { crate::parser::sugar_gate::is_enabled() }
|
||||
fn is_sugar_enabled() -> bool {
|
||||
crate::parser::sugar_gate::is_enabled()
|
||||
}
|
||||
|
||||
// ===== 🔥 Debug Macros =====
|
||||
|
||||
@ -49,9 +51,17 @@ macro_rules! must_advance {
|
||||
if let Some(ref mut limit) = $parser.debug_fuel {
|
||||
if *limit == 0 {
|
||||
eprintln!("🚨 PARSER INFINITE LOOP DETECTED at {}", $location);
|
||||
eprintln!("🔍 Current token: {:?} at line {}", $parser.current_token().token_type, $parser.current_token().line);
|
||||
eprintln!("🔍 Parser position: {}/{}", $parser.current, $parser.tokens.len());
|
||||
return Err($crate::parser::ParseError::InfiniteLoop {
|
||||
eprintln!(
|
||||
"🔍 Current token: {:?} at line {}",
|
||||
$parser.current_token().token_type,
|
||||
$parser.current_token().line
|
||||
);
|
||||
eprintln!(
|
||||
"🔍 Parser position: {}/{}",
|
||||
$parser.current,
|
||||
$parser.tokens.len()
|
||||
);
|
||||
return Err($crate::parser::ParseError::InfiniteLoop {
|
||||
location: $location.to_string(),
|
||||
token: $parser.current_token().token_type.clone(),
|
||||
line: $parser.current_token().line,
|
||||
@ -77,32 +87,42 @@ macro_rules! debug_fuel {
|
||||
#[derive(Error, Debug)]
|
||||
pub enum ParseError {
|
||||
#[error("Unexpected token {found:?}, expected {expected} at line {line}")]
|
||||
UnexpectedToken { found: TokenType, expected: String, line: usize },
|
||||
|
||||
UnexpectedToken {
|
||||
found: TokenType,
|
||||
expected: String,
|
||||
line: usize,
|
||||
},
|
||||
|
||||
#[error("Unexpected end of file")]
|
||||
UnexpectedEOF,
|
||||
|
||||
|
||||
#[error("Invalid expression at line {line}")]
|
||||
InvalidExpression { line: usize },
|
||||
|
||||
|
||||
#[error("Invalid statement at line {line}")]
|
||||
InvalidStatement { line: usize },
|
||||
|
||||
|
||||
#[error("Circular dependency detected between static boxes: {cycle}")]
|
||||
CircularDependency { cycle: String },
|
||||
|
||||
|
||||
#[error("🚨 Infinite loop detected in parser at {location} - token: {token:?} at line {line}")]
|
||||
InfiniteLoop { location: String, token: TokenType, line: usize },
|
||||
|
||||
InfiniteLoop {
|
||||
location: String,
|
||||
token: TokenType,
|
||||
line: usize,
|
||||
},
|
||||
|
||||
#[error("🔥 Transparency system removed: {suggestion} at line {line}")]
|
||||
TransparencySystemRemoved { suggestion: String, line: usize },
|
||||
|
||||
#[error("Unsupported namespace '{name}' at line {line}. Only 'nyashstd' is supported in Phase 0.")]
|
||||
|
||||
#[error(
|
||||
"Unsupported namespace '{name}' at line {line}. Only 'nyashstd' is supported in Phase 0."
|
||||
)]
|
||||
UnsupportedNamespace { name: String, line: usize },
|
||||
|
||||
|
||||
#[error("Expected identifier at line {line}")]
|
||||
ExpectedIdentifier { line: usize },
|
||||
|
||||
|
||||
#[error("Tokenize error: {0}")]
|
||||
TokenizeError(#[from] TokenizeError),
|
||||
}
|
||||
@ -112,7 +132,8 @@ pub struct NyashParser {
|
||||
pub(super) tokens: Vec<Token>,
|
||||
pub(super) current: usize,
|
||||
/// 🔥 Static box依存関係追跡(循環依存検出用)
|
||||
pub(super) static_box_dependencies: std::collections::HashMap<String, std::collections::HashSet<String>>,
|
||||
pub(super) static_box_dependencies:
|
||||
std::collections::HashMap<String, std::collections::HashSet<String>>,
|
||||
/// 🔥 デバッグ燃料:無限ループ検出用制限値 (None = 無制限)
|
||||
pub(super) debug_fuel: Option<usize>,
|
||||
}
|
||||
@ -122,11 +143,11 @@ impl ParserUtils for NyashParser {
|
||||
fn tokens(&self) -> &Vec<Token> {
|
||||
&self.tokens
|
||||
}
|
||||
|
||||
|
||||
fn current(&self) -> usize {
|
||||
self.current
|
||||
}
|
||||
|
||||
|
||||
fn current_mut(&mut self) -> &mut usize {
|
||||
&mut self.current
|
||||
}
|
||||
@ -142,97 +163,99 @@ impl NyashParser {
|
||||
debug_fuel: Some(100_000), // デフォルト値
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// 文字列からパース (トークナイズ + パース)
|
||||
pub fn parse_from_string(input: impl Into<String>) -> Result<ASTNode, ParseError> {
|
||||
Self::parse_from_string_with_fuel(input, Some(100_000))
|
||||
}
|
||||
|
||||
|
||||
/// 文字列からパース (デバッグ燃料指定版)
|
||||
/// fuel: Some(n) = n回まで、None = 無制限
|
||||
pub fn parse_from_string_with_fuel(input: impl Into<String>, fuel: Option<usize>) -> Result<ASTNode, ParseError> {
|
||||
pub fn parse_from_string_with_fuel(
|
||||
input: impl Into<String>,
|
||||
fuel: Option<usize>,
|
||||
) -> Result<ASTNode, ParseError> {
|
||||
let mut tokenizer = crate::tokenizer::NyashTokenizer::new(input);
|
||||
let tokens = tokenizer.tokenize()?;
|
||||
|
||||
|
||||
let mut parser = Self::new(tokens);
|
||||
parser.debug_fuel = fuel;
|
||||
let result = parser.parse();
|
||||
result
|
||||
}
|
||||
|
||||
|
||||
/// パース実行 - Program ASTを返す
|
||||
pub fn parse(&mut self) -> Result<ASTNode, ParseError> {
|
||||
self.parse_program()
|
||||
}
|
||||
|
||||
|
||||
// ===== パース関数群 =====
|
||||
|
||||
|
||||
/// プログラム全体をパース
|
||||
fn parse_program(&mut self) -> Result<ASTNode, ParseError> {
|
||||
let mut statements = Vec::new();
|
||||
let mut _statement_count = 0;
|
||||
|
||||
|
||||
while !self.is_at_end() {
|
||||
|
||||
// EOF tokenはスキップ
|
||||
if matches!(self.current_token().token_type, TokenType::EOF) {
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
// NEWLINE tokenはスキップ(文の区切りとして使用)
|
||||
if matches!(self.current_token().token_type, TokenType::NEWLINE) {
|
||||
self.advance();
|
||||
continue;
|
||||
}
|
||||
|
||||
|
||||
let statement = self.parse_statement()?;
|
||||
statements.push(statement);
|
||||
_statement_count += 1;
|
||||
}
|
||||
|
||||
|
||||
|
||||
// 🔥 すべてのstatic box解析後に循環依存検出
|
||||
self.check_circular_dependencies()?;
|
||||
|
||||
Ok(ASTNode::Program { statements, span: Span::unknown() })
|
||||
|
||||
Ok(ASTNode::Program {
|
||||
statements,
|
||||
span: Span::unknown(),
|
||||
})
|
||||
}
|
||||
// Statement parsing methods are now in statements.rs module
|
||||
|
||||
|
||||
/// 代入文または関数呼び出しをパース
|
||||
fn parse_assignment_or_function_call(&mut self) -> Result<ASTNode, ParseError> {
|
||||
|
||||
// まず左辺を式としてパース
|
||||
let expr = self.parse_expression()?;
|
||||
|
||||
|
||||
// 次のトークンが = または 複合代入演算子 なら代入文
|
||||
if self.match_token(&TokenType::ASSIGN) {
|
||||
self.advance(); // consume '='
|
||||
let value = Box::new(self.parse_expression()?);
|
||||
|
||||
|
||||
// 左辺が代入可能な形式かチェック
|
||||
match &expr {
|
||||
ASTNode::Variable { .. } |
|
||||
ASTNode::FieldAccess { .. } => {
|
||||
Ok(ASTNode::Assignment {
|
||||
target: Box::new(expr),
|
||||
value,
|
||||
span: Span::unknown(),
|
||||
})
|
||||
}
|
||||
ASTNode::Variable { .. } | ASTNode::FieldAccess { .. } => Ok(ASTNode::Assignment {
|
||||
target: Box::new(expr),
|
||||
value,
|
||||
span: Span::unknown(),
|
||||
}),
|
||||
_ => {
|
||||
let line = self.current_token().line;
|
||||
Err(ParseError::InvalidStatement { line })
|
||||
}
|
||||
}
|
||||
} else if self.match_token(&TokenType::PlusAssign) ||
|
||||
self.match_token(&TokenType::MinusAssign) ||
|
||||
self.match_token(&TokenType::MulAssign) ||
|
||||
self.match_token(&TokenType::DivAssign) {
|
||||
} else if self.match_token(&TokenType::PlusAssign)
|
||||
|| self.match_token(&TokenType::MinusAssign)
|
||||
|| self.match_token(&TokenType::MulAssign)
|
||||
|| self.match_token(&TokenType::DivAssign)
|
||||
{
|
||||
if !is_sugar_enabled() {
|
||||
let line = self.current_token().line;
|
||||
return Err(ParseError::UnexpectedToken {
|
||||
found: self.current_token().token_type.clone(),
|
||||
expected: "enable NYASH_SYNTAX_SUGAR_LEVEL=basic|full for '+=' and friends".to_string(),
|
||||
expected: "enable NYASH_SYNTAX_SUGAR_LEVEL=basic|full for '+=' and friends"
|
||||
.to_string(),
|
||||
line,
|
||||
});
|
||||
}
|
||||
@ -250,8 +273,17 @@ impl NyashParser {
|
||||
match &expr {
|
||||
ASTNode::Variable { .. } | ASTNode::FieldAccess { .. } => {
|
||||
let left_clone = expr.clone();
|
||||
let value = ASTNode::BinaryOp { operator: op, left: Box::new(left_clone), right: Box::new(rhs), span: Span::unknown() };
|
||||
Ok(ASTNode::Assignment { target: Box::new(expr), value: Box::new(value), span: Span::unknown() })
|
||||
let value = ASTNode::BinaryOp {
|
||||
operator: op,
|
||||
left: Box::new(left_clone),
|
||||
right: Box::new(rhs),
|
||||
span: Span::unknown(),
|
||||
};
|
||||
Ok(ASTNode::Assignment {
|
||||
target: Box::new(expr),
|
||||
value: Box::new(value),
|
||||
span: Span::unknown(),
|
||||
})
|
||||
}
|
||||
_ => {
|
||||
let line = self.current_token().line;
|
||||
@ -263,10 +295,10 @@ impl NyashParser {
|
||||
Ok(expr)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Expression parsing methods are now in expressions.rs module
|
||||
// Utility methods are now in common.rs module via ParserUtils trait
|
||||
// Item parsing methods are now in items.rs module
|
||||
|
||||
|
||||
// ===== 🔥 Static Box循環依存検出 =====
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user