lune-packaging/src/cli/cli.rs

180 lines
5.9 KiB
Rust
Raw Normal View History

use std::{fs::read_to_string, process::ExitCode};
2023-01-19 01:47:14 +00:00
use anyhow::Result;
use clap::{CommandFactory, Parser};
2023-01-19 01:47:14 +00:00
use lune::Lune;
use crate::utils::{files::find_parse_file_path, github::Client as GithubClient};
2023-01-19 01:47:14 +00:00
const LUNE_SELENE_FILE_NAME: &str = "lune.yml";
const LUNE_LUAU_FILE_NAME: &str = "luneTypes.d.luau";
2023-01-19 01:47:14 +00:00
/// Lune CLI
#[derive(Parser, Debug, Default)]
2023-01-19 01:47:14 +00:00
#[command(author, version, about, long_about = None)]
pub struct Cli {
/// Path to the file to run, or the name
/// of a luau file in a lune directory
///
/// Can be omitted when downloading type definitions
script_path: Option<String>,
/// Arguments to pass to the file as vararg (...)
script_args: Vec<String>,
/// Pass this flag to download the Selene type
/// definitions file to the current directory
#[clap(long)]
download_selene_types: bool,
/// Pass this flag to download the Luau type
/// definitions file to the current directory
#[clap(long)]
download_luau_types: bool,
}
#[allow(dead_code)]
2023-01-19 01:47:14 +00:00
impl Cli {
2023-01-19 05:23:06 +00:00
pub fn from_path<S>(path: S) -> Self
where
S: Into<String>,
{
2023-01-19 01:47:14 +00:00
Self {
script_path: Some(path.into()),
..Default::default()
2023-01-19 05:23:06 +00:00
}
}
pub fn from_path_with_args<S, A>(path: S, args: A) -> Self
where
S: Into<String>,
A: Into<Vec<String>>,
{
Self {
script_path: Some(path.into()),
script_args: args.into(),
..Default::default()
2023-01-19 01:47:14 +00:00
}
}
pub fn download_selene_types() -> Self {
Self {
download_selene_types: true,
..Default::default()
}
}
pub fn download_luau_types() -> Self {
Self {
download_luau_types: true,
..Default::default()
}
}
pub async fn run(self) -> Result<ExitCode> {
// Download definition files, if wanted
let download_types_requested = self.download_selene_types || self.download_luau_types;
if download_types_requested {
let client = GithubClient::new();
2023-01-24 18:48:37 +00:00
let release = client.fetch_release_for_this_version().await?;
if self.download_selene_types {
println!("Downloading Selene type definitions...");
client
.fetch_release_asset(&release, LUNE_SELENE_FILE_NAME)
2023-01-24 18:48:37 +00:00
.await?;
}
if self.download_luau_types {
println!("Downloading Luau type definitions...");
client
.fetch_release_asset(&release, LUNE_LUAU_FILE_NAME)
2023-01-24 18:48:37 +00:00
.await?;
}
}
if self.script_path.is_none() {
2023-01-21 02:05:51 +00:00
// Only downloading types without running a script is completely
// fine, and we should just exit the program normally afterwards
if download_types_requested {
return Ok(ExitCode::SUCCESS);
}
2023-01-21 02:05:51 +00:00
// HACK: We know that we didn't get any arguments here but since
// script_path is optional clap will not error on its own, to fix
// we will duplicate the cli command and make arguments required,
// which will then fail and print out the normal help message
let cmd = Cli::command();
cmd.arg_required_else_help(true).get_matches();
}
2023-01-19 01:47:14 +00:00
// Parse and read the wanted file
let file_path = find_parse_file_path(&self.script_path.unwrap())?;
let file_contents = read_to_string(&file_path)?;
// Display the file path relative to cwd with no extensions in stack traces
let file_display_name = file_path.with_extension("").display().to_string();
// Create a new lune object with all globals & run the script
let lune = Lune::new().with_args(self.script_args).with_all_globals();
let result = lune.run(&file_display_name, &file_contents).await;
Ok(match result {
Err(e) => {
eprintln!("{e}");
ExitCode::from(1)
}
Ok(code) => code,
})
2023-01-19 01:47:14 +00:00
}
}
#[cfg(test)]
mod tests {
use std::env::{current_dir, set_current_dir};
2023-01-24 18:48:37 +00:00
use anyhow::{bail, Context, Result};
use serde_json::Value;
use smol::fs::{create_dir_all, read_to_string, remove_file};
2023-01-24 18:48:37 +00:00
use super::{Cli, LUNE_LUAU_FILE_NAME, LUNE_SELENE_FILE_NAME};
async fn run_cli(cli: Cli) -> Result<()> {
let path = current_dir()
.context("Failed to get current dir")?
.join("bin");
create_dir_all(&path)
.await
.context("Failed to create bin dir")?;
set_current_dir(&path).context("Failed to set current dir")?;
cli.run().await?;
Ok(())
}
async fn ensure_file_exists_and_is_not_json(file_name: &str) -> Result<()> {
match read_to_string(file_name)
.await
.context("Failed to read definitions file")
{
Ok(file_contents) => match serde_json::from_str::<Value>(&file_contents) {
Err(_) => {
remove_file(file_name)
.await
.context("Failed to remove definitions file")?;
Ok(())
}
Ok(_) => bail!("Downloading selene definitions returned json, expected luau"),
},
Err(e) => bail!("Failed to download selene definitions!\n{e}"),
}
}
#[test]
fn download_selene_types() -> Result<()> {
smol::block_on(async {
run_cli(Cli::download_selene_types()).await?;
ensure_file_exists_and_is_not_json(LUNE_SELENE_FILE_NAME).await?;
Ok(())
})
}
#[test]
fn download_luau_types() -> Result<()> {
smol::block_on(async {
run_cli(Cli::download_luau_types()).await?;
ensure_file_exists_and_is_not_json(LUNE_LUAU_FILE_NAME).await?;
Ok(())
})
}
}