#![forbid(unsafe_code)]
#![doc = include_str!("../README.md")]
use core::{fmt, str::FromStr};
use std::{collections::BTreeMap, error::Error};
macro_rules! text_newtype {
($name:ident) => {
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[derive(Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub struct $name(String);
impl $name {
pub fn new(input: &str) -> Result<Self, PackageJsonTextError> {
let trimmed = input.trim();
if trimmed.is_empty() {
Err(PackageJsonTextError::Empty)
} else {
Ok(Self(trimmed.to_string()))
}
}
#[must_use]
pub fn as_str(&self) -> &str {
&self.0
}
}
impl fmt::Display for $name {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.write_str(self.as_str())
}
}
impl FromStr for $name {
type Err = PackageJsonTextError;
fn from_str(input: &str) -> Result<Self, Self::Err> {
Self::new(input)
}
}
};
}
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[derive(Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub struct PackageName(String);
impl PackageName {
pub fn new(input: &str) -> Result<Self, PackageJsonTextError> {
let trimmed = input.trim();
if trimmed.is_empty() {
return Err(PackageJsonTextError::Empty);
}
if trimmed.chars().any(char::is_whitespace) {
return Err(PackageJsonTextError::ContainsWhitespace);
}
if let Some(rest) = trimmed.strip_prefix('@') {
let Some((scope, name)) = rest.split_once('/') else {
return Err(PackageJsonTextError::InvalidScopedName);
};
if scope.is_empty() || name.is_empty() || name.contains('/') {
return Err(PackageJsonTextError::InvalidScopedName);
}
} else if trimmed.contains('/') {
return Err(PackageJsonTextError::InvalidName);
}
Ok(Self(trimmed.to_string()))
}
#[must_use]
pub fn as_str(&self) -> &str {
&self.0
}
#[must_use]
pub fn is_scoped(&self) -> bool {
self.0.starts_with('@')
}
}
impl fmt::Display for PackageName {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.write_str(self.as_str())
}
}
impl FromStr for PackageName {
type Err = PackageJsonTextError;
fn from_str(input: &str) -> Result<Self, Self::Err> {
Self::new(input)
}
}
impl TryFrom<&str> for PackageName {
type Error = PackageJsonTextError;
fn try_from(value: &str) -> Result<Self, Self::Error> {
Self::new(value)
}
}
text_newtype!(PackageVersion);
text_newtype!(PackageScriptName);
text_newtype!(PackageScript);
pub type DependencyMap = BTreeMap<PackageName, PackageVersion>;
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub enum DependencyKind {
Dependencies,
DevDependencies,
PeerDependencies,
OptionalDependencies,
BundleDependencies,
}
impl DependencyKind {
#[must_use]
pub const fn as_str(self) -> &'static str {
match self {
Self::Dependencies => "dependencies",
Self::DevDependencies => "devDependencies",
Self::PeerDependencies => "peerDependencies",
Self::OptionalDependencies => "optionalDependencies",
Self::BundleDependencies => "bundleDependencies",
}
}
}
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub enum PackageType {
Module,
CommonJs,
}
impl PackageType {
#[must_use]
pub const fn as_str(self) -> &'static str {
match self {
Self::Module => "module",
Self::CommonJs => "commonjs",
}
}
}
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[derive(Clone, Debug, Default, Eq, PartialEq)]
pub struct PackageJson {
name: Option<PackageName>,
version: Option<PackageVersion>,
package_type: Option<PackageType>,
scripts: BTreeMap<PackageScriptName, PackageScript>,
dependencies: BTreeMap<DependencyKind, DependencyMap>,
}
impl PackageJson {
#[must_use]
pub fn new() -> Self {
Self::default()
}
#[must_use]
pub fn with_name(mut self, name: PackageName) -> Self {
self.name = Some(name);
self
}
#[must_use]
pub fn with_version(mut self, version: PackageVersion) -> Self {
self.version = Some(version);
self
}
#[must_use]
pub const fn with_package_type(mut self, package_type: PackageType) -> Self {
self.package_type = Some(package_type);
self
}
#[must_use]
pub fn with_script(mut self, name: PackageScriptName, script: PackageScript) -> Self {
self.scripts.insert(name, script);
self
}
#[must_use]
pub fn with_dependency(
mut self,
kind: DependencyKind,
name: PackageName,
version: PackageVersion,
) -> Self {
self.dependencies
.entry(kind)
.or_default()
.insert(name, version);
self
}
#[must_use]
pub const fn name(&self) -> Option<&PackageName> {
self.name.as_ref()
}
#[must_use]
pub const fn version(&self) -> Option<&PackageVersion> {
self.version.as_ref()
}
#[must_use]
pub const fn package_type(&self) -> Option<PackageType> {
self.package_type
}
#[must_use]
pub const fn scripts(&self) -> &BTreeMap<PackageScriptName, PackageScript> {
&self.scripts
}
#[must_use]
pub const fn dependencies(&self) -> &BTreeMap<DependencyKind, DependencyMap> {
&self.dependencies
}
}
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum PackageJsonTextError {
Empty,
ContainsWhitespace,
InvalidScopedName,
InvalidName,
}
impl fmt::Display for PackageJsonTextError {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Empty => formatter.write_str("package metadata text cannot be empty"),
Self::ContainsWhitespace => {
formatter.write_str("package metadata text cannot contain whitespace")
}
Self::InvalidScopedName => {
formatter.write_str("scoped package names must look like @scope/name")
}
Self::InvalidName => formatter.write_str("package name has an invalid shape"),
}
}
}
impl Error for PackageJsonTextError {}
#[cfg(test)]
mod tests {
use super::{
DependencyKind, PackageJson, PackageJsonTextError, PackageName, PackageScript,
PackageScriptName, PackageType, PackageVersion,
};
#[test]
fn validates_package_names() -> Result<(), PackageJsonTextError> {
let scoped = PackageName::new("@rustuse/example")?;
assert!(scoped.is_scoped());
assert_eq!(
PackageName::new("bad name"),
Err(PackageJsonTextError::ContainsWhitespace)
);
assert_eq!(
PackageName::new("@scope"),
Err(PackageJsonTextError::InvalidScopedName)
);
Ok(())
}
#[test]
fn stores_package_metadata() -> Result<(), PackageJsonTextError> {
let manifest = PackageJson::new()
.with_name(PackageName::new("demo")?)
.with_version(PackageVersion::new("0.1.0")?)
.with_package_type(PackageType::Module)
.with_script(
PackageScriptName::new("test")?,
PackageScript::new("vitest")?,
)
.with_dependency(
DependencyKind::Dependencies,
PackageName::new("react")?,
PackageVersion::new("^18")?,
);
assert_eq!(manifest.name().map(PackageName::as_str), Some("demo"));
assert_eq!(manifest.scripts().len(), 1);
assert_eq!(manifest.dependencies().len(), 1);
Ok(())
}
}