2018-01-02 12:02:09 +00:00
|
|
|
//
|
|
|
|
// imag - the personal information management suite for the commandline
|
2018-02-07 01:48:53 +00:00
|
|
|
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
|
2018-01-02 12:02:09 +00:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; version
|
|
|
|
// 2.1 of the License.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
|
2018-09-27 06:13:58 +00:00
|
|
|
#![forbid(unsafe_code)]
|
|
|
|
|
2018-01-02 12:02:09 +00:00
|
|
|
#![deny(
|
|
|
|
non_camel_case_types,
|
|
|
|
non_snake_case,
|
|
|
|
path_statements,
|
|
|
|
trivial_numeric_casts,
|
|
|
|
unstable_features,
|
|
|
|
unused_allocation,
|
|
|
|
unused_import_braces,
|
|
|
|
unused_imports,
|
|
|
|
unused_must_use,
|
|
|
|
unused_mut,
|
|
|
|
unused_qualifications,
|
|
|
|
while_true,
|
|
|
|
)]
|
|
|
|
|
|
|
|
extern crate clap;
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
extern crate toml;
|
|
|
|
|
2018-02-20 16:45:14 +00:00
|
|
|
#[macro_use] extern crate libimagrt;
|
2018-02-14 22:38:18 +00:00
|
|
|
extern crate libimagerror;
|
|
|
|
|
2018-01-02 12:02:09 +00:00
|
|
|
mod ui;
|
|
|
|
|
|
|
|
use std::fs::OpenOptions;
|
|
|
|
use std::io::Write;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::path::Path;
|
|
|
|
use std::process::Command;
|
|
|
|
|
2018-02-14 22:38:18 +00:00
|
|
|
use libimagerror::exit::ExitUnwrap;
|
|
|
|
use libimagerror::io::ToExitCode;
|
2018-04-14 09:54:56 +00:00
|
|
|
use libimagrt::runtime::Runtime;
|
2018-02-14 22:38:18 +00:00
|
|
|
|
2018-02-10 14:44:50 +00:00
|
|
|
const CONFIGURATION_STR : &'static str = include_str!("../imagrc.toml");
|
|
|
|
|
2018-01-02 12:02:09 +00:00
|
|
|
const GITIGNORE_STR : &'static str = r#"
|
|
|
|
# We ignore the imagrc.toml file by default
|
|
|
|
#
|
|
|
|
# That is because we expect the user to put
|
|
|
|
# this dotfile into his dotfile repository
|
|
|
|
# and symlink it here.
|
|
|
|
# If you do not do this, feel free to remove
|
|
|
|
# this line from the gitignore and add the
|
|
|
|
# configuration to this git repository.
|
|
|
|
|
|
|
|
imagrc.toml
|
|
|
|
"#;
|
|
|
|
|
|
|
|
fn main() {
|
2018-02-20 16:45:14 +00:00
|
|
|
let version = make_imag_version!();
|
2018-04-14 09:54:56 +00:00
|
|
|
let app = ui::build_ui(Runtime::get_default_cli_builder(
|
|
|
|
"imag-init",
|
|
|
|
version.as_str(),
|
|
|
|
"Intializes the imag store, optionally with git"));
|
2018-01-02 12:02:09 +00:00
|
|
|
let matches = app.get_matches();
|
2018-02-14 22:38:18 +00:00
|
|
|
let mut out = ::std::io::stdout();
|
2018-01-02 12:02:09 +00:00
|
|
|
|
|
|
|
let path = matches
|
|
|
|
.value_of("path")
|
|
|
|
.map(String::from)
|
|
|
|
.map(PathBuf::from)
|
|
|
|
.unwrap_or_else(|| {
|
|
|
|
::std::env::var("HOME")
|
|
|
|
.map(PathBuf::from)
|
|
|
|
.map(|mut p| { p.push(".imag"); p })
|
|
|
|
.map(|path| if path.exists() {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "Path '{:?}' already exists!", path)
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
|
|
|
let _ = writeln!(out, "Cannot continue.")
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-02 12:02:09 +00:00
|
|
|
::std::process::exit(1)
|
|
|
|
} else {
|
|
|
|
path
|
|
|
|
})
|
|
|
|
.expect("Failed to retrieve/build path for imag directory.")
|
|
|
|
});
|
|
|
|
|
2018-03-23 09:12:39 +00:00
|
|
|
{
|
|
|
|
let mut store_path = path.clone();
|
|
|
|
store_path.push("store");
|
|
|
|
println!("Creating {}", store_path.display());
|
|
|
|
|
|
|
|
let _ = ::std::fs::create_dir_all(store_path)
|
|
|
|
.expect("Failed to create directory");
|
|
|
|
}
|
2018-01-02 12:02:09 +00:00
|
|
|
|
|
|
|
let config_path = {
|
|
|
|
let mut config_path = path.clone();
|
|
|
|
config_path.push("imagrc.toml");
|
|
|
|
config_path
|
|
|
|
};
|
|
|
|
|
|
|
|
let _ = OpenOptions::new()
|
|
|
|
.write(true)
|
|
|
|
.create(true)
|
|
|
|
.open(config_path)
|
|
|
|
.map(|mut f| {
|
|
|
|
let content = if matches.is_present("devel") {
|
|
|
|
get_config_devel()
|
|
|
|
} else {
|
|
|
|
get_config()
|
|
|
|
};
|
|
|
|
|
|
|
|
let _ = f.write_all(content.as_bytes())
|
|
|
|
.expect("Failed to write complete config to file");
|
|
|
|
})
|
|
|
|
.expect("Failed to open new configuration file");
|
|
|
|
|
|
|
|
if find_command("git").is_some() && !matches.is_present("nogit") {
|
|
|
|
// we initialize a git repository
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "Going to initialize a git repository in the imag directory...")
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-02 12:02:09 +00:00
|
|
|
|
|
|
|
let gitignore_path = {
|
|
|
|
let mut gitignore_path = path.clone();
|
|
|
|
gitignore_path.push(".gitignore");
|
|
|
|
gitignore_path.to_str().map(String::from).expect("Cannot convert path to string")
|
|
|
|
};
|
|
|
|
|
|
|
|
let _ = OpenOptions::new()
|
|
|
|
.write(true)
|
|
|
|
.create(true)
|
|
|
|
.open(gitignore_path.clone())
|
|
|
|
.map(|mut f| {
|
|
|
|
let _ = f.write_all(GITIGNORE_STR.as_bytes())
|
|
|
|
.expect("Failed to write complete gitignore to file");
|
|
|
|
})
|
|
|
|
.expect("Failed to open new configuration file");
|
|
|
|
|
|
|
|
let path_str = path.to_str().map(String::from).expect("Cannot convert path to string");
|
|
|
|
let worktree = format!("--work-tree={}", path_str);
|
2018-02-10 17:18:39 +00:00
|
|
|
let gitdir = format!("--git-dir={}/.git", path_str);
|
2018-01-02 12:02:09 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
let output = Command::new("git")
|
|
|
|
.args(&[&worktree, &gitdir, "--no-pager", "init"])
|
|
|
|
.output()
|
|
|
|
.expect("Calling 'git init' failed");
|
|
|
|
|
2018-01-04 11:10:07 +00:00
|
|
|
if output.status.success() {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stdout).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
|
|
|
let _ = writeln!(out, "'git {} {} --no-pager init' succeeded", worktree, gitdir)
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
} else {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stderr).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
::std::process::exit(output.status.code().unwrap_or(1));
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let output = Command::new("git")
|
|
|
|
.args(&[&worktree, &gitdir, "--no-pager", "add", &gitignore_path])
|
|
|
|
.output()
|
|
|
|
.expect("Calling 'git add' failed");
|
2018-01-04 11:10:07 +00:00
|
|
|
if output.status.success() {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stdout).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
|
|
|
let _ = writeln!(out, "'git {} {} --no-pager add {}' succeeded", worktree, gitdir, gitignore_path)
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
} else {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stderr).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
::std::process::exit(output.status.code().unwrap_or(1));
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let output = Command::new("git")
|
|
|
|
.args(&[&worktree, &gitdir, "--no-pager", "commit", &gitignore_path, "-m", "'Initial import'"])
|
|
|
|
.output()
|
|
|
|
.expect("Calling 'git commit' failed");
|
2018-01-04 11:10:07 +00:00
|
|
|
if output.status.success() {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stdout).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
|
|
|
let _ = writeln!(out, "'git {} {} --no-pager commit {} -m 'Initial import'' succeeded", worktree, gitdir, gitignore_path)
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
} else {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "{}", String::from_utf8(output.stderr).expect("No UTF-8 output"))
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-04 11:10:07 +00:00
|
|
|
::std::process::exit(output.status.code().unwrap_or(1));
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "git stuff finished!")
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-02 12:02:09 +00:00
|
|
|
} else {
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "No git repository will be initialized")
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
|
2018-02-14 22:38:18 +00:00
|
|
|
let _ = writeln!(out, "Ready. Have fun with imag!")
|
|
|
|
.to_exit_code()
|
|
|
|
.unwrap_or_exit();
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn get_config() -> String {
|
|
|
|
get_config_devel()
|
|
|
|
.replace(
|
|
|
|
r#"level = "debug""#,
|
|
|
|
r#"level = "info""#
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_config_devel() -> String {
|
|
|
|
String::from(CONFIGURATION_STR)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn find_command<P: AsRef<Path>>(exe_name: P) -> Option<PathBuf> {
|
|
|
|
::std::env::var_os("PATH")
|
|
|
|
.and_then(|paths| {
|
|
|
|
::std::env::split_paths(&paths)
|
|
|
|
.filter_map(|dir| {
|
|
|
|
let full_path = dir.join(&exe_name);
|
|
|
|
if full_path.is_file() {
|
|
|
|
Some(full_path)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.next()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use toml::from_str;
|
|
|
|
use toml::Value;
|
|
|
|
use super::get_config;
|
|
|
|
use super::get_config_devel;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_config() {
|
2018-07-19 18:18:04 +00:00
|
|
|
let val = from_str::<Value>(&get_config()[..]);
|
|
|
|
assert!(val.is_ok(), "Config parsing errored: {:?}", val);
|
|
|
|
|
|
|
|
let val = from_str::<Value>(&get_config_devel()[..]);
|
|
|
|
assert!(val.is_ok(), "Config parsing errored: {:?}", val);
|
2018-01-02 12:02:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|