Compare commits

...

2 Commits

Author SHA1 Message Date
73d31ea046 feat: simple connection to a php repl 2024-11-05 21:18:58 -05:00
f3f6e9fd44 feat: use a single symbol_table in a repl session 2024-11-05 19:38:22 -05:00
7 changed files with 163 additions and 20 deletions

View File

@ -21,23 +21,25 @@
## v0.1.4
- [ ] Synchronize the THP & PHP repls outputs
- [ ] Test semantic analysis
- [ ] Generate php code from current AST
- [ ] Parse obj/map/dict syntax
- [ ] Parse tuple syntax
- [ ] Parse class instantiation syntax
- [ ] Parse logic operators `&& ||`
- [ ] Parse namespace operator `::`
- [ ] Implement subtyping for numbers
## v0.1.3
- [ ] Test semantic analysis
- [ ] Generate php code from current AST
- [x] Typecheck and semantic check simple assignment
- [x] Test correct operator precedence
- [x] Parse assignments
- [x] Parse dot `.` operator
- [ ] Parse logic operators `&& ||`
- [x] Parse Array access `arr[pos]`
- [ ] Parse namespace operator `::`
- [ ] Implement subtyping for numbers
- [x] Spawn a PHP repl, and connect the THP repl to it
## v0.1.2

2
Cargo.lock generated
View File

