Added a cli and arg parsing
All checks were successful
Gitea Organization/dotfiles/pipeline/head This commit looks good
All checks were successful
Gitea Organization/dotfiles/pipeline/head This commit looks good
This commit is contained in:
parent
20f340e14f
commit
0fc89e6e40
@ -2,5 +2,6 @@
|
|||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="VcsDirectoryMappings">
|
<component name="VcsDirectoryMappings">
|
||||||
<mapping directory="" vcs="Git" />
|
<mapping directory="" vcs="Git" />
|
||||||
|
<mapping directory="$PROJECT_DIR$/repo" vcs="Git" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
7
Cargo.lock
generated
7
Cargo.lock
generated
@ -35,6 +35,12 @@ version = "1.3.2"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bpaf"
|
||||||
|
version = "0.7.8"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "863c0b21775e45ebf9bbb3a6d0cd9b3421c88a036e825359e3d4015561f3e23c"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bumpalo"
|
name = "bumpalo"
|
||||||
version = "3.12.0"
|
version = "3.12.0"
|
||||||
@ -89,6 +95,7 @@ name = "dotfiles_installer"
|
|||||||
version = "1.0.0"
|
version = "1.0.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"attohttpc",
|
"attohttpc",
|
||||||
|
"bpaf",
|
||||||
"dialoguer",
|
"dialoguer",
|
||||||
"indicatif",
|
"indicatif",
|
||||||
"serde",
|
"serde",
|
||||||
|
@ -6,9 +6,11 @@ edition = "2021"
|
|||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
toml = "0.7.2"
|
||||||
|
bpaf = "0.7.8"
|
||||||
|
|
||||||
dialoguer = "0.10.3"
|
dialoguer = "0.10.3"
|
||||||
indicatif = "0.17.3"
|
indicatif = "0.17.3"
|
||||||
toml = "0.7.2"
|
|
||||||
|
|
||||||
[dependencies.serde]
|
[dependencies.serde]
|
||||||
version = "1.0.152"
|
version = "1.0.152"
|
||||||
|
68
src/cli/mod.rs
Normal file
68
src/cli/mod.rs
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
use bpaf::{command, construct, Parser};
|
||||||
|
use crate::common::config::ModToml;
|
||||||
|
use crate::common::{operations, repository};
|
||||||
|
use crate::common::operations::NamedModule;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum CliCommand {
|
||||||
|
Install(Vec<String>),
|
||||||
|
Collect(Vec<String>),
|
||||||
|
Upload(String),
|
||||||
|
List(),
|
||||||
|
Pull(),
|
||||||
|
Diff()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_parser() -> impl Parser<CliCommand> {
|
||||||
|
let modules = bpaf::positional::<String>("Modules").many();
|
||||||
|
let install =
|
||||||
|
command("install", construct!(CliCommand::Install(modules)).to_options())
|
||||||
|
.help("Install modules");
|
||||||
|
|
||||||
|
let modules = bpaf::positional::<String>("Modules").many();
|
||||||
|
let collect =
|
||||||
|
command("collect", construct!(CliCommand::Collect(modules)).to_options())
|
||||||
|
.help("Collect modules");
|
||||||
|
|
||||||
|
let commit_msg = bpaf::positional::<String>("Commit message");
|
||||||
|
let upload =
|
||||||
|
command("upload", construct!(CliCommand::Upload(commit_msg)).to_options())
|
||||||
|
.help("Upload files");
|
||||||
|
|
||||||
|
let list = command("list", construct!(CliCommand::List()).to_options())
|
||||||
|
.help("List modules");
|
||||||
|
|
||||||
|
let pull = command("pull", construct!(CliCommand::Pull()).to_options())
|
||||||
|
.help("Downloads update");
|
||||||
|
|
||||||
|
let diff = command("diff", construct!(CliCommand::Diff()).to_options())
|
||||||
|
.help("Show diff of current changes");
|
||||||
|
|
||||||
|
construct!([
|
||||||
|
install,
|
||||||
|
collect,
|
||||||
|
upload,
|
||||||
|
list,
|
||||||
|
pull,
|
||||||
|
diff
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn run(config: &ModToml, command: CliCommand) {
|
||||||
|
|
||||||
|
let map_mod = |name: String| -> NamedModule {
|
||||||
|
match config.modules.get(&name) {
|
||||||
|
None => { println!("Module {name} doesn't exist"); std::process::exit(1); }
|
||||||
|
Some(v) => NamedModule(name, v.clone())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
match command {
|
||||||
|
CliCommand::Install(mods) => operations::install(&mods.into_iter().map(map_mod).collect::<Vec<NamedModule>>()),
|
||||||
|
CliCommand::Collect(mods) => operations::collect(&mods.into_iter().map(map_mod).collect::<Vec<NamedModule>>()),
|
||||||
|
CliCommand::Upload(msg) => operations::upload(|| msg),
|
||||||
|
CliCommand::List() => config.modules.iter().for_each(|m| println!("Module {}", m.0)),
|
||||||
|
CliCommand::Pull() => repository::pull_repo(),
|
||||||
|
CliCommand::Diff() => repository::diff()
|
||||||
|
}
|
||||||
|
}
|
5
src/common/mod.rs
Normal file
5
src/common/mod.rs
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
pub mod repository;
|
||||||
|
pub mod config;
|
||||||
|
pub mod operations;
|
||||||
|
pub mod utils;
|
||||||
|
pub mod update;
|
@ -1,8 +1,16 @@
|
|||||||
use crate::config::{ModToml, Module};
|
use crate::common::{repository, utils, config::{ModToml, Module}};
|
||||||
use crate::prompt::multi_select;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct NamedModule(String, Module);
|
pub struct NamedModule(pub String, pub Module);
|
||||||
|
|
||||||
|
impl From<&ModToml> for Vec<NamedModule> {
|
||||||
|
fn from(config: &ModToml) -> Self {
|
||||||
|
config.modules.iter()
|
||||||
|
.map(|v| NamedModule(v.0.clone(), v.1.clone()))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ToString for NamedModule {
|
impl ToString for NamedModule {
|
||||||
fn to_string(&self) -> String {
|
fn to_string(&self) -> String {
|
||||||
self.0.clone()
|
self.0.clone()
|
||||||
@ -28,12 +36,12 @@ fn install_mod(m: &NamedModule) {
|
|||||||
let pb = indicatif::ProgressBar::new(0);
|
let pb = indicatif::ProgressBar::new(0);
|
||||||
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>6}/{len:6}").unwrap().progress_chars("#>-"));
|
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>6}/{len:6}").unwrap().progress_chars("#>-"));
|
||||||
|
|
||||||
if let Err(err) = crate::utils::remove_dir(dst, &m.1.ignore, &pb) {
|
if let Err(err) = utils::remove_dir(dst, &m.1.ignore, &pb) {
|
||||||
println!("Failed to delete directory:\n{}", err);
|
println!("Failed to delete directory:\n{}", err);
|
||||||
continue 'content_iter;
|
continue 'content_iter;
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Err(err) = crate::utils::copy_dir(&src, dst, &[], &pb) {
|
if let Err(err) = utils::copy_dir(&src, dst, &[], &pb) {
|
||||||
println!("Failed to copy directory:\n{}", err);
|
println!("Failed to copy directory:\n{}", err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -53,7 +61,7 @@ fn collect_mod(m: &NamedModule) {
|
|||||||
let pb = indicatif::ProgressBar::new(0);
|
let pb = indicatif::ProgressBar::new(0);
|
||||||
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>6}/{len:6}").unwrap().progress_chars("#>-"));
|
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>6}/{len:6}").unwrap().progress_chars("#>-"));
|
||||||
|
|
||||||
if let Err(err) = crate::utils::remove_dir(&dst, &[], &pb) {
|
if let Err(err) = utils::remove_dir(&dst, &[], &pb) {
|
||||||
println!("Failed to delete directory:\n{}", err);
|
println!("Failed to delete directory:\n{}", err);
|
||||||
continue 'content_iter;
|
continue 'content_iter;
|
||||||
}
|
}
|
||||||
@ -63,7 +71,7 @@ fn collect_mod(m: &NamedModule) {
|
|||||||
continue 'content_iter;
|
continue 'content_iter;
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Err(err) = crate::utils::copy_dir(src, &dst, &m.1.ignore, &pb) {
|
if let Err(err) = utils::copy_dir(src, &dst, &m.1.ignore, &pb) {
|
||||||
println!("Failed to copy source content:\n{}", err);
|
println!("Failed to copy source content:\n{}", err);
|
||||||
continue 'content_iter;
|
continue 'content_iter;
|
||||||
}
|
}
|
||||||
@ -71,14 +79,9 @@ fn collect_mod(m: &NamedModule) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn install(config: &ModToml) {
|
pub fn install(mods: &[NamedModule]) {
|
||||||
let to_install = multi_select(
|
for m in mods {
|
||||||
Some("Which modules do you want to install?"),
|
install_mod(m);
|
||||||
config.modules.iter().map(|v| NamedModule(v.0.clone(), v.1.clone())).collect()
|
|
||||||
);
|
|
||||||
|
|
||||||
for m in to_install {
|
|
||||||
install_mod(&m);
|
|
||||||
|
|
||||||
if let Some(on_install) = &m.1.on_install {
|
if let Some(on_install) = &m.1.on_install {
|
||||||
match std::process::Command::new("sh").arg("-c").arg(on_install).status() {
|
match std::process::Command::new("sh").arg("-c").arg(on_install).status() {
|
||||||
@ -89,26 +92,21 @@ pub fn install(config: &ModToml) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn collect(config: &ModToml) {
|
pub fn collect(mods: &[NamedModule]) {
|
||||||
let to_collect = multi_select(
|
for m in mods {
|
||||||
Some("Which modules do you want to collect?"),
|
collect_mod(m);
|
||||||
config.modules.iter().map(|v| NamedModule(v.0.clone(), v.1.clone())).collect()
|
|
||||||
);
|
|
||||||
|
|
||||||
for m in to_collect {
|
|
||||||
collect_mod(&m);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn upload() {
|
pub fn upload<F: FnOnce() -> String>(get_commit_msg: F) {
|
||||||
if !crate::repository::is_clean() {
|
if !repository::is_clean() {
|
||||||
let msg = crate::prompt::input("Commit message");
|
let msg = get_commit_msg();
|
||||||
if !crate::repository::commit_changes(&msg) {
|
if !repository::commit_changes(&msg) {
|
||||||
println!("Failed to commit");
|
println!("Failed to commit");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !crate::repository::push_repo() {
|
if !repository::push_repo() {
|
||||||
println!("Failed to push to origin");
|
println!("Failed to push to origin");
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -29,7 +29,7 @@ pub fn check_for_updates(version: u64, name: &str) {
|
|||||||
|
|
||||||
fs::rename(&temp, &exe).unwrap();
|
fs::rename(&temp, &exe).unwrap();
|
||||||
|
|
||||||
std::process::Command::new(exe).exec();
|
std::process::Command::new(exe).args(std::env::args()).exec();
|
||||||
std::process::exit(1);
|
std::process::exit(1);
|
||||||
}
|
}
|
||||||
println!("No new version");
|
println!("No new version");
|
74
src/main.rs
74
src/main.rs
@ -1,34 +1,33 @@
|
|||||||
mod prompt;
|
mod common;
|
||||||
mod repository;
|
mod tui;
|
||||||
mod config;
|
mod cli;
|
||||||
mod operations;
|
|
||||||
mod utils;
|
use bpaf::{construct, Parser};
|
||||||
mod update;
|
use crate::common::{config, repository, update};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
enum MainMenu {
|
enum Commands {
|
||||||
Install,
|
Tui(),
|
||||||
Collect,
|
Cli(cli::CliCommand)
|
||||||
Pull,
|
|
||||||
Diff,
|
|
||||||
Upload,
|
|
||||||
Quit
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ToString for MainMenu {
|
|
||||||
fn to_string(&self) -> String {
|
|
||||||
match self {
|
|
||||||
MainMenu::Install => "Install files",
|
|
||||||
MainMenu::Collect => "Collect files",
|
|
||||||
MainMenu::Pull => "Pull from git",
|
|
||||||
MainMenu::Diff => "View git diff",
|
|
||||||
MainMenu::Upload => "(Commit) and push to git",
|
|
||||||
MainMenu::Quit => "Exit"
|
|
||||||
}.to_string()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
let tui_cmd =
|
||||||
|
bpaf::command("tui", construct!(Commands::Tui()).to_options())
|
||||||
|
.help("Terminal ui for interactive use");
|
||||||
|
|
||||||
|
let cli_parser = cli::get_parser();
|
||||||
|
|
||||||
|
let cli_cmd =
|
||||||
|
bpaf::command("cli", construct!(Commands::Cli(cli_parser)).to_options())
|
||||||
|
.help("Command line for script usage");
|
||||||
|
|
||||||
|
let opt = construct!([tui_cmd, cli_cmd])
|
||||||
|
.fallback(Commands::Tui())
|
||||||
|
.to_options()
|
||||||
|
.descr("Dotfiles multitool")
|
||||||
|
.run();
|
||||||
|
|
||||||
if let (Some(version), Some(name)) = (option_env!("BUILD_ID"), option_env!("JOB_BASE_NAME")) {
|
if let (Some(version), Some(name)) = (option_env!("BUILD_ID"), option_env!("JOB_BASE_NAME")) {
|
||||||
println!("Starting installer version {name}-{version}");
|
println!("Starting installer version {name}-{version}");
|
||||||
update::check_for_updates(version.parse().unwrap(), name);
|
update::check_for_updates(version.parse().unwrap(), name);
|
||||||
@ -55,25 +54,8 @@ fn main() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
'main_loop: loop {
|
match opt {
|
||||||
let res = prompt::select(
|
Commands::Tui() => tui::run(&mods),
|
||||||
Some("What do you want to do?"),
|
Commands::Cli(command) => cli::run(&mods, command)
|
||||||
vec![
|
|
||||||
MainMenu::Install,
|
|
||||||
MainMenu::Collect,
|
|
||||||
MainMenu::Pull,
|
|
||||||
MainMenu::Diff,
|
|
||||||
MainMenu::Upload,
|
|
||||||
MainMenu::Quit
|
|
||||||
]
|
|
||||||
);
|
|
||||||
match res {
|
|
||||||
MainMenu::Install => operations::install(&mods),
|
|
||||||
MainMenu::Collect => operations::collect(&mods),
|
|
||||||
MainMenu::Pull => repository::pull_repo(),
|
|
||||||
MainMenu::Diff => repository::diff(),
|
|
||||||
MainMenu::Upload => operations::upload(),
|
|
||||||
MainMenu::Quit => break 'main_loop
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,38 +0,0 @@
|
|||||||
use dialoguer::{Input, MultiSelect, Select};
|
|
||||||
|
|
||||||
pub fn select<F>(title: Option<&str>, mut options: Vec<F>) -> F
|
|
||||||
where F: ToString
|
|
||||||
{
|
|
||||||
let mut sel = Select::new();
|
|
||||||
sel.items(&options);
|
|
||||||
sel.default(0);
|
|
||||||
if let Some(title) = title {
|
|
||||||
sel.with_prompt(title);
|
|
||||||
}
|
|
||||||
let sel = sel.interact().expect("User didn't enter anything");
|
|
||||||
|
|
||||||
options.remove(sel)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn multi_select<F>(title: Option<&str>, options: Vec<F>) -> Vec<F>
|
|
||||||
where F: ToString
|
|
||||||
{
|
|
||||||
let mut sel = MultiSelect::new();
|
|
||||||
sel.items(&options);
|
|
||||||
if let Some(title) = title {
|
|
||||||
sel.with_prompt(title);
|
|
||||||
}
|
|
||||||
let sel = sel.interact().expect("User didn't enter anything");
|
|
||||||
|
|
||||||
options.into_iter()
|
|
||||||
.enumerate()
|
|
||||||
.filter_map(|(i, v)| sel.contains(&i).then_some(v))
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn input(title: &str) -> String {
|
|
||||||
Input::new()
|
|
||||||
.with_prompt(title)
|
|
||||||
.interact_text()
|
|
||||||
.expect("User didn't enter anything")
|
|
||||||
}
|
|
95
src/tui/mod.rs
Normal file
95
src/tui/mod.rs
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
use dialoguer::{Input, MultiSelect, Select};
|
||||||
|
use crate::common::{operations, repository};
|
||||||
|
use crate::common::config::ModToml;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
enum MainMenu {
|
||||||
|
Install,
|
||||||
|
Collect,
|
||||||
|
Pull,
|
||||||
|
Diff,
|
||||||
|
Upload,
|
||||||
|
Quit
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToString for MainMenu {
|
||||||
|
fn to_string(&self) -> String {
|
||||||
|
match self {
|
||||||
|
MainMenu::Install => "Install files",
|
||||||
|
MainMenu::Collect => "Collect files",
|
||||||
|
MainMenu::Pull => "Pull from git",
|
||||||
|
MainMenu::Diff => "View git diff",
|
||||||
|
MainMenu::Upload => "(Commit) and push to git",
|
||||||
|
MainMenu::Quit => "Exit"
|
||||||
|
}.to_string()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn select<F>(title: Option<&str>, mut options: Vec<F>) -> F
|
||||||
|
where F: ToString
|
||||||
|
{
|
||||||
|
let mut sel = Select::new();
|
||||||
|
sel.items(&options);
|
||||||
|
sel.default(0);
|
||||||
|
if let Some(title) = title {
|
||||||
|
sel.with_prompt(title);
|
||||||
|
}
|
||||||
|
let sel = sel.interact().expect("User didn't enter anything");
|
||||||
|
|
||||||
|
options.remove(sel)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn multi_select<F>(title: Option<&str>, options: Vec<F>) -> Vec<F>
|
||||||
|
where F: ToString
|
||||||
|
{
|
||||||
|
let mut sel = MultiSelect::new();
|
||||||
|
sel.items(&options);
|
||||||
|
if let Some(title) = title {
|
||||||
|
sel.with_prompt(title);
|
||||||
|
}
|
||||||
|
let sel = sel.interact().expect("User didn't enter anything");
|
||||||
|
|
||||||
|
options.into_iter()
|
||||||
|
.enumerate()
|
||||||
|
.filter_map(|(i, v)| sel.contains(&i).then_some(v))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn input(title: &str) -> String {
|
||||||
|
Input::new()
|
||||||
|
.with_prompt(title)
|
||||||
|
.interact_text()
|
||||||
|
.expect("User didn't enter anything")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn run(config: &ModToml) {
|
||||||
|
'main_loop: loop {
|
||||||
|
let res = select(
|
||||||
|
Some("What do you want to do?"),
|
||||||
|
vec![
|
||||||
|
MainMenu::Install,
|
||||||
|
MainMenu::Collect,
|
||||||
|
MainMenu::Pull,
|
||||||
|
MainMenu::Diff,
|
||||||
|
MainMenu::Upload,
|
||||||
|
MainMenu::Quit
|
||||||
|
]
|
||||||
|
);
|
||||||
|
match res {
|
||||||
|
MainMenu::Install =>
|
||||||
|
operations::install(&multi_select(
|
||||||
|
Some("Which modules do you want to install?"),
|
||||||
|
config.into()
|
||||||
|
)),
|
||||||
|
MainMenu::Collect =>
|
||||||
|
operations::collect(&multi_select(
|
||||||
|
Some("Which modules do you want to collect?"),
|
||||||
|
config.into()
|
||||||
|
)),
|
||||||
|
MainMenu::Pull => repository::pull_repo(),
|
||||||
|
MainMenu::Diff => repository::diff(),
|
||||||
|
MainMenu::Upload => operations::upload(|| input("Commit message")),
|
||||||
|
MainMenu::Quit => break 'main_loop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user