diff --git a/pkgr/Cargo.toml b/pkgr/Cargo.toml index df095d2..92a0e91 100644 --- a/pkgr/Cargo.toml +++ b/pkgr/Cargo.toml @@ -6,4 +6,8 @@ edition = "2021" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -manifest = { path = "../manifest" } \ No newline at end of file +fern = "0.6.2" +getopts = "0.2.21" +log = "0.4.19" +regex = "1.9.1" +manifest = { path = "../manifest" } diff --git a/pkgr/src/commands/mod.rs b/pkgr/src/commands/mod.rs new file mode 100644 index 0000000..61e3fe8 --- /dev/null +++ b/pkgr/src/commands/mod.rs @@ -0,0 +1,8 @@ +use crate::package::identifier::PackageIdentifier; + +pub enum Command { + Grab(PackageIdentifier), // grab a package from a remote source + Remove(PackageIdentifier), // remove a package from the local source + List, // list all packages in the local source + Update, +} \ No newline at end of file diff --git a/pkgr/src/main.rs b/pkgr/src/main.rs index f79c691..8d75350 100644 --- a/pkgr/src/main.rs +++ b/pkgr/src/main.rs @@ -1,2 +1,59 @@ +use std::env; +use std::process::exit; +use std::str::FromStr; + +use getopts::Options; +use log::{info, SetLoggerError}; + +mod commands; +mod package; + fn main() { + setup_logger() + .expect("Unable to setup logger."); + + let args: Vec<String> = env::args().collect(); + let program = args[0].clone(); + + let mut opts = Options::new(); + opts.optflag("h", "help", "Show help message."); + let matches = opts.parse(&args[1..]).unwrap_or_else(|e| panic!("{}", e.to_string())); + + if matches.opt_present("h") { + print_usage(&program, opts); + exit(1); + } + + let command = if !matches.free.is_empty() { + matches.free[0].clone() + } else { + print_usage(&program, opts); + exit(1); + }; + + info!("Identifier: {}", package::identifier::PackageIdentifier::from_str(&command).unwrap()); } + +fn print_usage(program: &str, opts: Options) { + let mut brief = format!("Usage: {} <COMMAND> [options]\n\n", program); + brief += &*format!("Commands: grab \n"); + brief += &*format!(" remove\n"); + print!("{}", opts.usage(&brief.trim_end())); +} + +fn setup_logger() -> Result<(), SetLoggerError> { + fern::Dispatch::new() + .format(|out, message, record| { + out.finish(format_args!( + "{} {}", + match record.level().to_string().chars().nth(0).unwrap_or('I') { + 'E' | 'W' => "!!", + _ => "**", + }, + message + )) + }) + .level(log::LevelFilter::Info) + .chain(std::io::stdout()) + .apply() +} \ No newline at end of file diff --git a/pkgr/src/package/identifier.rs b/pkgr/src/package/identifier.rs new file mode 100644 index 0000000..01f9f6b --- /dev/null +++ b/pkgr/src/package/identifier.rs @@ -0,0 +1,114 @@ +use std::str::FromStr; + +use regex::Regex; + +#[derive(Debug)] +pub enum PackageIdentifierError { + InvalidPackageLocator(String), + InvalidURI(String), +} + +impl std::fmt::Display for PackageIdentifierError { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + match self { + PackageIdentifierError::InvalidPackageLocator(s) => write!(f, "Invalid package locator: {}", s), + PackageIdentifierError::InvalidURI(s) => write!(f, "Invalid URI: {}", s), + } + } +} + +#[derive(Debug)] +pub enum PackageIdentifier { + PackageLocator(PackageLocator), + URI(String), +} + +impl std::fmt::Display for PackageIdentifier { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + match self { + PackageIdentifier::PackageLocator(pl) => write!(f, "{}", pl), + PackageIdentifier::URI(uri) => write!(f, "{}", uri), + } + } +} + +#[derive(Debug)] +pub struct PackageLocator { + pub name: String, + pub version: Option<u32>, + pub tags: Option<Vec<String>>, +} + +impl std::fmt::Display for PackageLocator { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + let mut s = self.name.clone(); + if let Some(version) = self.version { + s += &*format!("@{}", version); + } + if let Some(tags) = &self.tags { + s += &*format!(":{}", tags.join(",")); + } + write!(f, "{}", s) + } +} + +impl FromStr for PackageIdentifier { + type Err = PackageIdentifierError; + + fn from_str(s: &str) -> Result<Self, Self::Err> { + let uri_re = regex::Regex::new(r"^[a-zA-Z0-9]+://").unwrap(); + if uri_re.is_match(s) { + // there needs to be stuff after the protocol + let split = s.split("://").collect::<Vec<&str>>(); + if split.len() != 2 || split[1].is_empty() { + return Err(PackageIdentifierError::InvalidURI(s.to_string())); + } + Ok(PackageIdentifier::URI(s.to_string())) + } else { + let pl = match PackageLocator::from_str(s) { + Ok(pl) => pl, + Err(e) => return Err(PackageIdentifierError::InvalidPackageLocator(e.to_string())), + }; + Ok(PackageIdentifier::PackageLocator(pl)) + } + } +} + +impl FromStr for PackageLocator { + type Err = PackageIdentifierError; + + fn from_str(s: &str) -> Result<Self, Self::Err> { + #[allow(unused_assignments)] // false positive + let mut name = None; + let mut version = None; + let mut tags = None; + + if s.is_empty() { + return Err(PackageIdentifierError::InvalidPackageLocator(s.to_string())); + } + + let name_re = Regex::new("^([A-Za-z][A-Za-z0-9_.]+)").unwrap(); + let version_re = Regex::new("@([0-9]+)").unwrap(); + let tags_re = Regex::new(":([a-zA-Z0-9,._]+)").unwrap(); + + if let Some(caps) = name_re.captures(s) { + name = Some(caps.get(1).unwrap().as_str().to_string()); + } else { + return Err(PackageIdentifierError::InvalidPackageLocator(s.to_string())); + } + + if let Some(caps) = version_re.captures(s) { + version = Some(caps.get(1).unwrap().as_str().parse::<u32>().unwrap()); + } + + if let Some(caps) = tags_re.captures(s) { + tags = Some(caps.get(1).unwrap().as_str().split(",").map(|s| s.to_string()).collect()); + } + + Ok(PackageLocator { + name: name.unwrap(), + version, + tags, + }) + } +} \ No newline at end of file diff --git a/pkgr/src/package/mod.rs b/pkgr/src/package/mod.rs new file mode 100644 index 0000000..8f83cb4 --- /dev/null +++ b/pkgr/src/package/mod.rs @@ -0,0 +1 @@ +pub mod identifier; \ No newline at end of file