dioxus-use-js 0.4.7

A macro that generates Rust bindings to JavaScript or TypeScript functions, with compile time checks. For use with Dioxus.
Documentation
use std::env::current_dir;
use std::path::PathBuf;
use std::process::Command;

use bon::Builder;

#[cfg_attr(docsrs, doc(cfg(feature = "build")))]
#[derive(Builder)]
pub struct BunBuild {
    /// Files to build from
    src_files: Vec<PathBuf>,
    /// The output directory
    output_dir: PathBuf,
    /// If set the output will be a single file
    output_file: Option<String>,
    #[builder(default = true)]
    minify: bool,
    /// Extra bun build flags
    extra_flags: Option<Vec<String>>,
    /// If true, the normal `bun build ...` command will not execute if bun is not installed, but will instead print a warning.
    /// This may be useful in CI environment where bun is not installed (and is not needed), since the output was created during
    /// development and is already in source.
    #[builder(default = false)]
    skip_if_no_bun: bool,
}

impl BunBuild {
    pub fn run(&self) {
        fn bun_exists() -> bool {
            Command::new("bun")
                .arg("--version")
                .output()
                .map(|o| o.status.success())
                .unwrap_or(false)
        }
        if self.skip_if_no_bun && !bun_exists() {
            println!("cargo:warning=Bun does not exist and configuration allows this, skipping bun build.");
            return;
        }

        let src_files = self
            .src_files
            .iter()
            .map(|e| {
                e.to_str()
                    .expect("a src file is not a valid utf-8 string")
                    .to_owned()
            })
            .collect::<Vec<_>>();
        let output_dir = self
            .output_dir
            .to_str()
            .expect("The output file is not a valid utf-8 string")
            .to_owned();

        let mut args = Vec::new();
        args.push("build".to_owned());
        args.extend(src_files);

        if let Some(output_file) = &self.output_file {
            args.extend([
                "--outfile".to_owned(),
                self.output_dir
                    .join(output_file)
                    .to_str()
                    .unwrap()
                    .to_owned(),
            ])
        } else {
            args.extend(["--outdir".to_owned(), output_dir]);
        }
        if self.minify {
            args.push("--minify".to_owned());
        }
        args.push("--target=browser".to_owned());
        args.extend(self.extra_flags.iter().flatten().cloned());

        // Run bun build
        let output = Command::new("bun")
            .args(&args)
            .output()
            .expect("Failed to execute bun build");
        if !output.status.success() {
            panic!(
                "bun build failed.\nArgs: `{}`\nCurrent Directory: `{}`\nStderr:\n{}",
                args.join(" "),
                current_dir().unwrap().display(),
                String::from_utf8_lossy(&output.stderr)
            );
        }
    }
}