|
@@ -1,23 +1,17 @@
|
|
use microrm::{
|
|
use microrm::{
|
|
prelude::*,
|
|
prelude::*,
|
|
schema::{
|
|
schema::{
|
|
- datum::{
|
|
|
|
- ConcreteDatumList, DatumDiscriminatorRef, DatumList, QueryEquivalentList, StringQuery,
|
|
|
|
- },
|
|
|
|
- entity::EntityID,
|
|
|
|
|
|
+ datum::{ConcreteDatumList, Datum, DatumDiscriminator, DatumDiscriminatorRef},
|
|
|
|
+ entity::{Entity, EntityID, EntityPartList, EntityPartVisitor, EntityVisitor},
|
|
},
|
|
},
|
|
};
|
|
};
|
|
|
|
|
|
use super::Object;
|
|
use super::Object;
|
|
use crate::UIDCError;
|
|
use crate::UIDCError;
|
|
use clap::{FromArgMatches, Subcommand};
|
|
use clap::{FromArgMatches, Subcommand};
|
|
-use microrm::schema::{
|
|
|
|
- datum::{Datum, DatumDiscriminator},
|
|
|
|
- entity::{Entity, EntityPartList, EntityPartVisitor},
|
|
|
|
-};
|
|
|
|
|
|
|
|
-/// iterate across the list of unique parts (E::Uniques) and add args for each
|
|
|
|
-fn add_uniques<E: Entity>(mut cmd: clap::Command) -> clap::Command {
|
|
|
|
|
|
+/// iterate across the list of key parts (E::Keys) and add args for each
|
|
|
|
+fn add_keys<E: Entity>(mut cmd: clap::Command) -> clap::Command {
|
|
struct UVisitor<'a>(&'a mut clap::Command);
|
|
struct UVisitor<'a>(&'a mut clap::Command);
|
|
impl<'a> EntityPartVisitor for UVisitor<'a> {
|
|
impl<'a> EntityPartVisitor for UVisitor<'a> {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
@@ -28,12 +22,12 @@ fn add_uniques<E: Entity>(mut cmd: clap::Command) -> clap::Command {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- <E::Uniques as EntityPartList>::accept_part_visitor(&mut UVisitor(&mut cmd));
|
|
|
|
|
|
+ <E::Keys as EntityPartList>::accept_part_visitor(&mut UVisitor(&mut cmd));
|
|
|
|
|
|
cmd
|
|
cmd
|
|
}
|
|
}
|
|
|
|
|
|
-fn collect_uniques<E: Entity>(matches: &clap::ArgMatches) -> Vec<String> {
|
|
|
|
|
|
+fn collect_keys<E: Entity>(matches: &clap::ArgMatches) -> Vec<String> {
|
|
struct UVisitor<'a>(&'a clap::ArgMatches, &'a mut Vec<String>);
|
|
struct UVisitor<'a>(&'a clap::ArgMatches, &'a mut Vec<String>);
|
|
impl<'a> EntityPartVisitor for UVisitor<'a> {
|
|
impl<'a> EntityPartVisitor for UVisitor<'a> {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
@@ -46,37 +40,38 @@ fn collect_uniques<E: Entity>(matches: &clap::ArgMatches) -> Vec<String> {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- let mut unique_values = vec![];
|
|
|
|
- <E::Uniques as EntityPartList>::accept_part_visitor(&mut UVisitor(matches, &mut unique_values));
|
|
|
|
- unique_values
|
|
|
|
|
|
+ let mut key_values = vec![];
|
|
|
|
+ <E::Keys as EntityPartList>::accept_part_visitor(&mut UVisitor(matches, &mut key_values));
|
|
|
|
+ key_values
|
|
}
|
|
}
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
#[derive(Clone, Debug)]
|
|
pub enum InterfaceVerb<O: Object> {
|
|
pub enum InterfaceVerb<O: Object> {
|
|
Attach {
|
|
Attach {
|
|
- local_uniques: Vec<String>,
|
|
|
|
|
|
+ local_keys: Vec<String>,
|
|
relation: String,
|
|
relation: String,
|
|
- remote_uniques: Vec<String>,
|
|
|
|
|
|
+ remote_keys: Vec<String>,
|
|
},
|
|
},
|
|
Create(O::CreateParameters),
|
|
Create(O::CreateParameters),
|
|
Delete(Vec<String>),
|
|
Delete(Vec<String>),
|
|
Detach {
|
|
Detach {
|
|
- local_uniques: Vec<String>,
|
|
|
|
|
|
+ local_keys: Vec<String>,
|
|
relation: String,
|
|
relation: String,
|
|
- remote_uniques: Vec<String>,
|
|
|
|
|
|
+ remote_keys: Vec<String>,
|
|
},
|
|
},
|
|
ListAll,
|
|
ListAll,
|
|
Inspect(Vec<String>),
|
|
Inspect(Vec<String>),
|
|
|
|
+ Extra(O::ExtraCommands),
|
|
}
|
|
}
|
|
|
|
|
|
// helper alias for later
|
|
// helper alias for later
|
|
-type UniqueList<O> = <<O as Entity>::Uniques as EntityPartList>::DatumList;
|
|
|
|
|
|
+type UniqueList<E> = <<E as Entity>::Keys as EntityPartList>::DatumList;
|
|
|
|
|
|
impl<O: Object> InterfaceVerb<O> {
|
|
impl<O: Object> InterfaceVerb<O> {
|
|
fn parse_attachment(
|
|
fn parse_attachment(
|
|
matches: &clap::ArgMatches,
|
|
matches: &clap::ArgMatches,
|
|
) -> Result<(Vec<String>, String, Vec<String>), clap::Error> {
|
|
) -> Result<(Vec<String>, String, Vec<String>), clap::Error> {
|
|
- let local_uniques = collect_uniques::<O>(matches);
|
|
|
|
|
|
+ let local_keys = collect_keys::<O>(matches);
|
|
|
|
|
|
let (subcommand, submatches) = matches
|
|
let (subcommand, submatches) = matches
|
|
.subcommand()
|
|
.subcommand()
|
|
@@ -86,7 +81,7 @@ impl<O: Object> InterfaceVerb<O> {
|
|
struct RelationFinder<'l> {
|
|
struct RelationFinder<'l> {
|
|
subcommand: &'l str,
|
|
subcommand: &'l str,
|
|
submatches: &'l clap::ArgMatches,
|
|
submatches: &'l clap::ArgMatches,
|
|
- uniques: &'l mut Vec<String>,
|
|
|
|
|
|
+ keys: &'l mut Vec<String>,
|
|
}
|
|
}
|
|
impl<'l> EntityPartVisitor for RelationFinder<'l> {
|
|
impl<'l> EntityPartVisitor for RelationFinder<'l> {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
|
|
@@ -94,88 +89,220 @@ impl<O: Object> InterfaceVerb<O> {
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
- *self.uniques = collect_uniques::<EP::Entity>(self.submatches);
|
|
|
|
|
|
+ println!(
|
|
|
|
+ "RelationFinder found correct part for subcommand {}!",
|
|
|
|
+ self.subcommand
|
|
|
|
+ );
|
|
|
|
+ println!("EP: {}", std::any::type_name::<EP>());
|
|
|
|
+ // println!("EP: {}", std::any::
|
|
|
|
+ EP::Datum::accept_entity_visitor(self);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ impl<'l> EntityVisitor for RelationFinder<'l> {
|
|
|
|
+ fn visit<E: Entity>(&mut self) {
|
|
|
|
+ println!("\trelationfinder visiting entity {}", E::entity_name());
|
|
|
|
+ *self.keys = collect_keys::<E>(self.submatches);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- let mut remote_uniques = vec![];
|
|
|
|
|
|
+ let mut remote_keys = vec![];
|
|
O::accept_part_visitor(&mut RelationFinder {
|
|
O::accept_part_visitor(&mut RelationFinder {
|
|
subcommand,
|
|
subcommand,
|
|
submatches,
|
|
submatches,
|
|
- uniques: &mut remote_uniques,
|
|
|
|
|
|
+ keys: &mut remote_keys,
|
|
});
|
|
});
|
|
|
|
|
|
- Ok((local_uniques, subcommand.into(), remote_uniques))
|
|
|
|
|
|
+ Ok((local_keys, subcommand.into(), remote_keys))
|
|
}
|
|
}
|
|
|
|
|
|
- fn from_matches(matches: &clap::ArgMatches) -> Result<Self, clap::Error> {
|
|
|
|
- let (subcommand, matches) = matches
|
|
|
|
|
|
+ fn from_matches(parent_matches: &clap::ArgMatches) -> Result<Self, clap::Error> {
|
|
|
|
+ let (subcommand, matches) = parent_matches
|
|
.subcommand()
|
|
.subcommand()
|
|
.ok_or(clap::Error::new(clap::error::ErrorKind::MissingSubcommand))?;
|
|
.ok_or(clap::Error::new(clap::error::ErrorKind::MissingSubcommand))?;
|
|
|
|
|
|
Ok(match subcommand {
|
|
Ok(match subcommand {
|
|
"attach" => {
|
|
"attach" => {
|
|
- let (local_uniques, relation, remote_uniques) = Self::parse_attachment(matches)?;
|
|
|
|
|
|
+ let (local_keys, relation, remote_keys) = Self::parse_attachment(matches)?;
|
|
InterfaceVerb::Attach {
|
|
InterfaceVerb::Attach {
|
|
- local_uniques,
|
|
|
|
|
|
+ local_keys,
|
|
relation,
|
|
relation,
|
|
- remote_uniques,
|
|
|
|
|
|
+ remote_keys,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"create" => InterfaceVerb::Create(
|
|
"create" => InterfaceVerb::Create(
|
|
<O::CreateParameters as clap::FromArgMatches>::from_arg_matches(matches)?,
|
|
<O::CreateParameters as clap::FromArgMatches>::from_arg_matches(matches)?,
|
|
),
|
|
),
|
|
- "delete" => InterfaceVerb::Delete(collect_uniques::<O>(matches)),
|
|
|
|
|
|
+ "delete" => InterfaceVerb::Delete(collect_keys::<O>(matches)),
|
|
"detach" => {
|
|
"detach" => {
|
|
- let (local_uniques, relation, remote_uniques) = Self::parse_attachment(matches)?;
|
|
|
|
|
|
+ let (local_keys, relation, remote_keys) = Self::parse_attachment(matches)?;
|
|
InterfaceVerb::Detach {
|
|
InterfaceVerb::Detach {
|
|
- local_uniques,
|
|
|
|
|
|
+ local_keys,
|
|
relation,
|
|
relation,
|
|
- remote_uniques,
|
|
|
|
|
|
+ remote_keys,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"list" => InterfaceVerb::ListAll,
|
|
"list" => InterfaceVerb::ListAll,
|
|
- "inspect" => InterfaceVerb::Inspect(collect_uniques::<O>(matches)),
|
|
|
|
- _ => unreachable!(),
|
|
|
|
|
|
+ "inspect" => InterfaceVerb::Inspect(collect_keys::<O>(matches)),
|
|
|
|
+ cmd => {
|
|
|
|
+ if <O::ExtraCommands>::has_subcommand(cmd) {
|
|
|
|
+ InterfaceVerb::Extra(<O::ExtraCommands>::from_arg_matches(parent_matches)?)
|
|
|
|
+ }
|
|
|
|
+ else {
|
|
|
|
+ unreachable!()
|
|
|
|
+ }
|
|
|
|
+ },
|
|
})
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+/// helper type for attach and detach verbs
|
|
|
|
+struct Attacher<'l> {
|
|
|
|
+ do_attach: bool,
|
|
|
|
+ relation: &'l str,
|
|
|
|
+ remote_keys: &'l Vec<String>,
|
|
|
|
+ err: Option<UIDCError>,
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<'l> Attacher<'l> {
|
|
|
|
+ fn do_operation<E: Entity>(&mut self, map: &impl AssocInterface<RemoteEntity = E>) {
|
|
|
|
+ match map
|
|
|
|
+ .query_all()
|
|
|
|
+ .keyed(
|
|
|
|
+ UniqueList::<E>::build_equivalent(self.remote_keys.iter().map(String::as_str))
|
|
|
|
+ .unwrap(),
|
|
|
|
+ )
|
|
|
|
+ .get()
|
|
|
|
+ {
|
|
|
|
+ Ok(Some(obj)) => {
|
|
|
|
+ if self.do_attach {
|
|
|
|
+ self.err = map.connect_to(obj.id()).err().map(Into::into);
|
|
|
|
+ } else {
|
|
|
|
+ self.err = map.disconnect_from(obj.id()).err().map(Into::into);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ Ok(None) => {
|
|
|
|
+ self.err = Some(UIDCError::Abort("No such entity to connect to"));
|
|
|
|
+ }
|
|
|
|
+ Err(e) => {
|
|
|
|
+ self.err = Some(e.into());
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<'l> EntityPartVisitor for Attacher<'l> {
|
|
|
|
+ fn visit_datum<EP: microrm::schema::entity::EntityPart>(&mut self, datum: &EP::Datum) {
|
|
|
|
+ if EP::part_name() != self.relation {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ datum.accept_discriminator_ref(self);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+impl<'l> DatumDiscriminatorRef for Attacher<'l> {
|
|
|
|
+ fn visit_entity_id<E: Entity>(&mut self, _: &E::ID) {
|
|
|
|
+ unreachable!()
|
|
|
|
+ }
|
|
|
|
+ fn visit_serialized<T: serde::Serialize + serde::de::DeserializeOwned>(&mut self, _: &T) {
|
|
|
|
+ unreachable!()
|
|
|
|
+ }
|
|
|
|
+ fn visit_bare_field<T: Datum>(&mut self, _: &T) {
|
|
|
|
+ unreachable!()
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn visit_assoc_map<E: Entity>(&mut self, map: &AssocMap<E>) {
|
|
|
|
+ self.do_operation(map);
|
|
|
|
+ }
|
|
|
|
+ fn visit_assoc_range<R: microrm::schema::Relation>(
|
|
|
|
+ &mut self,
|
|
|
|
+ map: µrm::schema::AssocRange<R>,
|
|
|
|
+ ) {
|
|
|
|
+ self.do_operation(map);
|
|
|
|
+ }
|
|
|
|
+ fn visit_assoc_domain<R: microrm::schema::Relation>(
|
|
|
|
+ &mut self,
|
|
|
|
+ map: µrm::schema::AssocDomain<R>,
|
|
|
|
+ ) {
|
|
|
|
+ self.do_operation(map);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
#[derive(Debug)]
|
|
#[derive(Debug)]
|
|
pub struct ClapInterface<O: Object> {
|
|
pub struct ClapInterface<O: Object> {
|
|
verb: InterfaceVerb<O>,
|
|
verb: InterfaceVerb<O>,
|
|
_ghost: std::marker::PhantomData<O>,
|
|
_ghost: std::marker::PhantomData<O>,
|
|
}
|
|
}
|
|
|
|
|
|
-impl<O: Object> ClapInterface<O> {
|
|
|
|
|
|
+impl<O: Object<Error = UIDCError>> ClapInterface<O> {
|
|
pub fn perform(
|
|
pub fn perform(
|
|
&self,
|
|
&self,
|
|
|
|
+ data: &O::ExtraCommandData,
|
|
query_ctx: impl microrm::prelude::Queryable<EntityOutput = O>,
|
|
query_ctx: impl microrm::prelude::Queryable<EntityOutput = O>,
|
|
insert_ctx: &impl microrm::prelude::Insertable<O>,
|
|
insert_ctx: &impl microrm::prelude::Insertable<O>,
|
|
) -> Result<(), UIDCError> {
|
|
) -> Result<(), UIDCError> {
|
|
match &self.verb {
|
|
match &self.verb {
|
|
InterfaceVerb::Attach {
|
|
InterfaceVerb::Attach {
|
|
- local_uniques,
|
|
|
|
|
|
+ local_keys,
|
|
relation,
|
|
relation,
|
|
- remote_uniques,
|
|
|
|
|
|
+ remote_keys,
|
|
} => {
|
|
} => {
|
|
- // O::attach(query_ctx, local_uniques, relation, remote_uniques)?;
|
|
|
|
- todo!()
|
|
|
|
|
|
+ let outer_obj = query_ctx
|
|
|
|
+ .keyed(
|
|
|
|
+ UniqueList::<O>::build_equivalent(local_keys.iter().map(String::as_str))
|
|
|
|
+ .unwrap(),
|
|
|
|
+ )
|
|
|
|
+ .get()?
|
|
|
|
+ .ok_or(UIDCError::Abort("Could not find object to attach to"))?;
|
|
|
|
+
|
|
|
|
+ let mut attacher = Attacher {
|
|
|
|
+ do_attach: true,
|
|
|
|
+ relation,
|
|
|
|
+ remote_keys,
|
|
|
|
+ err: None,
|
|
|
|
+ };
|
|
|
|
+ outer_obj.accept_part_visitor_ref(&mut attacher);
|
|
|
|
+
|
|
|
|
+ if let Some(err) = attacher.err {
|
|
|
|
+ return Err(err);
|
|
|
|
+ }
|
|
}
|
|
}
|
|
InterfaceVerb::Create(params) => {
|
|
InterfaceVerb::Create(params) => {
|
|
insert_ctx.insert(O::create_from_params(params)?)?;
|
|
insert_ctx.insert(O::create_from_params(params)?)?;
|
|
}
|
|
}
|
|
- InterfaceVerb::Delete(uniques) => {
|
|
|
|
|
|
+ InterfaceVerb::Delete(keys) => {
|
|
query_ctx
|
|
query_ctx
|
|
- .unique(UniqueList::<O>::build_equivalent(uniques.iter().map(String::as_str)).unwrap())
|
|
|
|
|
|
+ .keyed(
|
|
|
|
+ UniqueList::<O>::build_equivalent(keys.iter().map(String::as_str))
|
|
|
|
+ .unwrap(),
|
|
|
|
+ )
|
|
.delete()?;
|
|
.delete()?;
|
|
}
|
|
}
|
|
InterfaceVerb::Detach {
|
|
InterfaceVerb::Detach {
|
|
- local_uniques,
|
|
|
|
|
|
+ local_keys,
|
|
relation,
|
|
relation,
|
|
- remote_uniques,
|
|
|
|
|
|
+ remote_keys,
|
|
} => {
|
|
} => {
|
|
- todo!()
|
|
|
|
|
|
+ let outer_obj = query_ctx
|
|
|
|
+ .keyed(
|
|
|
|
+ UniqueList::<O>::build_equivalent(local_keys.iter().map(String::as_str))
|
|
|
|
+ .unwrap(),
|
|
|
|
+ )
|
|
|
|
+ .get()?
|
|
|
|
+ .ok_or(UIDCError::Abort("Could not find object to detach from"))?;
|
|
|
|
+
|
|
|
|
+ let mut attacher = Attacher {
|
|
|
|
+ do_attach: false,
|
|
|
|
+ relation,
|
|
|
|
+ remote_keys,
|
|
|
|
+ err: None,
|
|
|
|
+ };
|
|
|
|
+ outer_obj.accept_part_visitor_ref(&mut attacher);
|
|
|
|
+
|
|
|
|
+ if let Some(err) = attacher.err {
|
|
|
|
+ return Err(err);
|
|
|
|
+ }
|
|
}
|
|
}
|
|
InterfaceVerb::ListAll => {
|
|
InterfaceVerb::ListAll => {
|
|
println!(
|
|
println!(
|
|
@@ -187,9 +314,12 @@ impl<O: Object> ClapInterface<O> {
|
|
println!(" - {}", obj.shortname());
|
|
println!(" - {}", obj.shortname());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- InterfaceVerb::Inspect(uniques) => {
|
|
|
|
|
|
+ InterfaceVerb::Inspect(keys) => {
|
|
let obj = query_ctx
|
|
let obj = query_ctx
|
|
- .unique(UniqueList::<O>::build_equivalent(uniques.iter().map(String::as_str)).unwrap())
|
|
|
|
|
|
+ .keyed(
|
|
|
|
+ UniqueList::<O>::build_equivalent(keys.iter().map(String::as_str))
|
|
|
|
+ .unwrap(),
|
|
|
|
+ )
|
|
.get()?
|
|
.get()?
|
|
.ok_or(UIDCError::Abort("no such element, cannot inspect"))?;
|
|
.ok_or(UIDCError::Abort("no such element, cannot inspect"))?;
|
|
println!("{:#?}", obj.as_ref());
|
|
println!("{:#?}", obj.as_ref());
|
|
@@ -241,7 +371,10 @@ impl<O: Object> ClapInterface<O> {
|
|
}
|
|
}
|
|
|
|
|
|
obj.accept_part_visitor_ref(&mut AssocFieldWalker);
|
|
obj.accept_part_visitor_ref(&mut AssocFieldWalker);
|
|
- }
|
|
|
|
|
|
+ },
|
|
|
|
+ InterfaceVerb::Extra(extra) => {
|
|
|
|
+ O::run_extra_command(data, extra, query_ctx, insert_ctx)?;
|
|
|
|
+ },
|
|
}
|
|
}
|
|
Ok(())
|
|
Ok(())
|
|
}
|
|
}
|
|
@@ -262,15 +395,15 @@ impl<O: Object> ClapInterface<O> {
|
|
}
|
|
}
|
|
fn visit_bare_field<T: Datum>(&mut self) {}
|
|
fn visit_bare_field<T: Datum>(&mut self) {}
|
|
fn visit_assoc_map<E: Entity>(&mut self) {
|
|
fn visit_assoc_map<E: Entity>(&mut self) {
|
|
- self.0.push(add_uniques::<E>(clap::Command::new(self.1)));
|
|
|
|
|
|
+ self.0.push(add_keys::<E>(clap::Command::new(self.1)));
|
|
}
|
|
}
|
|
fn visit_assoc_domain<R: microrm::schema::Relation>(&mut self) {
|
|
fn visit_assoc_domain<R: microrm::schema::Relation>(&mut self) {
|
|
self.0
|
|
self.0
|
|
- .push(add_uniques::<R::Range>(clap::Command::new(self.1)));
|
|
|
|
|
|
+ .push(add_keys::<R::Range>(clap::Command::new(self.1)));
|
|
}
|
|
}
|
|
fn visit_assoc_range<R: microrm::schema::Relation>(&mut self) {
|
|
fn visit_assoc_range<R: microrm::schema::Relation>(&mut self) {
|
|
self.0
|
|
self.0
|
|
- .push(add_uniques::<R::Domain>(clap::Command::new(self.1)));
|
|
|
|
|
|
+ .push(add_keys::<R::Domain>(clap::Command::new(self.1)));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -302,22 +435,28 @@ impl<O: Object> FromArgMatches for ClapInterface<O> {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-impl<O: Object> Subcommand for ClapInterface<O> {
|
|
|
|
|
|
+impl<O: Object<Error = UIDCError>> Subcommand for ClapInterface<O> {
|
|
fn has_subcommand(name: &str) -> bool {
|
|
fn has_subcommand(name: &str) -> bool {
|
|
todo!()
|
|
todo!()
|
|
}
|
|
}
|
|
|
|
|
|
fn augment_subcommands(cmd: clap::Command) -> clap::Command {
|
|
fn augment_subcommands(cmd: clap::Command) -> clap::Command {
|
|
- cmd.subcommand(
|
|
|
|
- add_uniques::<O>(clap::Command::new("attach"))
|
|
|
|
|
|
+ let cmd = cmd.subcommand(
|
|
|
|
+ add_keys::<O>(clap::Command::new("attach"))
|
|
|
|
+ .subcommands(Self::make_relation_subcommands())
|
|
|
|
+ .subcommand_required(true),
|
|
|
|
+ )
|
|
|
|
+ .subcommand(
|
|
|
|
+ add_keys::<O>(clap::Command::new("detach"))
|
|
.subcommands(Self::make_relation_subcommands())
|
|
.subcommands(Self::make_relation_subcommands())
|
|
.subcommand_required(true),
|
|
.subcommand_required(true),
|
|
)
|
|
)
|
|
.subcommand(<O::CreateParameters as clap::CommandFactory>::command().name("create"))
|
|
.subcommand(<O::CreateParameters as clap::CommandFactory>::command().name("create"))
|
|
- .subcommand(add_uniques::<O>(clap::Command::new("delete")))
|
|
|
|
- .subcommand(add_uniques::<O>(clap::Command::new("inspect")))
|
|
|
|
- .subcommand(clap::Command::new("list"))
|
|
|
|
- .subcommands(O::extra_commands())
|
|
|
|
|
|
+ .subcommand(add_keys::<O>(clap::Command::new("delete")))
|
|
|
|
+ .subcommand(add_keys::<O>(clap::Command::new("inspect")))
|
|
|
|
+ .subcommand(clap::Command::new("list"));
|
|
|
|
+
|
|
|
|
+ <O::ExtraCommands>::augment_subcommands(cmd)
|
|
}
|
|
}
|
|
|
|
|
|
fn augment_subcommands_for_update(cmd: clap::Command) -> clap::Command {
|
|
fn augment_subcommands_for_update(cmd: clap::Command) -> clap::Command {
|