//! The `ShulkerScript` language. //! //! `ShulkerScript` is a simple, imperative scripting language for creating Minecraft data packs. #![deny( missing_debug_implementations, missing_copy_implementations, clippy::nursery, rustdoc::broken_intra_doc_links, clippy::missing_errors_doc )] #![warn(missing_docs, clippy::all, clippy::pedantic)] #![allow(clippy::missing_panics_doc, clippy::missing_const_for_fn)] pub use shulkerbox; pub mod base; pub mod lexical; pub mod syntax; pub mod transpile; use std::path::Path; use base::{source_file::SourceFile, Error, FileProvider, Handler, Result}; use syntax::{parser::Parser, syntax_tree::program::ProgramFile}; #[cfg(feature = "shulkerbox")] use shulkerbox::{datapack::Datapack, virtual_fs::VFolder}; use crate::lexical::token_stream::TokenStream; /// Converts the given source code to tokens and returns a token stream. /// /// # Errors /// - If an error occurs while loading the [`SourceFile`]. /// /// # Examples /// ```no_run /// use std::path::Path; /// use shulkerscript::{tokenize, base::{FsProvider, PrintHandler}}; /// /// let token_stream = tokenize(&PrintHandler::new(), &FsProvider::default(), Path::new("path/to/file.shu"))?; /// # Ok::<(), shulkerscript::base::Error>(()) /// ``` pub fn tokenize( handler: &impl Handler, file_provider: &impl FileProvider, path: &Path, ) -> Result { tracing::info!("Tokenizing the source code at path: {}", path.display()); let source_file = SourceFile::load(path, file_provider)?; Ok(TokenStream::tokenize(&source_file, handler)) } /// Parses the given source code and returns the AST of the program. /// /// # Errors /// - If an error occurs during [`tokenize()`]. /// - If an error occurs while parsing the source code. /// /// # Examples /// ```no_run /// use std::path::Path; /// use shulkerscript::{parse, base::{FsProvider, PrintHandler}}; /// /// let program_file = parse(&PrintHandler::new(), &FsProvider::default(), Path::new("path/to/file.shu"))?; /// # Ok::<(), shulkerscript::base::Error>(()) /// ``` pub fn parse( handler: &impl Handler, file_provider: &impl FileProvider, path: &Path, ) -> Result { let tokens = tokenize(handler, file_provider, path)?; if handler.has_received() { return Err(Error::Other( "An error occurred while tokenizing the source code.", )); } tracing::info!("Parsing the source code at path: {}", path.display()); let mut parser = Parser::new(&tokens); let program = parser.parse_program(handler).ok_or(Error::Other( "An error occurred while parsing the source code.", ))?; if handler.has_received() { return Err(Error::Other( "An error occurred while parsing the source code.", )); } Ok(program) } /// Transpiles the given source code into a shulkerbox [`Datapack`]. /// /// # Parameters: /// - `script_paths`: A list of tuples containing the identifier and the path of each script file. /// /// # Errors /// - If an error occurs during [`parse()`] /// - If an error occurs while transpiling the source code. /// /// # Examples /// ```no_run /// use std::path::Path; /// use shulkerscript::{transpile, base::{FsProvider, PrintHandler}}; /// /// let datapack = transpile( /// &PrintHandler::new(), /// &FsProvider::default(), /// 48, /// &[ /// (String::from("fileA"), Path::new("path/to/fileA.shu")), /// (String::from("fileB"), Path::new("path/to/fileB.shu")) /// ])?; /// # Ok::<(), shulkerscript::base::Error>(()) /// ``` #[cfg(feature = "shulkerbox")] pub fn transpile( handler: &impl Handler, file_provider: &F, pack_format: u8, script_paths: &[(String, P)], ) -> Result where F: FileProvider, P: AsRef, { use transpile::Transpiler; let programs = script_paths .iter() .map(|(program_identifier, path)| { let program = parse(handler, file_provider, path.as_ref())?; Ok((program_identifier, program)) }) .collect::>(); if programs.iter().any(Result::is_err) { return Err(programs.into_iter().find_map(Result::err).unwrap()); } let programs = programs .into_iter() .filter_map(Result::ok) .collect::>(); tracing::info!("Transpiling the source code."); let mut transpiler = Transpiler::new(pack_format); transpiler.transpile(&programs, handler)?; let datapack = transpiler.into_datapack(); if handler.has_received() { return Err(Error::Other( "An error occurred while transpiling the source code.", )); } Ok(datapack) } /// Compiles the given source code. /// /// # Parameters: /// - `script_paths`: A list of tuples containing the identifier and the path of each script file. /// /// # Errors /// - If an error occurs during [`transpile()`] /// /// # Examples /// ```no_run /// use std::path::Path; /// use shulkerscript::{compile, base::{FsProvider, PrintHandler}}; /// /// let vfolder = compile( /// &PrintHandler::new(), /// &FsProvider::default(), /// 48, /// &[ /// (String::from("fileA"), Path::new("path/to/fileA.shu")), /// (String::from("fileB"), Path::new("path/to/fileB.shu")) /// ])?; /// # Ok::<(), shulkerscript::base::Error>(()) /// ``` #[cfg(feature = "shulkerbox")] pub fn compile( handler: &impl Handler, file_provider: &F, pack_format: u8, script_paths: &[(String, P)], ) -> Result where F: FileProvider, P: AsRef, { use shulkerbox::prelude::CompileOptions; let datapack = transpile(handler, file_provider, pack_format, script_paths)?; tracing::info!("Compiling the source code."); Ok(datapack.compile(&CompileOptions::default())) }