roto 0.10.0

a statically-typed, compiled, embedded scripting language
Documentation
use std::path::PathBuf;

use clap::{Parser, Subcommand};

use crate::{
    FileTree, RotoError, RotoReport, Runtime, runtime::OptCtx,
    tools::print::print_highlighted,
};

#[derive(Parser)]
#[command(version, about, long_about = None)]
#[command(propagate_version = true)]
struct Cli {
    #[command(subcommand)]
    command: Command,
}

#[derive(Subcommand)]
enum Command {
    /// Generate documentation for the runtime
    Doc {
        #[arg()]
        path: PathBuf,
    },
    /// Type check a script
    Check {
        #[arg()]
        file: PathBuf,
    },
    /// Test a script
    Test {
        #[arg()]
        file: PathBuf,
    },
    /// Run a script's function
    Run {
        #[arg()]
        file: PathBuf,
        #[arg(default_value = "main")]
        function: String,
    },
    /// Print a Roto file with syntax highlighting
    Print {
        #[arg()]
        file: PathBuf,
    },
}

/// Run a basic CLI for a given runtime
///
/// This is useful for providing to users to check their scripts or run their tests
/// with the runtime that the host application provides.
///
/// This CLI provides the following subcommands:
///
///  - `doc`: generate documentation
///  - `check`: type check a script
///  - `test`: run tests for a script
///  - `run`: run a function of a script
///
/// This function exits the program and does not return.
pub fn cli(rt: &Runtime<impl OptCtx>) -> ! {
    match cli_inner(rt) {
        Ok(()) => std::process::exit(0),
        Err(err) => {
            eprintln!("{err}");
            std::process::exit(1);
        }
    }
}

fn cli_inner(rt: &Runtime<impl OptCtx>) -> Result<(), RotoReport> {
    let cli = Cli::parse();

    match &cli.command {
        Command::Doc { path } => {
            rt.rt.print_documentation(path).unwrap();
        }
        Command::Check { file } => {
            FileTree::read(file)?.parse()?.typecheck(rt)?;
            println!("All ok!")
        }
        Command::Test { file } => {
            let Some(rt) = rt.clone().try_without_ctx() else {
                eprintln!("Can only run tests on a Runtime without Context");
                return Err(RotoReport {
                    errors: vec![RotoError::TestsFailed()],
                    ..Default::default()
                });
            };

            let mut p = FileTree::read(file)?
                .parse()?
                .typecheck(&rt)?
                .lower_to_mir()
                .lower_to_lir()
                .codegen();

            if let Err(()) = p.run_tests() {
                return Err(RotoReport {
                    errors: vec![RotoError::TestsFailed()],
                    ..Default::default()
                });
            }
        }
        Command::Run { file, function } => {
            let Some(rt) = rt.clone().try_without_ctx() else {
                return Err(RotoReport {
                    errors: vec![RotoError::Custom("Can only run a script with a Runtime without Context".into())],
                    ..Default::default()
                });
            };

            let mut p = FileTree::read(file)?
                .parse()?
                .typecheck(&rt)?
                .lower_to_mir()
                .lower_to_lir()
                .codegen();

            let f =
                p.get_function::<fn()>(function).map_err(|e| RotoReport {
                    errors: vec![RotoError::CouldNotRetrieveFunction(e)],
                    ..Default::default()
                })?;

            f.call()
        }
        Command::Print { file } => {
            let s = std::fs::read_to_string(file).unwrap();
            print_highlighted(&s);
        }
    }
    Ok(())
}