use std::path::{Path, PathBuf};
use thiserror::Error;
#[derive(Debug, Error)]
pub enum ComposerError {
#[error("composer I/O error: {0}")]
Io(#[from] std::io::Error),
#[error("composer JSON error: {0}")]
Json(#[from] serde_json::Error),
#[error("composer.json has no autoload section")]
MissingAutoload,
}
#[derive(Clone)]
pub struct Psr4Map {
project_entries: Vec<(String, PathBuf)>,
vendor_entries: Vec<(String, PathBuf)>,
project_extra_paths: Vec<PathBuf>,
vendor_extra_paths: Vec<PathBuf>,
#[allow(dead_code)] root: PathBuf,
}
fn ensure_trailing_backslash(prefix: &str) -> String {
if prefix.ends_with('\\') {
prefix.to_string()
} else {
format!("{prefix}\\")
}
}
fn collect_prefix_dirs(
value: &serde_json::Value,
prefix: &str,
base: &Path,
entries: &mut Vec<(String, PathBuf)>,
) {
let pfx = ensure_trailing_backslash(prefix);
if let Some(d) = value.as_str() {
entries.push((pfx, base.join(d)));
} else if let Some(arr) = value.as_array() {
for item in arr {
if let Some(d) = item.as_str() {
entries.push((pfx.clone(), base.join(d)));
}
}
}
}
fn collect_path_array(value: &serde_json::Value, base: &Path, out: &mut Vec<PathBuf>) {
if let Some(arr) = value.as_array() {
for item in arr {
if let Some(s) = item.as_str() {
out.push(base.join(s));
}
}
}
}
fn parse_autoload_section(
autoload: &serde_json::Value,
base: &Path,
entries: &mut Vec<(String, PathBuf)>,
extras: &mut Vec<PathBuf>,
) {
if let Some(map) = autoload.get("psr-4").and_then(|v| v.as_object()) {
for (prefix, dir) in map {
collect_prefix_dirs(dir, prefix, base, entries);
}
}
if let Some(map) = autoload.get("psr-0").and_then(|v| v.as_object()) {
for (_, dir) in map {
if let Some(d) = dir.as_str() {
extras.push(base.join(d));
} else if let Some(arr) = dir.as_array() {
for item in arr {
if let Some(d) = item.as_str() {
extras.push(base.join(d));
}
}
}
}
}
if let Some(cm) = autoload.get("classmap") {
collect_path_array(cm, base, extras);
}
if let Some(files) = autoload.get("files") {
collect_path_array(files, base, extras);
}
}
fn parse_vendor(root: &Path, entries: &mut Vec<(String, PathBuf)>, extras: &mut Vec<PathBuf>) {
let installed_path = root.join("vendor/composer/installed.json");
let content = match std::fs::read_to_string(&installed_path) {
Ok(c) => c,
Err(_) => return,
};
let value: serde_json::Value = match serde_json::from_str(&content) {
Ok(v) => v,
Err(_) => return,
};
let packages = if let Some(arr) = value.get("packages").and_then(|v| v.as_array()) {
arr.clone()
} else if let Some(arr) = value.as_array() {
arr.clone()
} else {
return;
};
let vendor_dir = root.join("vendor");
for pkg in &packages {
let pkg_name = pkg.get("name").and_then(|v| v.as_str()).unwrap_or("");
let pkg_dir = vendor_dir.join(pkg_name);
if let Some(autoload) = pkg.get("autoload") {
parse_autoload_section(autoload, &pkg_dir, entries, extras);
}
}
}
impl Psr4Map {
pub fn from_composer(root: &Path) -> Result<Self, ComposerError> {
let composer_path = root.join("composer.json");
let content = std::fs::read_to_string(&composer_path)?;
let value: serde_json::Value = serde_json::from_str(&content)?;
let has_autoload = value.get("autoload").is_some() || value.get("autoload-dev").is_some();
if !has_autoload {
return Err(ComposerError::MissingAutoload);
}
let mut project_entries: Vec<(String, PathBuf)> = Vec::new();
let mut project_extra_paths: Vec<PathBuf> = Vec::new();
if let Some(autoload) = value.get("autoload") {
parse_autoload_section(
autoload,
root,
&mut project_entries,
&mut project_extra_paths,
);
}
if let Some(autoload) = value.get("autoload-dev") {
parse_autoload_section(
autoload,
root,
&mut project_entries,
&mut project_extra_paths,
);
}
project_entries.sort_by_key(|b| std::cmp::Reverse(b.0.len()));
let mut vendor_entries: Vec<(String, PathBuf)> = Vec::new();
let mut vendor_extra_paths: Vec<PathBuf> = Vec::new();
parse_vendor(root, &mut vendor_entries, &mut vendor_extra_paths);
vendor_entries.sort_by_key(|b| std::cmp::Reverse(b.0.len()));
Ok(Psr4Map {
project_entries,
vendor_entries,
project_extra_paths,
vendor_extra_paths,
root: root.to_path_buf(),
})
}
pub fn project_files(&self) -> Vec<PathBuf> {
let mut out = Vec::new();
for (_, dir) in &self.project_entries {
crate::project::collect_php_files(dir, &mut out);
}
for path in &self.project_extra_paths {
collect_php_path(path, &mut out);
}
out
}
pub fn vendor_files(&self) -> Vec<PathBuf> {
let mut out = Vec::new();
for (_, dir) in &self.vendor_entries {
crate::project::collect_php_files(dir, &mut out);
}
for path in &self.vendor_extra_paths {
collect_php_path(path, &mut out);
}
out
}
pub fn resolve(&self, fqcn: &str) -> Option<PathBuf> {
for (prefix, dir) in self
.project_entries
.iter()
.chain(self.vendor_entries.iter())
{
if fqcn.starts_with(prefix.as_str()) {
let relative = &fqcn[prefix.len()..];
let file_path = dir.join(relative.replace('\\', "/")).with_extension("php");
if file_path.exists() {
return Some(file_path);
}
}
}
None
}
}
fn collect_php_path(path: &Path, out: &mut Vec<PathBuf>) {
let Ok(meta) = std::fs::metadata(path) else {
return;
};
if meta.is_file() {
if path.extension().and_then(|e| e.to_str()) == Some("php") {
out.push(path.to_path_buf());
}
} else if meta.is_dir() {
crate::project::collect_php_files(path, out);
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::fs;
fn make_temp_project(name: &str) -> PathBuf {
let dir = std::env::temp_dir().join(format!("mir_psr4_{name}"));
let _ = fs::remove_dir_all(&dir);
fs::create_dir_all(&dir).unwrap();
dir
}
#[test]
fn parse_project_entries() {
let root = make_temp_project("parse_project_entries");
fs::write(
root.join("composer.json"),
r#"{
"autoload": {
"psr-4": { "App\\": "src/", "App\\Models\\": "src/models/" }
},
"autoload-dev": {
"psr-4": { "Tests\\": "tests/" }
}
}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let prefixes: Vec<&str> = map
.project_entries
.iter()
.map(|(p, _)| p.as_str())
.collect();
assert!(prefixes.contains(&"App\\Models\\"), "missing App\\Models\\");
assert!(prefixes.contains(&"App\\"), "missing App\\");
assert!(prefixes.contains(&"Tests\\"), "missing Tests\\");
}
#[test]
fn longest_prefix_first() {
let root = make_temp_project("longest_prefix_first");
fs::write(
root.join("composer.json"),
r#"{
"autoload": {
"psr-4": { "App\\": "src/", "App\\Models\\": "src/models/" }
}
}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
assert_eq!(map.project_entries[0].0, "App\\Models\\");
}
#[test]
fn missing_autoload_section_is_error() {
let root = make_temp_project("missing_autoload");
fs::write(root.join("composer.json"), r#"{ "name": "my/pkg" }"#).unwrap();
let result = Psr4Map::from_composer(&root);
assert!(
matches!(result, Err(ComposerError::MissingAutoload)),
"expected MissingAutoload error"
);
}
#[test]
fn composer_v2_installed() {
let root = make_temp_project("composer_v2");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let vendor_dir = root.join("vendor/composer");
fs::create_dir_all(&vendor_dir).unwrap();
fs::write(
vendor_dir.join("installed.json"),
r#"{
"packages": [
{
"name": "vendor/pkg",
"autoload": { "psr-4": { "Vendor\\Pkg\\": "src/" } }
}
]
}"#,
)
.unwrap();
fs::create_dir_all(root.join("vendor/vendor/pkg/src")).unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let prefixes: Vec<&str> = map.vendor_entries.iter().map(|(p, _)| p.as_str()).collect();
assert!(prefixes.contains(&"Vendor\\Pkg\\"), "missing Vendor\\Pkg\\");
}
#[test]
fn composer_v1_installed() {
let root = make_temp_project("composer_v1");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let vendor_dir = root.join("vendor/composer");
fs::create_dir_all(&vendor_dir).unwrap();
fs::write(
vendor_dir.join("installed.json"),
r#"[
{
"name": "vendor/pkg",
"autoload": { "psr-4": { "Vendor\\Pkg\\": "src/" } }
}
]"#,
)
.unwrap();
fs::create_dir_all(root.join("vendor/vendor/pkg/src")).unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let prefixes: Vec<&str> = map.vendor_entries.iter().map(|(p, _)| p.as_str()).collect();
assert!(prefixes.contains(&"Vendor\\Pkg\\"), "missing Vendor\\Pkg\\");
}
#[test]
fn missing_installed_json() {
let root = make_temp_project("missing_installed");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
assert!(map.vendor_entries.is_empty());
}
#[test]
fn project_files_returns_php_files() {
let root = make_temp_project("project_files");
let src = root.join("src");
fs::create_dir_all(&src).unwrap();
fs::write(src.join("Foo.php"), "<?php class Foo {}").unwrap();
fs::write(src.join("README.md"), "not php").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.project_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("Foo.php"));
}
#[test]
fn resolve_existing_file() {
let root = make_temp_project("resolve_existing");
let models = root.join("src/models");
fs::create_dir_all(&models).unwrap();
fs::write(models.join("User.php"), "<?php class User {}").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\Models\\":"src/models/","App\\":"src/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let result = map.resolve("App\\Models\\User");
assert!(result.is_some(), "expected a resolved path");
assert!(result.unwrap().ends_with("User.php"));
}
#[test]
fn resolve_missing_file() {
let root = make_temp_project("resolve_missing");
fs::create_dir_all(root.join("src")).unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let result = map.resolve("App\\Models\\User");
assert!(result.is_none());
}
#[test]
fn boundary_check() {
let root = make_temp_project("boundary_check");
fs::create_dir_all(root.join("src")).unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let result = map.resolve("Application\\Foo");
assert!(
result.is_none(),
"App\\ prefix must not match Application\\Foo"
);
}
#[test]
fn array_valued_psr4_dirs() {
let root = make_temp_project("array_dirs");
let src = root.join("src");
let lib = root.join("lib");
fs::create_dir_all(&src).unwrap();
fs::create_dir_all(&lib).unwrap();
fs::write(src.join("Foo.php"), "<?php class Foo {}").unwrap();
fs::write(lib.join("Bar.php"), "<?php class Bar {}").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":["src/","lib/"]}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
assert_eq!(
map.project_entries.len(),
2,
"expected 2 entries for array-valued dir"
);
let files = map.project_files();
assert_eq!(files.len(), 2, "expected Foo.php and Bar.php");
}
#[test]
fn project_classmap_dir_is_collected() {
let root = make_temp_project("project_classmap");
let lib = root.join("lib");
fs::create_dir_all(&lib).unwrap();
fs::write(lib.join("Legacy.php"), "<?php class Legacy {}").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"classmap":["lib/"]}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.project_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("Legacy.php"));
}
#[test]
fn project_files_autoload_is_collected() {
let root = make_temp_project("project_files_autoload");
fs::write(root.join("helpers.php"), "<?php function my_helper() {}").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"files":["helpers.php"]}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.project_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("helpers.php"));
}
#[test]
fn project_psr0_dir_is_collected() {
let root = make_temp_project("project_psr0");
let lib = root.join("legacy");
fs::create_dir_all(&lib).unwrap();
fs::write(lib.join("Old.php"), "<?php class Old {}").unwrap();
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-0":{"":"legacy/"}}}"#,
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.project_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("Old.php"));
}
#[test]
fn vendor_classmap_is_collected() {
let root = make_temp_project("vendor_classmap");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let vendor_dir = root.join("vendor/composer");
fs::create_dir_all(&vendor_dir).unwrap();
fs::write(
vendor_dir.join("installed.json"),
r#"{
"packages": [{
"name": "vendor/pkg",
"autoload": { "classmap": ["src/"] }
}]
}"#,
)
.unwrap();
let pkg_src = root.join("vendor/vendor/pkg/src");
fs::create_dir_all(&pkg_src).unwrap();
fs::write(pkg_src.join("Legacy.php"), "<?php class Legacy {}").unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.vendor_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("Legacy.php"));
}
#[test]
fn vendor_files_autoload_is_collected() {
let root = make_temp_project("vendor_files_autoload");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let vendor_dir = root.join("vendor/composer");
fs::create_dir_all(&vendor_dir).unwrap();
fs::write(
vendor_dir.join("installed.json"),
r#"{
"packages": [{
"name": "vendor/pkg",
"autoload": { "files": ["bootstrap.php"] }
}]
}"#,
)
.unwrap();
let pkg_dir = root.join("vendor/vendor/pkg");
fs::create_dir_all(&pkg_dir).unwrap();
fs::write(
pkg_dir.join("bootstrap.php"),
"<?php function pkg_bootstrap() {}",
)
.unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.vendor_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("bootstrap.php"));
}
#[test]
fn vendor_psr0_is_collected() {
let root = make_temp_project("vendor_psr0");
fs::write(
root.join("composer.json"),
r#"{"autoload":{"psr-4":{"App\\":"src/"}}}"#,
)
.unwrap();
let vendor_dir = root.join("vendor/composer");
fs::create_dir_all(&vendor_dir).unwrap();
fs::write(
vendor_dir.join("installed.json"),
r#"{
"packages": [{
"name": "vendor/pkg",
"autoload": { "psr-0": { "Old_": "src/" } }
}]
}"#,
)
.unwrap();
let pkg_src = root.join("vendor/vendor/pkg/src/Old");
fs::create_dir_all(&pkg_src).unwrap();
fs::write(pkg_src.join("Thing.php"), "<?php class Old_Thing {}").unwrap();
let map = Psr4Map::from_composer(&root).unwrap();
let files = map.vendor_files();
assert_eq!(files.len(), 1);
assert!(files[0].ends_with("Thing.php"));
}
}