use colored::{ColoredString, Colorize};
use include_dir::{include_dir, Dir};
use lazy_static::lazy_static;
use pax_manifest::HostCrateInfo;
use pax_runtime::api::serde::Deserialize;
use std::fs;
use std::path::{Path, PathBuf};
use std::process::Command;
use std::str::FromStr;
use std::sync::{Arc, Mutex};
use syn::spanned::Spanned;
use syn::visit::Visit;
use syn::{parse_file, ItemStruct};
use toml_edit;
use toml_edit::Document;
lazy_static! {
#[allow(non_snake_case)]
pub static ref PAX_BADGE: ColoredString = "[Pax]".bold().on_black().white();
pub static ref DIR_IGNORE_LIST_MACOS : Vec<&'static str> = vec!["target", ".build", ".git", "tests"];
pub static ref DIR_IGNORE_LIST_WEB : Vec<&'static str> = vec![".git"];
}
pub static PAX_CREATE_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/new-project/new-project-template");
pub static PAX_WEB_INTERFACE_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/interfaces/web/public/");
pub static PAX_MACOS_INTERFACE_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/interfaces/macos/");
pub static PAX_IOS_INTERFACE_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/interfaces/ios/");
pub static PAX_SWIFT_CARTRIDGE_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/swift/pax-swift-cartridge/");
pub static PAX_SWIFT_COMMON_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/files/swift/pax-swift-common/");
pub const PAX_CREATE_LIBDEV_TEMPLATE_DIR_NAME: &str = "new-libdev-project-template";
pub const INTERFACE_DIR_NAME: &str = "interface";
pub const BUILD_DIR_NAME: &str = "build";
pub const PUBLIC_DIR_NAME: &str = "public";
pub const ASSETS_DIR_NAME: &str = "assets";
pub const ERR_SPAWN: &str = "failed to spawn child";
pub const ALL_PKGS: &[&str] = &[
"pax-chassis-common",
"pax-chassis-ios",
"pax-chassis-macos",
"pax-chassis-web",
"pax-cli",
"pax-compiler",
"pax-designtime",
"pax-kit",
"pax-runtime",
"pax-runtime-api",
"pax-engine",
"pax-macro",
"pax-message",
"pax-std",
"pax-manifest",
"pax-lang",
];
#[derive(Debug, Deserialize)]
struct Metadata {
packages: Vec<Package>,
}
#[derive(Debug, Deserialize)]
struct Package {
name: String,
version: String,
}
pub fn set_path_on_pax_dependencies(full_path: &Path) {
let mut doc = fs::read_to_string(&full_path.join("Cargo.toml"))
.expect("Failed to read Cargo.toml")
.parse::<toml_edit::Document>()
.expect("Failed to parse Cargo.toml");
if let Some(deps) = doc
.as_table_mut()
.entry("dependencies")
.or_insert_with(toml_edit::table)
.as_table_mut()
{
let keys: Vec<String> = deps
.iter()
.filter_map(|(key, _)| {
if key.starts_with("pax-") {
Some(key.to_string())
} else {
None
}
})
.collect();
for key in keys {
let dep_entry = deps.get_mut(&key).unwrap();
if let toml_edit::Item::Value(toml_edit::Value::InlineTable(ref mut dep_table)) =
dep_entry
{
dep_table.insert(
"path",
toml_edit::Value::String(toml_edit::Formatted::new(
".pax/pkg/".to_string() + &key,
)),
);
}
}
}
fs::write(&full_path.join("Cargo.toml"), doc.to_string())
.expect("Failed to write modified Cargo.toml");
}
pub fn update_pax_dependency_versions(doc: &mut Document, ctx_version: &str) {
if let Some(deps) = doc
.as_table_mut()
.entry("dependencies")
.or_insert_with(toml_edit::table)
.as_table_mut()
{
let keys: Vec<String> = deps
.iter()
.filter_map(|(key, _)| {
if key.starts_with("pax-") {
Some(key.to_string())
} else {
None
}
})
.collect();
for key in keys {
let dep_entry = deps.get_mut(&key).unwrap();
if let toml_edit::Item::Value(toml_edit::Value::InlineTable(ref mut dep_table)) =
dep_entry
{
dep_table.insert(
"version",
toml_edit::Value::String(toml_edit::Formatted::new(ctx_version.to_string())),
);
} else {
let dep_string = format!("version = \"{}\"", ctx_version);
*dep_entry = toml_edit::Item::from_str(&dep_string).unwrap_or_default();
}
}
}
}
const ERR_LOCK: &str = "Failed to lock process_child_ids mutex";
pub fn wait_with_output(
process_child_ids: &Arc<Mutex<Vec<u64>>>,
child: std::process::Child,
) -> std::process::Output {
let child_id: u64 = child.id().into();
process_child_ids.lock().expect(ERR_LOCK).push(child_id);
let output = child
.wait_with_output()
.expect("Failed to wait for child process");
process_child_ids
.lock()
.expect(ERR_LOCK)
.retain(|&id| id != child_id);
output
}
pub fn get_or_create_pax_directory(project_path: &PathBuf) -> PathBuf {
let working_path = std::path::Path::new(project_path).join(".pax");
std::fs::create_dir_all(&working_path).unwrap();
fs::canonicalize(working_path).unwrap()
}
pub fn get_version_of_whitelisted_packages(path: &str) -> Result<String, &'static str> {
let mut cmd = Command::new("cargo");
let output = cmd
.arg("metadata")
.arg("--format-version=1")
.current_dir(path)
.output()
.expect("Failed to execute `cargo metadata`");
if !output.status.success() {
eprintln!("{}", String::from_utf8_lossy(&output.stderr));
panic!("Failed to get metadata from Cargo");
}
let metadata: Metadata =
serde_json::from_slice(&output.stdout).expect("Failed to parse JSON from `cargo metadata`");
let mut tracked_version: Option<String> = None;
for package in &metadata.packages {
if ALL_PKGS.contains(&package.name.as_str()) {
if let Some(ref version) = tracked_version {
if package.version != *version {
panic!(
"Version mismatch for {}: expected {}, found {}",
package.name, version, package.version
);
}
} else {
tracked_version = Some(package.version.clone());
}
}
}
tracked_version.ok_or("Cannot build a Pax project without a `pax-*` dependency somewhere in your project's dependency graph. Add e.g. `pax-engine` to your Cargo.toml to resolve this error.")
}
pub fn copy_dir_recursively(
src: &Path,
dest: &Path,
ignore_list: &[&str],
) -> Result<(), Box<dyn std::error::Error>> {
if src.is_dir() {
if ignore_list.contains(&src.file_name().unwrap().to_str().unwrap()) {
return Ok(());
}
fs::create_dir_all(dest)?;
for entry in fs::read_dir(src)? {
let entry = entry?;
let path = entry.path();
let dest_child = dest.join(path.file_name().ok_or("Invalid file name")?);
copy_dir_recursively(&path, &dest_child, ignore_list)?;
}
} else {
fs::copy(src, dest)?;
}
Ok(())
}
pub fn get_host_crate_info(cargo_toml_path: &Path) -> HostCrateInfo {
let existing_cargo_toml = toml_edit::Document::from_str(
&fs::read_to_string(fs::canonicalize(cargo_toml_path).unwrap()).unwrap(),
)
.expect("Error loading host Cargo.toml");
let name = existing_cargo_toml["package"]["name"]
.as_str()
.unwrap()
.to_string();
let identifier = name.replace("-", "_"); let import_prefix = format!("{}::pax_reexports::", &identifier);
HostCrateInfo {
name,
identifier,
import_prefix,
}
}