use super::PackageArtifact;
use crate::core::config::ResolvedCrateConfig;
use crate::publish::platform::RustTarget;
use anyhow::{Context, Result};
use std::fs;
use std::path::{Path, PathBuf};
pub fn package_ruby(
config: &ResolvedCrateConfig,
target: &RustTarget,
workspace_root: &Path,
output_dir: &Path,
version: &str,
) -> Result<PackageArtifact> {
let gem_name = config.ruby_gem_name();
let platform = target.platform_for(crate::core::config::extras::Language::Ruby);
let pkg_dir_str = config.package_dir(crate::core::config::extras::Language::Ruby);
let pkg_dir = workspace_root.join(&pkg_dir_str);
if !pkg_dir.exists() {
anyhow::bail!("Ruby package directory does not exist: {}", pkg_dir.display());
}
let rb_crate = crate::publish::crate_name_from_output(config, crate::core::config::extras::Language::Ruby)
.unwrap_or_else(|| format!("{}-rb", config.name));
let lib_filename = target.shared_lib_name(&rb_crate.replace('-', "_"));
let native_lib = find_ruby_native_lib(workspace_root, target, &rb_crate, &lib_filename)?;
let lib_dest_dir = pkg_dir.join("lib").join(&gem_name);
fs::create_dir_all(&lib_dest_dir).with_context(|| format!("creating {}", lib_dest_dir.display()))?;
let lib_dest = lib_dest_dir.join(&lib_filename);
fs::copy(&native_lib, &lib_dest).with_context(|| format!("copying native lib to {}", lib_dest.display()))?;
let mut rb_files: Vec<String> = scan_rb_files(&pkg_dir.join("lib"))
.unwrap_or_default()
.into_iter()
.filter_map(|p| p.strip_prefix(&pkg_dir).ok().map(|r| r.to_string_lossy().into_owned()))
.collect();
rb_files.sort();
let native_lib_path = format!("lib/{gem_name}/{lib_filename}");
if !rb_files.contains(&native_lib_path) {
rb_files.push(native_lib_path);
}
let gemspec_name = format!("{gem_name}-platform.gemspec");
let gemspec_path = pkg_dir.join(&gemspec_name);
let platform_gemspec = generate_platform_gemspec(&gem_name, version, &platform, &rb_files)?;
fs::write(&gemspec_path, platform_gemspec)?;
let build_cmd = format!("gem build {gemspec_name}");
crate::publish::run_shell_command_in(&build_cmd, &pkg_dir)?;
let gem_file = find_gem_file(&pkg_dir, &gem_name, version, &platform)
.with_context(|| format!("gem build did not produce expected .gem in {}", pkg_dir.display()))?;
let gem_filename = gem_file
.file_name()
.context("gem has no filename")?
.to_string_lossy()
.to_string();
let dest = output_dir.join(&gem_filename);
fs::copy(&gem_file, &dest)?;
let _ = fs::remove_file(&gemspec_path);
let _ = fs::remove_file(&lib_dest);
Ok(PackageArtifact {
path: dest,
name: gem_filename,
checksum: None,
})
}
fn find_ruby_native_lib(
workspace_root: &Path,
target: &RustTarget,
rb_crate: &str,
lib_filename: &str,
) -> Result<PathBuf> {
let cross = workspace_root
.join("target")
.join(&target.triple)
.join("release")
.join(lib_filename);
if cross.exists() {
return Ok(cross);
}
let native = workspace_root.join("target/release").join(lib_filename);
if native.exists() {
return Ok(native);
}
let in_crate = workspace_root
.join("crates")
.join(rb_crate)
.join("target")
.join("release")
.join(lib_filename);
if in_crate.exists() {
return Ok(in_crate);
}
anyhow::bail!(
"Ruby native lib '{lib_filename}' not found in target dirs for {}",
target.triple
)
}
fn scan_rb_files(lib_dir: &Path) -> Result<Vec<PathBuf>> {
let mut found = Vec::new();
if !lib_dir.exists() {
return Ok(found);
}
for entry in fs::read_dir(lib_dir).with_context(|| format!("reading {}", lib_dir.display()))? {
let entry = entry?;
let path = entry.path();
if path.is_dir() {
for sub in fs::read_dir(&path).with_context(|| format!("reading {}", path.display()))? {
let sub = sub?;
let sub_path = sub.path();
if sub_path.extension().is_some_and(|e| e == "rb") {
found.push(sub_path);
}
}
} else if path.extension().is_some_and(|e| e == "rb") {
found.push(path);
}
}
Ok(found)
}
fn generate_platform_gemspec(gem_name: &str, version: &str, platform: &str, files: &[String]) -> Result<String> {
let files_ruby = files
.iter()
.map(|f| format!(" {f:?}"))
.collect::<Vec<_>>()
.join(",\n");
Ok(format!(
r#"# frozen_string_literal: true
Gem::Specification.new do |spec|
spec.name = {gem_name:?}
spec.version = {version:?}
spec.platform = {platform:?}
spec.summary = "{gem_name} native extension"
spec.files = [
{files_ruby}
]
spec.require_paths = ["lib"]
end
"#
))
}
fn find_gem_file(dir: &Path, gem_name: &str, version: &str, platform: &str) -> Result<PathBuf> {
let expected = dir.join(format!("{gem_name}-{version}-{platform}.gem"));
if expected.exists() {
return Ok(expected);
}
let candidates: Vec<PathBuf> = fs::read_dir(dir)?
.filter_map(|e| e.ok())
.map(|e| e.path())
.filter(|p| {
p.extension().is_some_and(|e| e == "gem")
&& p.file_name().is_some_and(|n| n.to_string_lossy().contains(version))
})
.collect();
candidates
.into_iter()
.next()
.with_context(|| format!("no .gem file for {gem_name}-{version} found in {}", dir.display()))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn generate_platform_gemspec_includes_native_and_wrapper_files() {
let files = vec![
"lib/mylib.rb".to_string(),
"lib/mylib/version.rb".to_string(),
"lib/mylib/native.rb".to_string(),
"lib/mylib/libmylib_rb.so".to_string(),
];
let spec = generate_platform_gemspec("mylib", "1.0.0", "x86_64-linux", &files).unwrap();
assert!(spec.contains("mylib"), "gem name present");
assert!(spec.contains("1.0.0"), "version present");
assert!(spec.contains("x86_64-linux"), "platform present");
assert!(spec.contains("libmylib_rb.so"), "native lib present");
assert!(spec.contains("lib/mylib.rb"), "top-level wrapper present");
assert!(spec.contains("lib/mylib/version.rb"), "version wrapper present");
assert!(spec.contains("lib/mylib/native.rb"), "native wrapper present");
}
#[test]
fn scan_rb_files_finds_wrappers_and_skips_non_rb() {
let tmp = tempfile::TempDir::new().unwrap();
let lib_dir = tmp.path().join("lib");
let sub_dir = lib_dir.join("mylib");
std::fs::create_dir_all(&sub_dir).unwrap();
std::fs::write(lib_dir.join("mylib.rb"), b"").unwrap();
std::fs::write(sub_dir.join("version.rb"), b"").unwrap();
std::fs::write(sub_dir.join("native.rb"), b"").unwrap();
std::fs::write(sub_dir.join("libmylib_rb.so"), b"").unwrap();
let mut found = scan_rb_files(&lib_dir).unwrap();
found.sort();
let names: Vec<String> = found
.iter()
.map(|p| p.file_name().unwrap().to_string_lossy().into_owned())
.collect();
assert!(names.contains(&"mylib.rb".to_string()), "top-level wrapper found");
assert!(names.contains(&"version.rb".to_string()), "version.rb found");
assert!(names.contains(&"native.rb".to_string()), "native.rb found");
assert!(!names.contains(&"libmylib_rb.so".to_string()), ".so excluded from scan");
}
#[test]
fn find_gem_file_expected_path() {
let tmp = tempfile::TempDir::new().unwrap();
let gem_path = tmp.path().join("mygem-1.0.0-x86_64-linux.gem");
std::fs::write(&gem_path, b"fake").unwrap();
let result = find_gem_file(tmp.path(), "mygem", "1.0.0", "x86_64-linux").unwrap();
assert_eq!(result, gem_path);
}
#[test]
fn find_gem_file_missing_errors() {
let tmp = tempfile::TempDir::new().unwrap();
let result = find_gem_file(tmp.path(), "mygem", "1.0.0", "x86_64-linux");
assert!(result.is_err());
}
}