Compare commits
No commits in common. "1245db813b38636e919e746af2852b468295e78a" and "0f85701cc40174f0ec4200355234f6358e840411" have entirely different histories.
1245db813b
...
0f85701cc4
51
src/main.rs
51
src/main.rs
|
@ -1,30 +1,7 @@
|
|||
use rustible::Facts;
|
||||
use rustible::modules::{PlaybookRun, Module, Command};
|
||||
|
||||
use rustible::modules::{
|
||||
Module, PlaybookRun,
|
||||
package::{PackageModule as Package, PackageState, PackageError},
|
||||
command::{CommandModule as Command, CommandError},
|
||||
};
|
||||
|
||||
#[derive(Clone, Debug, serde::Serialize)]
|
||||
pub enum PlaybookError {
|
||||
Command(CommandError),
|
||||
Package(PackageError),
|
||||
}
|
||||
|
||||
impl From<CommandError> for PlaybookError {
|
||||
fn from(e: CommandError) -> PlaybookError {
|
||||
PlaybookError::Command(e)
|
||||
}
|
||||
}
|
||||
|
||||
impl From<PackageError> for PlaybookError {
|
||||
fn from(e: PackageError) -> PlaybookError {
|
||||
PlaybookError::Package(e)
|
||||
}
|
||||
}
|
||||
|
||||
fn main() -> Result<(), PlaybookError> {
|
||||
fn main() -> Result<(), rustible::modules::command::CommandError> {
|
||||
let facts = Facts::new();
|
||||
println!("Hello, world! Running system {}", facts.os.family().as_str());
|
||||
|
||||
|
@ -43,30 +20,8 @@ fn main() -> Result<(), PlaybookError> {
|
|||
.arg("/tmp/lol")
|
||||
.build();
|
||||
playbook.run(cmd)?;
|
||||
//let res = cmd.run
|
||||
|
||||
let pkg = Package::new()
|
||||
.name("hello")
|
||||
.state(PackageState::Present)
|
||||
.build();
|
||||
playbook.run(pkg)?;
|
||||
|
||||
let pkg = Package::new()
|
||||
.name("hello")
|
||||
.state(PackageState::Absent)
|
||||
.build();
|
||||
playbook.run(pkg)?;
|
||||
|
||||
let pkg = Package::new()
|
||||
.name(vec!("hello", "sl"))
|
||||
.state(PackageState::Present)
|
||||
.build();
|
||||
playbook.run(pkg)?;
|
||||
|
||||
let pkg = Package::new()
|
||||
.name(&[ "hello", "sl" ])
|
||||
.state(PackageState::Absent)
|
||||
.build();
|
||||
playbook.run(pkg)?;
|
||||
playbook.print_json_pretty();
|
||||
|
||||
Ok(())
|
||||
|
|
|
@ -79,7 +79,6 @@ impl PlaybookRun {
|
|||
|
||||
let module_name = cmd.module_name().to_string();
|
||||
let args = cmd.serialize_args();
|
||||
println!("Running module {} with args:{:#?}\n\n", &module_name, &args);
|
||||
|
||||
let res = cmd.run();
|
||||
|
||||
|
@ -130,4 +129,4 @@ pub trait Module<A: Serialize, S: Serialize, E: Serialize> {
|
|||
|
||||
/// Declare module types
|
||||
pub use command::CommandModule as Command;
|
||||
pub use package::{PackageModule as Package, PackageState};
|
||||
|
||||
|
|
|
@ -2,10 +2,6 @@ use crate::utils::cmd::Cmd;
|
|||
|
||||
use super::{PackageError, PackageList, SpecificPackageManager};
|
||||
|
||||
pub fn apt() -> Cmd {
|
||||
Cmd::new("apt").env("DEBIAN_FRONTEND", "noninteractive")
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct DebianPackageManager;
|
||||
|
||||
|
@ -15,7 +11,7 @@ impl SpecificPackageManager for DebianPackageManager {
|
|||
}
|
||||
|
||||
fn update(&self) -> Result<(), PackageError> {
|
||||
let res = apt().arg("update").run()?;
|
||||
let res = Cmd::new("apt").arg("update").run()?;
|
||||
|
||||
if ! res.success {
|
||||
return Err(PackageError::CmdFail(res));
|
||||
|
@ -24,7 +20,7 @@ impl SpecificPackageManager for DebianPackageManager {
|
|||
}
|
||||
|
||||
fn install(&self, list: PackageList) -> Result<(), PackageError> {
|
||||
let res = apt().arg("install").args(list.list()).run()?;
|
||||
let res = Cmd::new("apt").arg("install").args(list.list()).run()?;
|
||||
|
||||
if ! res.success {
|
||||
return Err(PackageError::CmdFail(res));
|
||||
|
@ -32,12 +28,7 @@ impl SpecificPackageManager for DebianPackageManager {
|
|||
Ok(())
|
||||
}
|
||||
|
||||
fn remove(&self, list: PackageList) -> Result<(), PackageError> {
|
||||
let res = apt().arg("remove").args(list.list()).run()?;
|
||||
|
||||
if ! res.success {
|
||||
return Err(PackageError::CmdFail(res));
|
||||
}
|
||||
Ok(())
|
||||
fn remove(&self, _list: PackageList) -> Result<(), PackageError> {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
|
|
@ -101,17 +101,3 @@ impl PackageArgsBuilder<PackageList, PackageState, Box<dyn SpecificPackageManage
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl PackageArgsBuilder<PackageList, PackageState, NoManager> {
|
||||
pub fn build(self) -> PackageModule<NoManager> {
|
||||
let Self { name, state, .. } = self;
|
||||
|
||||
PackageModule {
|
||||
args: PackageArgs {
|
||||
name,
|
||||
state,
|
||||
},
|
||||
manager: NoManager,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,80 +1,110 @@
|
|||
use serde::Serialize;
|
||||
|
||||
use crate::facts::{Facts, os::OsFamily};
|
||||
use crate::utils::cmd::CmdOutput;
|
||||
|
||||
use std::boxed::Box;
|
||||
|
||||
pub mod pacman;
|
||||
use pacman::ArchlinuxPackageManager;
|
||||
pub mod apt;
|
||||
use apt::DebianPackageManager;
|
||||
|
||||
pub mod builder;
|
||||
use builder::{NoPackage, PackageArgsBuilder};
|
||||
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
pub struct PackageList {
|
||||
pub list: Vec<String>,
|
||||
pub struct PackageArgs {
|
||||
name: PackageList,
|
||||
state: PackageState,
|
||||
}
|
||||
|
||||
impl PackageList {
|
||||
pub fn list(&self) -> & [ String ] {
|
||||
&self.list
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
pub enum PackageState {
|
||||
Present,
|
||||
Absent,
|
||||
Latest,
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct PackageModule<Manager> {
|
||||
manager: Manager,
|
||||
args: PackageArgs,
|
||||
}
|
||||
|
||||
impl<Manager> Module<PackageArgs, (), PackageError> for PackageModule<Manager> {
|
||||
fn clone_args(&self) -> PackageArgs {
|
||||
self.args.clone()
|
||||
}
|
||||
|
||||
fn module_name() -> &'static str {
|
||||
"package"
|
||||
}
|
||||
}
|
||||
|
||||
pub fn add<T: IntoPackageList>(self, add: T) -> PackageList {
|
||||
let Self { mut list } = self;
|
||||
impl Module<PackageArgs, (), PackageError> for PackageModule<Box<dyn SpecificPackageManager>> {
|
||||
fn run(&self) -> Result<(), PackageError> {
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
let extend_with = add.into_package_list();
|
||||
list.extend(extend_with.list);
|
||||
|
||||
PackageList {
|
||||
list
|
||||
impl Module<PackageArgs, (), PackageError> for PackageModule<NoManager> {
|
||||
fn with_facts(self, facts: &Facts) -> PackageModule<Box<dyn SpecificPackageManager>> {
|
||||
let Self { args, .. } = self;
|
||||
|
||||
let manager = match facts.os.family() {
|
||||
OsFamily::Debian => Box::new(DebianPackageManager),
|
||||
OsFamily::Archlinux => Box::new(ArchlinuxPackageManager),
|
||||
};
|
||||
|
||||
PackageModule {
|
||||
args,
|
||||
manager,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Turn a stringy value or list of values into an actual package list.
|
||||
/// Only implemented for &str and &str/String slices because apparently
|
||||
/// there may be a future implementation that turns a slice of stringy values
|
||||
/// into a stringy value, resulting in conflicting implementations.
|
||||
/// https://stackoverflow.com/questions/63136970/how-do-i-work-around-the-upstream-crates-may-add-a-new-impl-of-trait-error
|
||||
pub trait IntoPackageList {
|
||||
fn into_package_list(self) -> PackageList;
|
||||
}
|
||||
|
||||
impl IntoPackageList for &str {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: vec!(self.to_string()),
|
||||
impl PackageModule {
|
||||
pub fn new() -> PackageArgsBuilder {
|
||||
PackageArgsBuilder::new()
|
||||
}
|
||||
|
||||
pub fn from_args_with_facts(args: PackageArgs, facts: &Facts) -> PackageModule {
|
||||
let manager = match facts.os.family() {
|
||||
OsFamily::Debian => PackageModule(Box::new(DebianPackageManager)),
|
||||
OsFamily::Archlinux => PackageModule(Box::new(ArchlinuxPackageManager)),
|
||||
};
|
||||
PackageModule {
|
||||
manager,
|
||||
args,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn from_args_with(args: PackageArgs, manager: SpecificPackageManager) -> PackageManager {
|
||||
PackageModule {
|
||||
manager: Box::new(manager),
|
||||
args,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl IntoPackageList for String {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: vec!(self),
|
||||
}
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
pub enum PackageError {
|
||||
IoError(String),
|
||||
CmdFail(CmdOutput),
|
||||
}
|
||||
|
||||
impl From<std::io::Error> for PackageError {
|
||||
fn from(e: std::io::Error) -> PackageError {
|
||||
PackageError::IoError(e.to_string())
|
||||
}
|
||||
}
|
||||
|
||||
impl<const C: usize> IntoPackageList for & [ &str; C ] {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: self.iter().map(|x| x.to_string()).collect(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<const C: usize> IntoPackageList for & [ String; C ] {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: self.to_vec(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl IntoPackageList for Vec<&str> {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: self.iter().map(|x| x.to_string()).collect(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl IntoPackageList for Vec<String> {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: self,
|
||||
}
|
||||
}
|
||||
pub trait SpecificPackageManager: std::fmt::Debug {
|
||||
fn name(&self) -> &'static str;
|
||||
fn update(&self) -> Result<(), PackageError>;
|
||||
fn install(&self, list: PackageList) -> Result<(), PackageError>;
|
||||
fn remove(&self, list: PackageList) -> Result<(), PackageError>;
|
||||
}
|
||||
|
|
|
@ -6,16 +6,13 @@ use crate::utils::cmd::CmdOutput;
|
|||
|
||||
use std::boxed::Box;
|
||||
|
||||
pub mod list;
|
||||
pub use list::{PackageList, IntoPackageList};
|
||||
|
||||
pub mod pacman;
|
||||
use pacman::ArchlinuxPackageManager;
|
||||
pub mod apt;
|
||||
use apt::DebianPackageManager;
|
||||
|
||||
pub mod builder;
|
||||
pub use builder::{PackageArgsBuilder, NoManager};
|
||||
pub use builder::PackageArgsBuilder;
|
||||
|
||||
pub type PackageStatus = ();
|
||||
|
||||
|
@ -38,6 +35,10 @@ pub struct PackageModule<Manager> {
|
|||
args: PackageArgs,
|
||||
}
|
||||
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct NoManager;
|
||||
|
||||
impl Module<PackageArgs, (), PackageError> for PackageModule<Box<dyn SpecificPackageManager>> {
|
||||
fn serialize_args(&self) -> serde_json::Value {
|
||||
serde_json::to_value(&self.args).unwrap()
|
||||
|
@ -48,17 +49,7 @@ impl Module<PackageArgs, (), PackageError> for PackageModule<Box<dyn SpecificPac
|
|||
}
|
||||
|
||||
fn run(self) -> Result<(), PackageError> {
|
||||
let packages = self.args.name;
|
||||
|
||||
match &self.args.state {
|
||||
PackageState::Present => {
|
||||
self.manager.install(packages)
|
||||
}, PackageState::Absent => {
|
||||
self.manager.remove(packages)
|
||||
}, PackageState::Latest => {
|
||||
unimplemented!()
|
||||
}
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -102,4 +93,39 @@ pub trait SpecificPackageManager: std::fmt::Debug {
|
|||
fn update(&self) -> Result<(), PackageError>;
|
||||
fn install(&self, list: PackageList) -> Result<(), PackageError>;
|
||||
fn remove(&self, list: PackageList) -> Result<(), PackageError>;
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
pub struct PackageList {
|
||||
pub list: Vec<String>,
|
||||
}
|
||||
|
||||
impl PackageList {
|
||||
pub fn list(&self) -> & [ String ] {
|
||||
&self.list
|
||||
}
|
||||
|
||||
pub fn add<T: IntoPackageList>(self, add: T) -> PackageList {
|
||||
let Self { mut list } = self;
|
||||
|
||||
let extend_with = add.into_package_list();
|
||||
list.extend(extend_with.list);
|
||||
|
||||
PackageList {
|
||||
list
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub trait IntoPackageList {
|
||||
fn into_package_list(self) -> PackageList;
|
||||
}
|
||||
|
||||
impl <T: Into<Vec<String>>> IntoPackageList for T {
|
||||
fn into_package_list(self) -> PackageList {
|
||||
PackageList {
|
||||
list: self.into()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -30,13 +30,7 @@ impl SpecificPackageManager for ArchlinuxPackageManager {
|
|||
Ok(())
|
||||
}
|
||||
|
||||
fn remove(&self, list: PackageList) -> Result<(), PackageError> {
|
||||
let res = Cmd::new("pacman").arg("-R").args(list.list()).run()?;
|
||||
|
||||
if ! res.success {
|
||||
return Err(PackageError::CmdFail(res));
|
||||
}
|
||||
|
||||
Ok(())
|
||||
fn remove(&self, _list: PackageList) -> Result<(), PackageError> {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,6 +1,5 @@
|
|||
use duct::cmd as duct_cmd;
|
||||
|
||||
use std::collections::HashMap;
|
||||
use std::path::PathBuf;
|
||||
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
|
@ -9,7 +8,6 @@ pub struct Cmd {
|
|||
args: Vec<String>,
|
||||
stdin: Option<String>,
|
||||
chdir: Option<PathBuf>,
|
||||
env: HashMap<String, String>,
|
||||
}
|
||||
|
||||
impl Cmd {
|
||||
|
@ -19,12 +17,11 @@ impl Cmd {
|
|||
args: Vec::new(),
|
||||
stdin: None,
|
||||
chdir: None,
|
||||
env: HashMap::new(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn arg<T: AsRef<str>>(self, arg: T) -> Cmd {
|
||||
let Self { program, mut args, stdin, chdir, env, .. } = self;
|
||||
let Self { program, mut args, stdin, chdir, .. } = self;
|
||||
|
||||
args.push(arg.as_ref().to_string());
|
||||
|
||||
|
@ -33,13 +30,12 @@ impl Cmd {
|
|||
args,
|
||||
stdin,
|
||||
chdir,
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn args<T: AsRef<str>>(self, new_args: & [ T ]) -> Cmd {
|
||||
|
||||
let Self { program, mut args, stdin, chdir, env, .. } = self;
|
||||
let Self { program, mut args, stdin, chdir, .. } = self;
|
||||
|
||||
for arg in new_args {
|
||||
args.push(arg.as_ref().to_string());
|
||||
|
@ -50,12 +46,10 @@ impl Cmd {
|
|||
args,
|
||||
stdin,
|
||||
chdir,
|
||||
env,
|
||||
}
|
||||
}
|
||||
} }
|
||||
|
||||
pub fn stdin<T: AsRef<str>>(self, input: T) -> Cmd {
|
||||
let Self { program, args, mut stdin, chdir, env, .. } = self;
|
||||
let Self { program, args, mut stdin, chdir, .. } = self;
|
||||
|
||||
stdin = if let Some(mut s) = stdin {
|
||||
s.push_str(input.as_ref());
|
||||
|
@ -69,47 +63,25 @@ impl Cmd {
|
|||
args,
|
||||
stdin,
|
||||
chdir,
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn chdir<T: AsRef<str>>(self, dir: T) -> Cmd {
|
||||
let Self { program, args, stdin, env, .. } = self;
|
||||
let Self { program, args, stdin, .. } = self;
|
||||
|
||||
Cmd {
|
||||
program,
|
||||
args,
|
||||
stdin,
|
||||
chdir: Some(PathBuf::from(dir.as_ref())),
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn env<K: AsRef<str>, V: AsRef<str>>(self, key: K, value: V) -> Cmd {
|
||||
let Self { program, args, stdin, chdir, mut env, .. } = self;
|
||||
|
||||
env.insert(key.as_ref().to_string(), value.as_ref().to_string());
|
||||
|
||||
Cmd {
|
||||
program,
|
||||
args,
|
||||
stdin,
|
||||
chdir,
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn run(self) -> Result<CmdOutput, std::io::Error> {
|
||||
let Self { program, args, stdin, chdir, env, .. } = self;
|
||||
let Self { program, args, stdin, chdir, .. } = self;
|
||||
|
||||
let mut cmd = duct_cmd(&program, &args);
|
||||
|
||||
if ! env.is_empty() {
|
||||
for (key, value) in env {
|
||||
cmd = cmd.env(key, value);
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(stdin) = &stdin {
|
||||
cmd = cmd.stdin_bytes(stdin.as_bytes());
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue
Block a user