kcommand.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. #include <stdlib.h>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <assert.h>
  5. #include <faux/str.h>
  6. #include <faux/list.h>
  7. #include <faux/error.h>
  8. #include <klish/khelper.h>
  9. #include <klish/kparam.h>
  10. #include <klish/kcommand.h>
  11. struct kcommand_s {
  12. char *name;
  13. char *help;
  14. faux_list_t *params;
  15. };
  16. // Simple attributes
  17. // Name
  18. KGET_STR(command, name);
  19. KSET_STR_ONCE(command, name);
  20. // Help
  21. KGET_STR(command, help);
  22. KSET_STR(command, help);
  23. // PARAM list
  24. KCMP_NESTED_BY_KEY(command, param, name);
  25. KADD_NESTED(command, param);
  26. KFIND_NESTED(command, param);
  27. static kcommand_t *kcommand_new_empty(void)
  28. {
  29. kcommand_t *command = NULL;
  30. command = faux_zmalloc(sizeof(*command));
  31. assert(command);
  32. if (!command)
  33. return NULL;
  34. // Initialize
  35. command->name = NULL;
  36. command->help = NULL;
  37. command->params = faux_list_new(FAUX_LIST_UNSORTED, FAUX_LIST_UNIQUE,
  38. NULL, kcommand_param_kcompare,
  39. (void (*)(void *))kcommand_free);
  40. assert(command->params);
  41. return command;
  42. }
  43. kcommand_t *kcommand_new(const icommand_t *info, kcommand_error_e *error)
  44. {
  45. kcommand_t *command = NULL;
  46. command = kcommand_new_empty();
  47. assert(command);
  48. if (!command) {
  49. if (error)
  50. *error = KCOMMAND_ERROR_ALLOC;
  51. return NULL;
  52. }
  53. if (!info)
  54. return command;
  55. if (!kcommand_parse(command, info, error)) {
  56. kcommand_free(command);
  57. return NULL;
  58. }
  59. return command;
  60. }
  61. void kcommand_free(kcommand_t *command)
  62. {
  63. if (!command)
  64. return;
  65. faux_str_free(command->name);
  66. faux_str_free(command->help);
  67. faux_list_free(command->params);
  68. faux_free(command);
  69. }
  70. const char *kcommand_strerror(kcommand_error_e error)
  71. {
  72. const char *str = NULL;
  73. switch (error) {
  74. case KCOMMAND_ERROR_OK:
  75. str = "Ok";
  76. break;
  77. case KCOMMAND_ERROR_INTERNAL:
  78. str = "Internal error";
  79. break;
  80. case KCOMMAND_ERROR_ALLOC:
  81. str = "Memory allocation error";
  82. break;
  83. case KCOMMAND_ERROR_ATTR_NAME:
  84. str = "Illegal 'name' attribute";
  85. break;
  86. case KCOMMAND_ERROR_ATTR_HELP:
  87. str = "Illegal 'help' attribute";
  88. break;
  89. default:
  90. str = "Unknown error";
  91. break;
  92. }
  93. return str;
  94. }
  95. bool_t kcommand_parse(kcommand_t *command, const icommand_t *info, kcommand_error_e *error)
  96. {
  97. // Name [mandatory]
  98. if (faux_str_is_empty(info->name)) {
  99. if (error)
  100. *error = KCOMMAND_ERROR_ATTR_NAME;
  101. return BOOL_FALSE;
  102. } else {
  103. if (!kcommand_set_name(command, info->name)) {
  104. if (error)
  105. *error = KCOMMAND_ERROR_ATTR_NAME;
  106. return BOOL_FALSE;
  107. }
  108. }
  109. // Help
  110. if (!faux_str_is_empty(info->name)) {
  111. if (!kcommand_set_help(command, info->help)) {
  112. if (error)
  113. *error = KCOMMAND_ERROR_ATTR_HELP;
  114. return BOOL_FALSE;
  115. }
  116. }
  117. return BOOL_TRUE;
  118. }
  119. bool_t kcommand_nested_from_icommand(kcommand_t *kcommand, icommand_t *icommand,
  120. faux_error_t *error_stack)
  121. {
  122. bool_t retval = BOOL_TRUE;
  123. if (!kcommand || !icommand) {
  124. faux_error_add(error_stack,
  125. kcommand_strerror(KCOMMAND_ERROR_INTERNAL));
  126. return BOOL_FALSE;
  127. }
  128. /*
  129. // ACTION list
  130. if (icommand->actions) {
  131. iaction_t **p_iaction = NULL;
  132. for (p_iaction = *icommand->actions; *p_iaction; p_iaction++) {
  133. kaction_t *kaction = NULL;
  134. iaction_t *iaction = *p_iaction;
  135. iaction = iaction;
  136. printf("action\n");
  137. // kaction = kaction_from_iaction(iaction, error_stack);
  138. // if (!kaction) {
  139. // retval = BOOL_FALSE;
  140. // continue;
  141. // }
  142. kaction = kaction;
  143. }
  144. }
  145. */
  146. return retval;
  147. }
  148. kcommand_t *kcommand_from_icommand(icommand_t *icommand, faux_error_t *error_stack)
  149. {
  150. kcommand_t *kcommand = NULL;
  151. kcommand_error_e kcommand_error = KCOMMAND_ERROR_OK;
  152. kcommand = kcommand_new(icommand, &kcommand_error);
  153. if (!kcommand) {
  154. char *msg = NULL;
  155. msg = faux_str_sprintf("COMMAND \"%s\": %s",
  156. icommand->name ? icommand->name : "(null)",
  157. kcommand_strerror(kcommand_error));
  158. faux_error_add(error_stack, msg);
  159. faux_str_free(msg);
  160. return NULL;
  161. }
  162. printf("command %s\n", kcommand_name(kcommand));
  163. // Parse nested elements
  164. if (!kcommand_nested_from_icommand(kcommand, icommand, error_stack)) {
  165. char *msg = NULL;
  166. msg = faux_str_sprintf("COMMAND \"%s\": Illegal nested elements",
  167. kcommand_name(kcommand));
  168. faux_error_add(error_stack, msg);
  169. faux_str_free(msg);
  170. kcommand_free(kcommand);
  171. return NULL;
  172. }
  173. return kcommand;
  174. }