use anyhow::Context;
use serde::Deserialize;
use std::collections::HashMap;
use std::fs;
use std::path::{Path, PathBuf};
use std::process::Command;
#[cfg(unix)]
use std::os::unix::fs::PermissionsExt;
mod symbols;
mod util;
pub use anyhow::Result;
const BUNDLE_HOME: &str = "target/bundled";
fn build_usage_string(command_name: &str) -> String {
format!(
"Usage:
{command_name} bundle <package> [--release]
{command_name} bundle -p <package1> -p <package2> ... [--release]
{command_name} bundle-universal <package> [--release] (macOS only)
{command_name} bundle-universal -p <package1> -p <package2> ... [--release] (macOS only)
All other 'cargo build' options are supported, including '--target' and '--profile'."
)
}
type BundlerConfig = HashMap<String, PackageConfig>;
#[derive(Debug, Clone, Deserialize)]
struct PackageConfig {
name: Option<String>,
}
#[derive(Debug, Clone, Copy)]
pub enum CompilationTarget {
Linux(Architecture),
MacOS(Architecture),
MacOSUniversal,
Windows(Architecture),
}
#[derive(Debug, Clone, Copy)]
pub enum Architecture {
X86,
X86_64,
AArch64,
}
#[derive(Debug, Clone, Copy)]
pub enum BundleType {
Plugin,
Binary,
}
pub fn main() -> Result<()> {
let args = std::env::args().skip(1);
main_with_args("cargo xtask", args)
}
pub fn main_with_args(command_name: &str, args: impl IntoIterator<Item = String>) -> Result<()> {
chdir_workspace_root()?;
let mut args = args.into_iter();
let usage_string = build_usage_string(command_name);
let command = args
.next()
.with_context(|| format!("Missing command name\n\n{usage_string}",))?;
match command.as_str() {
"bundle" => {
let (packages, other_args) = split_bundle_args(args, &usage_string)?;
build(&packages, &other_args)?;
bundle(&packages[0], &other_args, false)?;
for package in packages.into_iter().skip(1) {
bundle(&package, &other_args, false)?;
}
Ok(())
}
"bundle-universal" => {
let (packages, other_args) = split_bundle_args(args, &usage_string)?;
for arg in &other_args {
if arg == "--target" || arg.starts_with("--target=") {
anyhow::bail!(
"'{command_name} xtask bundle-universal' is incompatible with the '{arg}' \
option."
)
}
}
let mut x86_64_args = other_args.clone();
x86_64_args.push(String::from("--target=x86_64-apple-darwin"));
build(&packages, &x86_64_args)?;
let mut aarch64_args = other_args.clone();
aarch64_args.push(String::from("--target=aarch64-apple-darwin"));
build(&packages, &aarch64_args)?;
bundle(&packages[0], &other_args, true)?;
for package in packages.into_iter().skip(1) {
bundle(&package, &other_args, true)?;
}
Ok(())
}
"known-packages" => list_known_packages(),
_ => anyhow::bail!("Unknown command '{command}'\n\n{usage_string}"),
}
}
pub fn chdir_workspace_root() -> Result<()> {
let project_dir = std::env::var("CARGO_MANIFEST_DIR")
.map(PathBuf::from)
.or_else(|_| std::env::current_dir())
.context(
"'$CARGO_MANIFEST_DIR' was not set and the current working directory could not be \
found",
)?;
let workspace_root = project_dir
.ancestors()
.filter(|dir| dir.join("Cargo.toml").exists())
.last()
.with_context(|| {
format!(
"Could not find a 'Cargo.toml' file in '{}' or any of its parent directories",
project_dir.display()
)
})?;
std::env::set_current_dir(workspace_root)
.context("Could not change to workspace root directory")
}
pub fn build(packages: &[String], args: &[String]) -> Result<()> {
let package_args = packages.iter().flat_map(|package| ["-p", package]);
let status = Command::new("cargo")
.arg("build")
.args(package_args)
.args(args)
.status()
.with_context(|| format!("Could not call cargo to build {}", packages.join(", ")))?;
if !status.success() {
anyhow::bail!("Could not build {}", packages.join(", "));
} else {
Ok(())
}
}
pub fn bundle(package: &str, args: &[String], universal: bool) -> Result<()> {
let mut build_type_dir = "debug";
let mut cross_compile_target: Option<String> = None;
for arg_idx in (0..args.len()).rev() {
let arg = &args[arg_idx];
match arg.as_str() {
"--profile" => {
build_type_dir = args.get(arg_idx + 1).context("Missing profile name")?;
}
"--release" => build_type_dir = "release",
"--target" => {
cross_compile_target = Some(
args.get(arg_idx + 1)
.context("Missing cross-compile target")?
.to_owned(),
);
}
arg if arg.starts_with("--profile=") => {
build_type_dir = arg
.strip_prefix("--profile=")
.context("Missing profile name")?;
}
arg if arg.starts_with("--target=") => {
cross_compile_target = Some(
arg.strip_prefix("--target=")
.context("Missing cross-compile target")?
.to_owned(),
);
}
_ => (),
}
}
if universal {
let x86_64_target_base = target_base(Some("x86_64-apple-darwin"))?.join(build_type_dir);
let x86_64_bin_path = x86_64_target_base.join(binary_basename(
package,
CompilationTarget::MacOS(Architecture::X86_64),
));
let x86_64_lib_path = x86_64_target_base.join(library_basename(
package,
CompilationTarget::MacOS(Architecture::X86_64),
));
let aarch64_target_base = target_base(Some("aarch64-apple-darwin"))?.join(build_type_dir);
let aarch64_bin_path = aarch64_target_base.join(binary_basename(
package,
CompilationTarget::MacOS(Architecture::AArch64),
));
let aarch64_lib_path = aarch64_target_base.join(library_basename(
package,
CompilationTarget::MacOS(Architecture::AArch64),
));
let build_bin = x86_64_bin_path.exists() && aarch64_bin_path.exists();
let build_lib = x86_64_lib_path.exists() && aarch64_lib_path.exists();
if !build_bin && !build_lib {
anyhow::bail!("Could not find built libraries for universal build.");
}
eprintln!();
if build_bin {
bundle_binary(
package,
&[&x86_64_bin_path, &aarch64_bin_path],
CompilationTarget::MacOSUniversal,
)?;
}
if build_lib {
bundle_plugin(
package,
&[&x86_64_lib_path, &aarch64_lib_path],
CompilationTarget::MacOSUniversal,
)?;
}
} else {
let compilation_target = compilation_target(cross_compile_target.as_deref())?;
let target_base = target_base(cross_compile_target.as_deref())?.join(build_type_dir);
let bin_path = target_base.join(binary_basename(package, compilation_target));
let lib_path = target_base.join(library_basename(package, compilation_target));
if !bin_path.exists() && !lib_path.exists() {
anyhow::bail!(
r#"Could not find a built library at '{}'.
Hint: Maybe you forgot to add:
[lib]
crate-type = ["cdylib"]
to your Cargo.toml file?"#,
lib_path.display()
);
}
eprintln!();
if bin_path.exists() {
bundle_binary(package, &[&bin_path], compilation_target)?;
}
if lib_path.exists() {
bundle_plugin(package, &[&lib_path], compilation_target)?;
}
}
Ok(())
}
fn bundle_binary(
package: &str,
bin_paths: &[&Path],
compilation_target: CompilationTarget,
) -> Result<()> {
let bundle_name = match load_bundler_config()?.and_then(|c| c.get(package).cloned()) {
Some(PackageConfig { name: Some(name) }) => name,
_ => package.to_string(),
};
let standalone_bundle_binary_name =
standalone_bundle_binary_name(&bundle_name, compilation_target);
let standalone_binary_path = Path::new(BUNDLE_HOME).join(&standalone_bundle_binary_name);
fs::create_dir_all(standalone_binary_path.parent().unwrap())
.context("Could not create standalone bundle directory")?;
util::reflink_or_combine(bin_paths, &standalone_binary_path, compilation_target)
.context("Could not create standalone bundle")?;
#[cfg(unix)]
if let Ok(metadata) = fs::metadata(&standalone_binary_path) {
let mut permissions = metadata.permissions();
permissions.set_mode(permissions.mode() | 0b0001001001);
fs::set_permissions(&standalone_binary_path, permissions).with_context(|| {
format!(
"Could not make '{}' executable",
standalone_binary_path.display()
)
})?;
}
let standalone_bundle_home = Path::new(BUNDLE_HOME).join(
Path::new(&standalone_bundle_binary_name)
.components()
.next()
.expect("Malformed standalone binary path"),
);
maybe_create_macos_bundle_metadata(
package,
&bundle_name,
&standalone_bundle_home,
compilation_target,
BundleType::Binary,
)?;
maybe_codesign(&standalone_bundle_home, compilation_target);
eprintln!(
"Created a standalone bundle at '{}'",
standalone_bundle_home.display()
);
Ok(())
}
fn bundle_plugin(
package: &str,
lib_paths: &[&Path],
compilation_target: CompilationTarget,
) -> Result<()> {
let bundle_name = match load_bundler_config()?.and_then(|c| c.get(package).cloned()) {
Some(PackageConfig { name: Some(name) }) => name,
_ => package.to_string(),
};
let first_lib_path = lib_paths.first().context("Empty library paths slice")?;
let bundle_clap = symbols::exported(first_lib_path, "clap_entry")
.with_context(|| format!("Could not parse '{}'", first_lib_path.display()))?;
let bundle_vst2 = symbols::exported(first_lib_path, "VSTPluginMain")
.with_context(|| format!("Could not parse '{}'", first_lib_path.display()))?;
let bundle_vst3 = symbols::exported(first_lib_path, "GetPluginFactory")
.with_context(|| format!("Could not parse '{}'", first_lib_path.display()))?;
let bundled_plugin = bundle_clap || bundle_vst2 || bundle_vst3;
if bundle_clap {
let clap_bundle_library_name = clap_bundle_library_name(&bundle_name, compilation_target);
let clap_lib_path = Path::new(BUNDLE_HOME).join(&clap_bundle_library_name);
fs::create_dir_all(clap_lib_path.parent().unwrap())
.context("Could not create CLAP bundle directory")?;
util::reflink_or_combine(lib_paths, &clap_lib_path, compilation_target)
.context("Could not create CLAP bundle")?;
let clap_bundle_home = Path::new(BUNDLE_HOME).join(
Path::new(&clap_bundle_library_name)
.components()
.next()
.expect("Malformed CLAP library path"),
);
maybe_create_macos_bundle_metadata(
package,
&bundle_name,
&clap_bundle_home,
compilation_target,
BundleType::Plugin,
)?;
maybe_codesign(&clap_bundle_home, compilation_target);
eprintln!("Created a CLAP bundle at '{}'", clap_bundle_home.display());
}
if bundle_vst2 {
let vst2_bundle_library_name = vst2_bundle_library_name(&bundle_name, compilation_target);
let vst2_lib_path = Path::new(BUNDLE_HOME).join(&vst2_bundle_library_name);
fs::create_dir_all(vst2_lib_path.parent().unwrap())
.context("Could not create VST2 bundle directory")?;
util::reflink_or_combine(lib_paths, &vst2_lib_path, compilation_target)
.context("Could not create VST2 bundle")?;
let vst2_bundle_home = Path::new(BUNDLE_HOME).join(
Path::new(&vst2_bundle_library_name)
.components()
.next()
.expect("Malformed VST2 library path"),
);
maybe_create_macos_bundle_metadata(
package,
&bundle_name,
&vst2_bundle_home,
compilation_target,
BundleType::Plugin,
)?;
maybe_codesign(&vst2_bundle_home, compilation_target);
eprintln!("Created a VST2 bundle at '{}'", vst2_bundle_home.display());
}
if bundle_vst3 {
let vst3_lib_path =
Path::new(BUNDLE_HOME).join(vst3_bundle_library_name(&bundle_name, compilation_target));
fs::create_dir_all(vst3_lib_path.parent().unwrap())
.context("Could not create VST3 bundle directory")?;
util::reflink_or_combine(lib_paths, &vst3_lib_path, compilation_target)
.context("Could not create VST3 bundle")?;
let vst3_bundle_home = vst3_lib_path
.parent()
.unwrap()
.parent()
.unwrap()
.parent()
.unwrap();
maybe_create_macos_bundle_metadata(
package,
&bundle_name,
vst3_bundle_home,
compilation_target,
BundleType::Plugin,
)?;
maybe_codesign(vst3_bundle_home, compilation_target);
eprintln!("Created a VST3 bundle at '{}'", vst3_bundle_home.display());
}
if !bundled_plugin {
eprintln!("Not creating any plugin bundles because the package does not export any plugins")
}
Ok(())
}
pub fn list_known_packages() -> Result<()> {
if let Some(config) = load_bundler_config()? {
for package in config.keys() {
println!("{package}");
}
}
Ok(())
}
fn load_bundler_config() -> Result<Option<BundlerConfig>> {
let bundler_config_path = Path::new("bundler.toml");
if !bundler_config_path.exists() {
return Ok(None);
}
let result = toml::from_str(
&fs::read_to_string(bundler_config_path)
.with_context(|| format!("Could not read '{}'", bundler_config_path.display()))?,
)
.with_context(|| format!("Could not parse '{}'", bundler_config_path.display()))?;
Ok(Some(result))
}
fn split_bundle_args(
args: impl Iterator<Item = String>,
usage_string: &str,
) -> Result<(Vec<String>, Vec<String>)> {
let mut args = args.peekable();
let mut packages = Vec::new();
if args.peek().map(|s| s.as_str()) == Some("-p") {
while args.peek().map(|s| s.as_str()) == Some("-p") {
packages.push(
args.nth(1)
.with_context(|| format!("Missing package name after -p\n\n{usage_string}"))?,
);
}
} else {
packages.push(
args.next()
.with_context(|| format!("Missing package name\n\n{usage_string}"))?,
);
};
let other_args: Vec<_> = args.collect();
Ok((packages, other_args))
}
fn compilation_target(cross_compile_target: Option<&str>) -> Result<CompilationTarget> {
match cross_compile_target {
Some("i686-unknown-linux-gnu") => Ok(CompilationTarget::Linux(Architecture::X86)),
Some("i686-apple-darwin") => Ok(CompilationTarget::MacOS(Architecture::X86)),
Some("i686-pc-windows-gnu") | Some("i686-pc-windows-msvc") => {
Ok(CompilationTarget::Windows(Architecture::X86))
}
Some("x86_64-unknown-linux-gnu") => Ok(CompilationTarget::Linux(Architecture::X86_64)),
Some("x86_64-apple-darwin") => Ok(CompilationTarget::MacOS(Architecture::X86_64)),
Some("x86_64-pc-windows-gnu") | Some("x86_64-pc-windows-msvc") => {
Ok(CompilationTarget::Windows(Architecture::X86_64))
}
Some("aarch64-unknown-linux-gnu") => Ok(CompilationTarget::Linux(Architecture::AArch64)),
Some("aarch64-apple-darwin") => Ok(CompilationTarget::MacOS(Architecture::AArch64)),
Some("aarch64-pc-windows-gnu") | Some("aarch64-pc-windows-msvc") => {
Ok(CompilationTarget::Windows(Architecture::AArch64))
}
Some(target) => anyhow::bail!("Unhandled cross-compilation target: {}", target),
None => {
#[cfg(target_arch = "x86")]
let architecture = Architecture::X86;
#[cfg(target_arch = "x86_64")]
let architecture = Architecture::X86_64;
#[cfg(target_arch = "aarch64")]
let architecture = Architecture::AArch64;
#[cfg(target_os = "linux")]
return Ok(CompilationTarget::Linux(architecture));
#[cfg(target_os = "macos")]
return Ok(CompilationTarget::MacOS(architecture));
#[cfg(target_os = "windows")]
return Ok(CompilationTarget::Windows(architecture));
}
}
}
fn target_base(cross_compile_target: Option<&str>) -> Result<PathBuf> {
match cross_compile_target {
Some(target) => Ok(Path::new("target").join(target)),
None => Ok(PathBuf::from("target")),
}
}
fn binary_basename(package: &str, target: CompilationTarget) -> String {
let bin_name = package.replace('-', "_");
match target {
CompilationTarget::Linux(_)
| CompilationTarget::MacOS(_)
| CompilationTarget::MacOSUniversal => bin_name,
CompilationTarget::Windows(_) => format!("{bin_name}.exe"),
}
}
fn library_basename(package: &str, target: CompilationTarget) -> String {
let lib_name = package.replace('-', "_");
match target {
CompilationTarget::Linux(_) => format!("lib{lib_name}.so"),
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal => {
format!("lib{lib_name}.dylib")
}
CompilationTarget::Windows(_) => format!("{lib_name}.dll"),
}
}
fn standalone_bundle_binary_name(package: &str, target: CompilationTarget) -> String {
match target {
CompilationTarget::Linux(_) => package.to_owned(),
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal => {
format!("{package}.app/Contents/MacOS/{package}")
}
CompilationTarget::Windows(_) => format!("{package}.exe"),
}
}
fn clap_bundle_library_name(package: &str, target: CompilationTarget) -> String {
match target {
CompilationTarget::Linux(_) | CompilationTarget::Windows(_) => format!("{package}.clap"),
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal => {
format!("{package}.clap/Contents/MacOS/{package}")
}
}
}
fn vst2_bundle_library_name(package: &str, target: CompilationTarget) -> String {
match target {
CompilationTarget::Linux(_) => format!("{package}.so"),
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal => {
format!("{package}.vst/Contents/MacOS/{package}")
}
CompilationTarget::Windows(_) => format!("{package}.dll"),
}
}
fn vst3_bundle_library_name(package: &str, target: CompilationTarget) -> String {
match target {
CompilationTarget::Linux(Architecture::X86) => {
format!("{package}.vst3/Contents/i386-linux/{package}.so")
}
CompilationTarget::Linux(Architecture::X86_64) => {
format!("{package}.vst3/Contents/x86_64-linux/{package}.so")
}
CompilationTarget::Linux(Architecture::AArch64) => {
format!("{package}.vst3/Contents/aarch64-linux/{package}.so")
}
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal => {
format!("{package}.vst3/Contents/MacOS/{package}")
}
CompilationTarget::Windows(Architecture::X86) => {
format!("{package}.vst3/Contents/x86-win/{package}.vst3")
}
CompilationTarget::Windows(Architecture::X86_64) => {
format!("{package}.vst3/Contents/x86_64-win/{package}.vst3")
}
CompilationTarget::Windows(Architecture::AArch64) => {
format!("{package}.vst3/Contents/arm_64-win/{package}.vst3")
}
}
}
pub fn maybe_create_macos_bundle_metadata(
package: &str,
display_name: &str,
bundle_home: &Path,
target: CompilationTarget,
bundle_type: BundleType,
) -> Result<()> {
if !matches!(
target,
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal
) {
return Ok(());
}
let package_type = match bundle_type {
BundleType::Plugin => "BNDL",
BundleType::Binary => "APPL",
};
fs::write(
bundle_home.join("Contents").join("PkgInfo"),
format!("{package_type}????"),
)
.context("Could not create PkgInfo file")?;
fs::write(
bundle_home.join("Contents").join("Info.plist"),
format!(r#"<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist>
<dict>
<key>CFBundleExecutable</key>
<string>{display_name}</string>
<key>CFBundleIconFile</key>
<string></string>
<key>CFBundleIdentifier</key>
<string>com.nih-plug.{package}</string>
<key>CFBundleName</key>
<string>{display_name}</string>
<key>CFBundleDisplayName</key>
<string>{display_name}</string>
<key>CFBundlePackageType</key>
<string>{package_type}</string>
<key>CFBundleSignature</key>
<string>????</string>
<key>CFBundleShortVersionString</key>
<string>1.0.0</string>
<key>CFBundleVersion</key>
<string>1.0.0</string>
<key>NSHumanReadableCopyright</key>
<string></string>
<key>NSHighResolutionCapable</key>
<true/>
</dict>
</plist>
"#),
)
.context("Could not create Info.plist file")?;
Ok(())
}
pub fn maybe_codesign(bundle_home: &Path, target: CompilationTarget) {
if !matches!(
target,
CompilationTarget::MacOS(_) | CompilationTarget::MacOSUniversal
) {
return;
}
let success = Command::new("codesign")
.arg("-f")
.arg("-s")
.arg("-")
.arg(bundle_home)
.status()
.is_ok();
if !success {
eprintln!(
"WARNING: Could not self-sign '{}', it may fail to run depending on the environment",
bundle_home.display()
)
}
}