use std::collections::HashMap; use std::path::PathBuf; use serde::{Deserialize, Serialize}; use serde_with::{serde_as, skip_serializing_none}; mod util; #[serde_as] #[skip_serializing_none] #[derive(Serialize, Deserialize)] #[serde(rename_all = "camelCase")] pub struct PackageJson { pub name: String, pub version: String, pub author: Option, #[serde(default = "default_as_false")] #[serde(skip_serializing_if = "is_false")] pub private: bool, pub license: Option, pub description: Option, pub main: Option, pub repository: Option, pub scripts: Option, pub dependencies: Option, pub dev_dependencies: Option, } // serde :/ #[allow(clippy::trivially_copy_pass_by_ref)] fn is_false(value: &bool) -> bool { !value } fn default_as_false() -> bool { false } pub type Scripts = HashMap; pub type Dependencies = HashMap; #[derive(thiserror::Error, Debug)] pub enum Error { #[error("deserialization error: {0}")] Serde(#[from] serde_json::Error), #[cfg(feature = "tokio")] #[error("package.json not found")] NotFound, #[cfg(feature = "tokio")] #[error("I/O error: {0}")] Io(#[from] std::io::Error), #[cfg(feature = "tokio")] #[error("file is invalid utf-8")] Utf8(#[from] std::string::FromUtf8Error), } impl PackageJson { pub fn parse(json: &str) -> Result { Ok(serde_json::from_str(json)?) } #[cfg(feature = "tokio")] pub async fn from_file(max_attempts: usize) -> Result<(Self, PathBuf), Error> { let path = util::find_package_json(max_attempts).await?; let Some(path) = path else { return Err(Error::NotFound); }; let file = tokio::fs::read(path.clone()).await?; let file = String::from_utf8(file)?; Ok((Self::parse(&file)?, path)) } }