@ -152,7 +152,7 @@ dependencies = [
[[package]]
name = "thp"
version = "0.1.2"
version = "0.1.3"
dependencies = [
"ariadne",
"colored",

View File

@ -1,6 +1,6 @@
[package]
name = "thp"
version = "0.1.2"
version = "0.1.3"
edition = "2021"

View File

@ -12,6 +12,16 @@ impl Transpilable for PFile<'_> {
}
}
impl PFile<'_> {
pub fn transpile_without_header(&self) -> String {
let mut fragments = vec![];
for statement in self.statements.iter() {
fragments.push(statement.transpile());
}
fragments.join("\n")
}
}
#[cfg(test)]
mod tests {
use crate::{

View File

@ -1,7 +1,12 @@
use std::io::{self, Write};
use ::std::io::{self, BufRead, BufReader, Error, Read, Write};
use ::std::process::{Command, Stdio};
use ::std::thread;
use ::std::time::Duration;
use crate::codegen::Transpilable;
use crate::error_handling::PrintableError;
use crate::semantic::std;
use crate::semantic::symbol_table::SymbolTable;
use super::lexic;
use super::syntax;
@ -10,11 +15,95 @@ use crate::php_ast::transformers::PHPTransformable;
/// Executes the REPL, reading from stdin, compiling and emitting PHP to stdout
pub fn run() -> io::Result<()> {
// attempt to spawn a php repl
let php_repl = Command::new("php")
.arg("-a")
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.spawn();
let mut php_repl = match php_repl {
Ok(c) => c,
Err(error) => {
eprintln!("Couldn't open a PHP REPL session: {:?}", error);
return Err(error);
}
};
let mut php_stdin = match php_repl.stdin.take() {
Some(handle) => handle,
None => {
eprintln!("Error: couldn't get stdin handle from PHP REPL");
return Err(Error::new(
io::ErrorKind::Other,
"Can't get PHP REPL stdin handle",
));
}
};
let mut php_stdout = match php_repl.stdout.take() {
Some(h) => h,
None => {
eprintln!("Error: couldn't get stdout handle from PHP REPL");
return Err(Error::new(
io::ErrorKind::Other,
"Can't get PHP REPL stdout handle",
));
}
};
let stdin = io::stdin();
let mut buffer = String::new();
let mut repl_symbol_table = SymbolTable::new();
std::populate(&mut repl_symbol_table);
// start a thread that prints whatever php sends back
let php_stdout_handle = thread::spawn(move || {
let mut reader = BufReader::new(php_stdout);
loop {
// sleep for 50ms
thread::sleep(Duration::from_millis(50));
// read a line from php
let mut line = String::new();
match reader.read_line(&mut line) {
Ok(n) => {
if n == 0 {
// EOF
break;
}
if n == 1 {
// just a newline
continue;
}
// Suppress some php outputs
if line == "Interactive shell\n" {
continue;
}
// Ignore anything that starts with `php > `
if line.starts_with("php > ") {
continue;
}
print!("php output: `{line}`")
}
Err(error) => {
// log error and exit
eprint!("Error while reading from PHP STDOUT: {:?}", error);
break;
}
};
}
println!("php stdout thread finished");
});
println!("REPL: Enter expressions to evaluate. Type Ctrl-D to exit.");
loop {
let result = loop {
// TODO: syncronize the writes to thp stdout
// such that the php output doesnt overlap with this
print!("> ");
io::stdout().flush()?;
buffer.clear();
@ -26,18 +115,46 @@ pub fn run() -> io::Result<()> {
break Ok(());
}
Ok(_) => {
compile(&buffer);
match compile(&buffer, &mut repl_symbol_table) {
Some(php_code) => {
// TODO: this cant be efficient, fix
let php_code = format!("{php_code}\n");
// send php code
//println!("{php_code}");
match php_stdin.write_all(php_code.as_bytes()) {
Ok(_) => {}
Err(error) => {
eprintln!("Error writing the generated code to the PHP process.");
break Err(error);
}
};
// the php repl should respond with its output, and that
// will be printed by another thread
}
None => {}
}
}
Err(error) => {
eprintln!("Error reading stdin.");
break Err(error);
}
};
}
};
// kill the php process
php_repl.kill().expect("Couldnt KILL child php process...");
php_stdout_handle
.join()
.expect("STDOUT thread failed to join...");
result
}
/// Full pipeline from THP source code to PHP output
fn compile(input: &String) {
/// Compiles THP code and returns the generated PHP code as a String
fn compile(input: &String, symbol_table: &mut SymbolTable) -> Option<String> {
//
// Lexical analysis
//
@ -45,7 +162,7 @@ fn compile(input: &String) {
Ok(t) => t,
Err(error) => {
error.print_ariadne(input);
return;
return None;
}
};
@ -56,19 +173,19 @@ fn compile(input: &String) {
Ok(ast) => ast,
Err(error) => {
error.print_ariadne(input);
return;
return None;
}
};
//
// Semantic analysis
//
let res1 = crate::semantic::check_semantics(&ast);
let res1 = crate::semantic::check_semantics_with(&ast, symbol_table);
match res1 {
Ok(_) => {}
Err(error) => {
error.print_ariadne(input);
return;
return None;
}
}
@ -80,5 +197,5 @@ fn compile(input: &String) {
//
// Codegen
//
println!("{}", php_ast.transpile());
Some(php_ast.transpile_without_header())
}

View File

@ -2,11 +2,12 @@ use crate::{error_handling::MistiError, syntax::ast::ModuleAST};
mod checks;
mod impls;
mod std;
mod symbol_table;
pub mod std;
pub mod symbol_table;
mod types;
use impls::SemanticCheck;
use symbol_table::SymbolTable;
// What to do?
// 1. Create a mutable symbol table
@ -25,6 +26,15 @@ pub fn check_semantics(ast: &ModuleAST) -> Result<(), MistiError> {
ast.check_semantics(&global_scope)
}
/// Checks that the AST is semantically correct.
/// Accepts a handle to a symbol table to operate with.
pub fn check_semantics_with(
ast: &ModuleAST,
symbol_table: &mut SymbolTable,
) -> Result<(), MistiError> {
ast.check_semantics(&symbol_table)
}
#[cfg(test)]
mod tests {
use crate::semantic::types::Type;

View File

@ -19,4 +19,8 @@ pub fn populate(table: &mut SymbolTable) {
// + operator (Int, Int) -> Int
let plus_op = Type::Function(vec![INT.into(), INT.into()], INT.into());
table.insert("+".into(), plus_op);
// - operator (Int, Int) -> Int
let plus_op = Type::Function(vec![INT.into(), INT.into()], INT.into());
table.insert("-".into(), plus_op);
}