Merge pull request #1481 from matthiasbeyer/imag-category/init
imag-category: Initial import
This commit is contained in:
commit
16dafb17f4
9 changed files with 443 additions and 0 deletions
|
@ -2,6 +2,7 @@
|
||||||
members = [
|
members = [
|
||||||
"bin/core/imag",
|
"bin/core/imag",
|
||||||
"bin/core/imag-annotate",
|
"bin/core/imag-annotate",
|
||||||
|
"bin/core/imag-category",
|
||||||
"bin/core/imag-diagnostics",
|
"bin/core/imag-diagnostics",
|
||||||
"bin/core/imag-edit",
|
"bin/core/imag-edit",
|
||||||
"bin/core/imag-git",
|
"bin/core/imag-git",
|
||||||
|
|
39
bin/core/imag-category/Cargo.toml
Normal file
39
bin/core/imag-category/Cargo.toml
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
[package]
|
||||||
|
name = "imag-category"
|
||||||
|
version = "0.8.0"
|
||||||
|
authors = ["Matthias Beyer <mail@beyermatthias.de>"]
|
||||||
|
|
||||||
|
description = "Part of the imag core distribution: imag-category command"
|
||||||
|
|
||||||
|
keywords = ["imag", "PIM", "personal", "information", "management"]
|
||||||
|
readme = "../../../README.md"
|
||||||
|
license = "LGPL-2.1"
|
||||||
|
|
||||||
|
documentation = "https://imag-pim.org/doc/"
|
||||||
|
repository = "https://github.com/matthiasbeyer/imag"
|
||||||
|
homepage = "http://imag-pim.org"
|
||||||
|
|
||||||
|
build = "../../../build.rs"
|
||||||
|
|
||||||
|
[badges]
|
||||||
|
travis-ci = { repository = "matthiasbeyer/imag" }
|
||||||
|
is-it-maintained-issue-resolution = { repository = "matthiasbeyer/imag" }
|
||||||
|
is-it-maintained-open-issues = { repository = "matthiasbeyer/imag" }
|
||||||
|
maintenance = { status = "actively-developed" }
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
log = "0.4.0"
|
||||||
|
toml = "0.4"
|
||||||
|
toml-query = "0.6"
|
||||||
|
|
||||||
|
libimagstore = { version = "0.8.0", path = "../../../lib/core/libimagstore" }
|
||||||
|
libimagrt = { version = "0.8.0", path = "../../../lib/core/libimagrt" }
|
||||||
|
libimagerror = { version = "0.8.0", path = "../../../lib/core/libimagerror" }
|
||||||
|
libimagentrycategory = { version = "0.8.0", path = "../../../lib/entry/libimagentrycategory" }
|
||||||
|
libimaginteraction = { version = "0.8.0", path = "../../../lib/etc/libimaginteraction" }
|
||||||
|
|
||||||
|
[dependencies.clap]
|
||||||
|
version = "^2.29"
|
||||||
|
default-features = false
|
||||||
|
features = ["color", "suggestions", "wrap_help"]
|
||||||
|
|
1
bin/core/imag-category/README.md
Symbolic link
1
bin/core/imag-category/README.md
Symbolic link
|
@ -0,0 +1 @@
|
||||||
|
../../../doc/src/04020-module-category.md
|
240
bin/core/imag-category/src/main.rs
Normal file
240
bin/core/imag-category/src/main.rs
Normal file
|
@ -0,0 +1,240 @@
|
||||||
|
//
|
||||||
|
// imag - the personal information management suite for the commandline
|
||||||
|
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
|
||||||
|
//
|
||||||
|
// 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
|
||||||
|
//
|
||||||
|
|
||||||
|
#![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;
|
||||||
|
#[macro_use]
|
||||||
|
extern crate log;
|
||||||
|
|
||||||
|
extern crate libimagentrycategory;
|
||||||
|
extern crate libimagerror;
|
||||||
|
#[macro_use] extern crate libimagrt;
|
||||||
|
extern crate libimagstore;
|
||||||
|
extern crate libimaginteraction;
|
||||||
|
|
||||||
|
use libimagerror::trace::MapErrTrace;
|
||||||
|
use libimagerror::exit::ExitUnwrap;
|
||||||
|
use libimagerror::io::ToExitCode;
|
||||||
|
use libimagrt::runtime::Runtime;
|
||||||
|
use libimagrt::setup::generate_runtime_setup;
|
||||||
|
use libimagstore::storeid::IntoStoreId;
|
||||||
|
|
||||||
|
mod ui;
|
||||||
|
|
||||||
|
use std::io::Write;
|
||||||
|
use std::io::Read;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
use libimagentrycategory::store::CategoryStore;
|
||||||
|
use libimagstore::storeid::StoreIdIterator;
|
||||||
|
use libimagstore::iter::get::StoreIdGetIteratorExtension;
|
||||||
|
use libimagerror::iter::TraceIterator;
|
||||||
|
use libimagentrycategory::entry::EntryCategory;
|
||||||
|
use libimagentrycategory::category::Category;
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let version = make_imag_version!();
|
||||||
|
let rt = generate_runtime_setup("imag-category",
|
||||||
|
&version,
|
||||||
|
"Add a category to entries and manage categories",
|
||||||
|
ui::build_ui);
|
||||||
|
|
||||||
|
rt.cli()
|
||||||
|
.subcommand_name()
|
||||||
|
.map(|name| {
|
||||||
|
match name {
|
||||||
|
"set" => set(&rt),
|
||||||
|
"get" => get(&rt),
|
||||||
|
"list-category" => list_category(&rt),
|
||||||
|
"create-category" => create_category(&rt),
|
||||||
|
"delete-category" => delete_category(&rt),
|
||||||
|
"list-categories" => list_categories(&rt),
|
||||||
|
other => {
|
||||||
|
debug!("Unknown command");
|
||||||
|
let _ = rt.handle_unknown_subcommand("imag-category", other, rt.cli())
|
||||||
|
.map_err_trace_exit_unwrap(1)
|
||||||
|
.code()
|
||||||
|
.map(::std::process::exit);
|
||||||
|
},
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set(rt: &Runtime) {
|
||||||
|
let scmd = rt.cli().subcommand_matches("set").unwrap(); // safed by main()
|
||||||
|
let name = scmd.value_of("set-name").map(String::from).unwrap(); // safed by clap
|
||||||
|
let sids = match scmd.value_of("set-ids") {
|
||||||
|
Some(path) => vec![PathBuf::from(path).into_storeid().map_err_trace_exit_unwrap(1)],
|
||||||
|
None => if rt.cli().is_present("entries-from-stdin") {
|
||||||
|
let stdin = rt.stdin().unwrap_or_else(|| {
|
||||||
|
error!("Cannot get handle to stdin");
|
||||||
|
::std::process::exit(1)
|
||||||
|
});
|
||||||
|
|
||||||
|
let mut buf = String::new();
|
||||||
|
let _ = stdin.lock().read_to_string(&mut buf).unwrap_or_else(|_| {
|
||||||
|
error!("Failed to read from stdin");
|
||||||
|
::std::process::exit(1)
|
||||||
|
});
|
||||||
|
|
||||||
|
buf.lines()
|
||||||
|
.map(PathBuf::from)
|
||||||
|
.map(|p| p.into_storeid().map_err_trace_exit_unwrap(1))
|
||||||
|
.collect()
|
||||||
|
} else {
|
||||||
|
error!("Something weird happened. I was not able to find the path of the entries to edit");
|
||||||
|
::std::process::exit(1)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
StoreIdIterator::new(Box::new(sids.into_iter().map(Ok)))
|
||||||
|
.into_get_iter(rt.store())
|
||||||
|
.trace_unwrap_exit(1)
|
||||||
|
.map(|o| o.unwrap_or_else(|| {
|
||||||
|
error!("Did not find one entry");
|
||||||
|
::std::process::exit(1)
|
||||||
|
}))
|
||||||
|
.for_each(|mut entry| {
|
||||||
|
let _ = entry
|
||||||
|
.set_category_checked(rt.store(), &name)
|
||||||
|
.map_err_trace_exit_unwrap(1);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get(rt: &Runtime) {
|
||||||
|
let scmd = rt.cli().subcommand_matches("get").unwrap(); // safed by main()
|
||||||
|
let sids = match scmd.value_of("get-ids") {
|
||||||
|
Some(path) => vec![PathBuf::from(path).into_storeid().map_err_trace_exit_unwrap(1)],
|
||||||
|
None => if rt.cli().is_present("entries-from-stdin") {
|
||||||
|
let stdin = rt.stdin().unwrap_or_else(|| {
|
||||||
|
error!("Cannot get handle to stdin");
|
||||||
|
::std::process::exit(1)
|
||||||
|
});
|
||||||
|
|
||||||
|
let mut buf = String::new();
|
||||||
|
let _ = stdin.lock().read_to_string(&mut buf).unwrap_or_else(|_| {
|
||||||
|
error!("Failed to read from stdin");
|
||||||
|
::std::process::exit(1)
|
||||||
|
});
|
||||||
|
|
||||||
|
buf.lines()
|
||||||
|
.map(PathBuf::from)
|
||||||
|
.map(|p| p.into_storeid().map_err_trace_exit_unwrap(1))
|
||||||
|
.collect()
|
||||||
|
} else {
|
||||||
|
error!("Something weird happened. I was not able to find the path of the entries to edit");
|
||||||
|
::std::process::exit(1)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let out = rt.stdout();
|
||||||
|
let mut outlock = out.lock();
|
||||||
|
|
||||||
|
StoreIdIterator::new(Box::new(sids.into_iter().map(Ok)))
|
||||||
|
.into_get_iter(rt.store())
|
||||||
|
.trace_unwrap_exit(1)
|
||||||
|
.map(|o| o.unwrap_or_else(|| {
|
||||||
|
error!("Did not find one entry");
|
||||||
|
::std::process::exit(1)
|
||||||
|
}))
|
||||||
|
.map(|entry| entry.get_category().map_err_trace_exit_unwrap(1))
|
||||||
|
.for_each(|name| {
|
||||||
|
let _ = writeln!(outlock, "{}", name).to_exit_code().unwrap_or_exit();
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn list_category(rt: &Runtime) {
|
||||||
|
let scmd = rt.cli().subcommand_matches("list-category").unwrap(); // safed by main()
|
||||||
|
let name = scmd.value_of("list-category-name").map(String::from).unwrap(); // safed by clap
|
||||||
|
|
||||||
|
if let Some(category) = rt.store().get_category_by_name(&name).map_err_trace_exit_unwrap(1) {
|
||||||
|
let out = rt.stdout();
|
||||||
|
let mut outlock = out.lock();
|
||||||
|
|
||||||
|
category
|
||||||
|
.get_entries(rt.store())
|
||||||
|
.map_err_trace_exit_unwrap(1)
|
||||||
|
.for_each(|entry| {
|
||||||
|
writeln!(outlock, "{}", entry.map_err_trace_exit_unwrap(1).get_location())
|
||||||
|
.to_exit_code()
|
||||||
|
.unwrap_or_exit();
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
info!("No category named '{}'", name);
|
||||||
|
::std::process::exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_category(rt: &Runtime) {
|
||||||
|
let scmd = rt.cli().subcommand_matches("create-category").unwrap(); // safed by main()
|
||||||
|
let name = scmd.value_of("create-category-name").map(String::from).unwrap(); // safed by clap
|
||||||
|
|
||||||
|
let _ = rt
|
||||||
|
.store()
|
||||||
|
.create_category(&name)
|
||||||
|
.map_err_trace_exit_unwrap(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn delete_category(rt: &Runtime) {
|
||||||
|
use libimaginteraction::ask::ask_bool;
|
||||||
|
|
||||||
|
let scmd = rt.cli().subcommand_matches("delete-category").unwrap(); // safed by main()
|
||||||
|
let name = scmd.value_of("delete-category-name").map(String::from).unwrap(); // safed by clap
|
||||||
|
let ques = format!("Do you really want to delete category '{}' and remove links to all categorized enties?", name);
|
||||||
|
let answer = ask_bool(&ques, Some(false));
|
||||||
|
|
||||||
|
if answer {
|
||||||
|
info!("Deleting category '{}'", name);
|
||||||
|
let _ = rt
|
||||||
|
.store()
|
||||||
|
.delete_category(&name)
|
||||||
|
.map_err_trace_exit_unwrap(1);
|
||||||
|
} else {
|
||||||
|
info!("Not doing anything");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn list_categories(rt: &Runtime) {
|
||||||
|
let out = rt.stdout();
|
||||||
|
let mut outlock = out.lock();
|
||||||
|
|
||||||
|
rt.store()
|
||||||
|
.all_category_names()
|
||||||
|
.map_err_trace_exit_unwrap(1)
|
||||||
|
.for_each(|name| {
|
||||||
|
writeln!(outlock, "{}", name.map_err_trace_exit_unwrap(1))
|
||||||
|
.to_exit_code()
|
||||||
|
.unwrap_or_exit();
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
118
bin/core/imag-category/src/ui.rs
Normal file
118
bin/core/imag-category/src/ui.rs
Normal file
|
@ -0,0 +1,118 @@
|
||||||
|
//
|
||||||
|
// imag - the personal information management suite for the commandline
|
||||||
|
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
|
||||||
|
//
|
||||||
|
// 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
|
||||||
|
//
|
||||||
|
|
||||||
|
use clap::{Arg, ArgGroup, App, SubCommand};
|
||||||
|
|
||||||
|
pub fn build_ui<'a>(app: App<'a, 'a>) -> App<'a, 'a> {
|
||||||
|
app
|
||||||
|
.subcommand(SubCommand::with_name("create-category")
|
||||||
|
.about("Create a new category")
|
||||||
|
.version("0.1")
|
||||||
|
.arg(Arg::with_name("create-category-name")
|
||||||
|
.index(1)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(true)
|
||||||
|
.multiple(false)
|
||||||
|
.help("The name of the new category")
|
||||||
|
.value_name("NAME"))
|
||||||
|
)
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("delete-category")
|
||||||
|
.about("Delete a new category")
|
||||||
|
.version("0.1")
|
||||||
|
.arg(Arg::with_name("delete-category-name")
|
||||||
|
.index(1)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(true)
|
||||||
|
.multiple(false)
|
||||||
|
.help("The name of the category to delete")
|
||||||
|
.value_name("NAME"))
|
||||||
|
)
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("list-categories")
|
||||||
|
.about("Show all category names")
|
||||||
|
.version("0.1"))
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("list-category")
|
||||||
|
.about("List all entries for a category")
|
||||||
|
.version("0.1")
|
||||||
|
.arg(Arg::with_name("list-category-name")
|
||||||
|
.index(1)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(true)
|
||||||
|
.multiple(false)
|
||||||
|
.help("The name of the category to list all entries for")
|
||||||
|
.value_name("NAME"))
|
||||||
|
)
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("set")
|
||||||
|
.about("Set the category of entries")
|
||||||
|
.version("0.1")
|
||||||
|
.arg(Arg::with_name("set-name")
|
||||||
|
.index(1)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(true)
|
||||||
|
.multiple(false)
|
||||||
|
.help("The name of the category to list all entries for")
|
||||||
|
.value_name("NAME"))
|
||||||
|
|
||||||
|
.arg(Arg::with_name("set-ids")
|
||||||
|
.index(2)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(false)
|
||||||
|
.multiple(true)
|
||||||
|
.help("The entries to set the category for")
|
||||||
|
.value_name("ID"))
|
||||||
|
.arg(Arg::with_name("entries-from-stdin")
|
||||||
|
.long("ids-from-stdin")
|
||||||
|
.short("I")
|
||||||
|
.takes_value(false)
|
||||||
|
.required(false)
|
||||||
|
.multiple(false)
|
||||||
|
.help("Read the ids for the entries from stdin"))
|
||||||
|
|
||||||
|
.group(ArgGroup::with_name("input-method")
|
||||||
|
.args(&["set-ids", "entries-from-stdin"])
|
||||||
|
.required(true))
|
||||||
|
)
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("get")
|
||||||
|
.about("Get the category of the entry")
|
||||||
|
.version("0.1")
|
||||||
|
.arg(Arg::with_name("get-ids")
|
||||||
|
.index(1)
|
||||||
|
.takes_value(true)
|
||||||
|
.required(false)
|
||||||
|
.multiple(true)
|
||||||
|
.help("The id of the Entry to get the category for")
|
||||||
|
.value_name("ID"))
|
||||||
|
.arg(Arg::with_name("entries-from-stdin")
|
||||||
|
.long("ids-from-stdin")
|
||||||
|
.short("I")
|
||||||
|
.takes_value(false)
|
||||||
|
.required(false)
|
||||||
|
.multiple(false)
|
||||||
|
.help("Read the ids for the entries from stdin"))
|
||||||
|
|
||||||
|
.group(ArgGroup::with_name("input-method")
|
||||||
|
.args(&["get-ids", "entries-from-stdin"])
|
||||||
|
.required(true))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
8
doc/src/04020-module-category.md
Normal file
8
doc/src/04020-module-category.md
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
## Category {#sec:modules:category}
|
||||||
|
|
||||||
|
A tool to create categories and set/get them for entries.
|
||||||
|
|
||||||
|
The difference between a category and a tag is that a category must exist
|
||||||
|
before it can be used and all entries of a category are linked to the
|
||||||
|
"category entry" internally.
|
||||||
|
|
|
@ -41,6 +41,8 @@ pub trait EntryCategory {
|
||||||
|
|
||||||
fn has_category(&self) -> Result<bool>;
|
fn has_category(&self) -> Result<bool>;
|
||||||
|
|
||||||
|
fn remove_category(&mut self) -> Result<()>;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl EntryCategory for Entry {
|
impl EntryCategory for Entry {
|
||||||
|
@ -82,4 +84,20 @@ impl EntryCategory for Entry {
|
||||||
.map(|x| x.is_some())
|
.map(|x| x.is_some())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Remove the category setting
|
||||||
|
///
|
||||||
|
/// # Warning
|
||||||
|
///
|
||||||
|
/// This does _only_ remove the category setting in the header. This does _not_ remove the
|
||||||
|
/// internal link to the category entry, nor does it remove the category from the store.
|
||||||
|
fn remove_category(&mut self) -> Result<()> {
|
||||||
|
use toml_query::delete::TomlValueDeleteExt;
|
||||||
|
|
||||||
|
self.get_header_mut()
|
||||||
|
.delete("category.value")
|
||||||
|
.chain_err(|| CEK::HeaderWriteError)
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -82,9 +82,26 @@ impl CategoryStore for Store {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Delete a category
|
/// Delete a category
|
||||||
|
///
|
||||||
|
/// Automatically removes all category settings from entries which are linked to this category.
|
||||||
fn delete_category(&self, name: &str) -> Result<()> {
|
fn delete_category(&self, name: &str) -> Result<()> {
|
||||||
|
use libimagentrylink::internal::InternalLinker;
|
||||||
|
use category::Category;
|
||||||
|
|
||||||
trace!("Deleting category: '{}'", name);
|
trace!("Deleting category: '{}'", name);
|
||||||
let sid = mk_category_storeid(self.path().clone(), name)?;
|
let sid = mk_category_storeid(self.path().clone(), name)?;
|
||||||
|
|
||||||
|
{
|
||||||
|
let mut category = self.get(sid.clone())?
|
||||||
|
.ok_or_else(|| CEK::CategoryDoesNotExist)
|
||||||
|
.map_err(CE::from_kind)?;
|
||||||
|
|
||||||
|
for entry in category.get_entries(self)? {
|
||||||
|
let mut entry = entry?;
|
||||||
|
let _ = category.remove_internal_link(&mut entry)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
self.delete(sid).map_err(CE::from)
|
self.delete(sid).map_err(CE::from)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,6 +62,7 @@ CRATES=(
|
||||||
./bin/core/imag-edit
|
./bin/core/imag-edit
|
||||||
./bin/core/imag-ids
|
./bin/core/imag-ids
|
||||||
./bin/core/imag-git
|
./bin/core/imag-git
|
||||||
|
./bin/core/imag-category
|
||||||
./bin/core/imag
|
./bin/core/imag
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue