clap.rs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. use super::{Object,ObjectExt};
  2. #[derive(Debug)]
  3. pub struct ClapInterface<O: ObjectExt> {
  4. verb: InterfaceVerb<O>,
  5. _ghost: std::marker::PhantomData<O>,
  6. }
  7. impl<O: ObjectExt> ClapInterface<O> {
  8. pub fn perform(
  9. &self,
  10. query_ctx: impl microrm::prelude::Queryable<EntityOutput = O>,
  11. insert_ctx: &impl microrm::prelude::Insertable<O>,
  12. ) -> Result<(), UIDCError> {
  13. match &self.verb {
  14. InterfaceVerb::Attach { relation, remote_keys } => {
  15. todo!()
  16. }
  17. InterfaceVerb::Create(params) => {
  18. O::create(insert_ctx, &params)?;
  19. }
  20. InterfaceVerb::Delete(keys) => {
  21. O::delete(query_ctx, O::build_keys(keys))?;
  22. }
  23. InterfaceVerb::Detach => {
  24. todo!()
  25. }
  26. InterfaceVerb::ListAll => {
  27. O::list_all(query_ctx)?;
  28. }
  29. InterfaceVerb::Inspect(keys) => {
  30. O::inspect(query_ctx, O::build_keys(keys))?;
  31. }
  32. }
  33. Ok(())
  34. }
  35. /// iterate across the list of key parts (O::Uniques) and add args for each
  36. fn add_keys(mut cmd: clap::Command) -> clap::Command {
  37. struct UVisitor<'a>(&'a mut clap::Command);
  38. impl<'a> EntityPartVisitor for UVisitor<'a> {
  39. fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
  40. let arg = clap::Arg::new(EP::part_name())
  41. .required(true)
  42. .help(EP::desc());
  43. *self.0 = self.0.clone().arg(arg);
  44. }
  45. }
  46. <O::Uniques as EntityPartList>::accept_part_visitor(&mut UVisitor(&mut cmd));
  47. cmd
  48. }
  49. fn make_relation_subcommands() -> impl Iterator<Item = clap::Command> {
  50. let mut out = vec![];
  51. struct PartVisitor<'l>(&'l mut Vec<clap::Command>);
  52. impl<'l> EntityPartVisitor for PartVisitor<'l> {
  53. fn visit<EP: microrm::schema::entity::EntityPart>(&mut self) {
  54. struct Discriminator<'l>(&'l mut Vec<clap::Command>, &'static str);
  55. impl<'l> DatumDiscriminator for Discriminator<'l> {
  56. fn visit_entity_id<E: Entity>(&mut self) {}
  57. fn visit_serialized<T: serde::Serialize + serde::de::DeserializeOwned>(
  58. &mut self,
  59. ) {
  60. }
  61. fn visit_bare_field<T: Datum>(&mut self) {}
  62. fn visit_assoc_map<E: Entity>(&mut self) {
  63. self.0.push(clap::Command::new(self.1));
  64. }
  65. fn visit_assoc_domain<R: microrm::schema::Relation>(&mut self) {
  66. self.0.push(clap::Command::new(self.1));
  67. }
  68. fn visit_assoc_range<R: microrm::schema::Relation>(&mut self) {
  69. self.0.push(clap::Command::new(self.1));
  70. }
  71. }
  72. <EP::Datum as Datum>::accept_discriminator(&mut Discriminator(
  73. self.0,
  74. EP::part_name(),
  75. ));
  76. }
  77. }
  78. O::accept_part_visitor(&mut PartVisitor(&mut out));
  79. out.into_iter()
  80. }
  81. }
  82. impl<O: ObjectExt> FromArgMatches for ClapInterface<O> {
  83. fn from_arg_matches(matches: &clap::ArgMatches) -> Result<Self, clap::Error> {
  84. let verb = InterfaceVerb::from_matches(matches);
  85. Ok(Self {
  86. verb: verb?,
  87. _ghost: Default::default(),
  88. })
  89. }
  90. fn update_from_arg_matches(&mut self, matches: &clap::ArgMatches) -> Result<(), clap::Error> {
  91. todo!()
  92. }
  93. }
  94. impl<O: ObjectExt> Subcommand for ClapInterface<O> {
  95. fn has_subcommand(name: &str) -> bool {
  96. todo!()
  97. }
  98. fn augment_subcommands(cmd: clap::Command) -> clap::Command {
  99. cmd.subcommand(
  100. Self::add_keys(clap::Command::new("attach"))
  101. .subcommands(Self::make_relation_subcommands())
  102. .subcommand_required(true),
  103. )
  104. .subcommand(<O::CreateParameters as clap::CommandFactory>::command().name("create"))
  105. .subcommand(Self::add_keys(clap::Command::new("delete")))
  106. .subcommand(Self::add_keys(clap::Command::new("inspect")))
  107. .subcommand(clap::Command::new("list"))
  108. .subcommands(O::extra_commands())
  109. }
  110. fn augment_subcommands_for_update(cmd: clap::Command) -> clap::Command {
  111. todo!()
  112. }
  113. }