phprs 0.1.9

A PHP interpreter with build/package manager written in Rust
Documentation
//! Compiler
//!
//! Compilation
//!
//! This module handles compilation of PHP code to opcodes

#[cfg(test)]
mod tests;

pub mod context;
pub mod control_flow;
pub mod expression;
pub mod function;
pub mod function_table;
pub mod statement;

use crate::engine::lexer::{Lexer, TokenType};
use crate::engine::vm::OpArray;
pub use context::CompileContext;
use statement::parse_statement;

/// Compile PHP code string to opcodes
pub fn compile_string(code: &str, filename: &str) -> Result<OpArray, String> {
    let (op_array, _) = compile_string_with_functions(code, filename)?;
    Ok(op_array)
}

/// Compile PHP code string to opcodes and return the function table
pub fn compile_string_with_functions(code: &str, filename: &str) -> Result<(OpArray, function_table::FunctionTable), String> {
    let mut context = CompileContext::new();
    context.set_filename(filename);

    let mut lexer = Lexer::new(code);

    // Skip opening tag if present
    let mut token = lexer.next_token()?;
    if token.token_type == TokenType::T_OPEN_TAG {
        token = lexer.next_token()?;
    }

    while token.token_type != TokenType::T_EOF {
        token = parse_statement(&mut lexer, &mut context, token)?;
    }

    let ft = std::mem::replace(&mut context.function_table, function_table::FunctionTable::new());
    Ok((context.finalize(), ft))
}

/// Compile file to opcodes
pub fn compile_file(filename: &str) -> Result<OpArray, String> {
    use std::fs;
    let code = fs::read_to_string(filename).map_err(|e| format!("Failed to read file: {e}"))?;
    compile_string(&code, filename)
}