shell_new.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*
  2. * shell_new.c
  3. */
  4. #include "private.h"
  5. #include <assert.h>
  6. #include <stdlib.h>
  7. #include <unistd.h>
  8. #include "lub/string.h"
  9. /*-------------------------------------------------------- */
  10. static void clish_shell_init(clish_shell_t * this,
  11. const clish_shell_hooks_t * hooks,
  12. void *cookie, FILE * istream,
  13. FILE * ostream,
  14. bool_t stop_on_error)
  15. {
  16. clish_ptype_t *tmp_ptype = NULL;
  17. /* initialise the tree of views */
  18. lub_bintree_init(&this->view_tree,
  19. clish_view_bt_offset(),
  20. clish_view_bt_compare, clish_view_bt_getkey);
  21. /* initialise the tree of ptypes */
  22. lub_bintree_init(&this->ptype_tree,
  23. clish_ptype_bt_offset(),
  24. clish_ptype_bt_compare, clish_ptype_bt_getkey);
  25. /* initialise the tree of vars */
  26. lub_bintree_init(&this->var_tree,
  27. clish_var_bt_offset(),
  28. clish_var_bt_compare, clish_var_bt_getkey);
  29. assert((NULL != hooks) && (NULL != hooks->script_fn));
  30. /* set up defaults */
  31. this->client_hooks = hooks;
  32. this->client_cookie = cookie;
  33. this->global = NULL;
  34. this->startup = NULL;
  35. this->state = SHELL_STATE_INITIALISING;
  36. this->overview = NULL;
  37. this->tinyrl = clish_shell_tinyrl_new(istream, ostream, 0);
  38. this->current_file = NULL;
  39. this->pwdv = NULL;
  40. this->pwdc = 0;
  41. this->depth = -1; /* Current depth is undefined */
  42. this->client = NULL;
  43. this->lockfile = lub_string_dup(CLISH_LOCK_PATH);
  44. this->default_shebang = lub_string_dup("/bin/sh");
  45. this->fifo_name = NULL;
  46. this->interactive = BOOL_TRUE; /* The interactive shell by default. */
  47. /* Create internal ptypes and params */
  48. /* Current depth */
  49. tmp_ptype = clish_shell_find_create_ptype(this,
  50. "__DEPTH", "Depth", "[0-9]+",
  51. CLISH_PTYPE_REGEXP, CLISH_PTYPE_NONE);
  52. assert(tmp_ptype);
  53. this->param_depth = clish_param_new("__cur_depth",
  54. "Current depth", tmp_ptype);
  55. clish_param__set_hidden(this->param_depth, BOOL_TRUE);
  56. /* Current pwd */
  57. tmp_ptype = clish_shell_find_create_ptype(this,
  58. "__PWD", "Path", ".+",
  59. CLISH_PTYPE_REGEXP, CLISH_PTYPE_NONE);
  60. assert(tmp_ptype);
  61. this->param_pwd = clish_param_new("__cur_pwd",
  62. "Current path", tmp_ptype);
  63. clish_param__set_hidden(this->param_pwd, BOOL_TRUE);
  64. /* Interactive */
  65. tmp_ptype = clish_shell_find_create_ptype(this,
  66. "__INTERACTIVE", "Interactive flag", "[01]",
  67. CLISH_PTYPE_REGEXP, CLISH_PTYPE_NONE);
  68. assert(tmp_ptype);
  69. this->param_interactive = clish_param_new("__interactive",
  70. "Interactive flag", tmp_ptype);
  71. clish_param__set_hidden(this->param_interactive, BOOL_TRUE);
  72. /* Push non-NULL istream */
  73. if (istream)
  74. clish_shell_push_fd(this, istream, stop_on_error);
  75. }
  76. /*--------------------------------------------------------- */
  77. static void clish_shell_fini(clish_shell_t * this)
  78. {
  79. clish_view_t *view;
  80. clish_ptype_t *ptype;
  81. clish_var_t *var;
  82. unsigned i;
  83. /* delete each VIEW held */
  84. while ((view = lub_bintree_findfirst(&this->view_tree))) {
  85. lub_bintree_remove(&this->view_tree, view);
  86. clish_view_delete(view);
  87. }
  88. /* delete each PTYPE held */
  89. while ((ptype = lub_bintree_findfirst(&this->ptype_tree))) {
  90. lub_bintree_remove(&this->ptype_tree, ptype);
  91. clish_ptype_delete(ptype);
  92. }
  93. /* delete each VAR held */
  94. while ((var = lub_bintree_findfirst(&this->var_tree))) {
  95. lub_bintree_remove(&this->var_tree, var);
  96. clish_var_delete(var);
  97. }
  98. /* free the textual details */
  99. lub_string_free(this->overview);
  100. /* remove the startup command */
  101. if (this->startup)
  102. clish_command_delete(this->startup);
  103. /* clean up the file stack */
  104. while (BOOL_TRUE == clish_shell_pop_file(this));
  105. /* delete the tinyrl object */
  106. clish_shell_tinyrl_delete(this->tinyrl);
  107. /* finalize each of the pwd strings */
  108. for (i = 0; i < this->pwdc; i++) {
  109. clish_shell__fini_pwd(this->pwdv[i]);
  110. free(this->pwdv[i]);
  111. }
  112. /* free the pwd vector */
  113. free(this->pwdv);
  114. konf_client_free(this->client);
  115. /* Free internal params */
  116. clish_param_delete(this->param_depth);
  117. clish_param_delete(this->param_pwd);
  118. clish_param_delete(this->param_interactive);
  119. lub_string_free(this->lockfile);
  120. lub_string_free(this->default_shebang);
  121. if (this->fifo_name) {
  122. unlink(this->fifo_name);
  123. lub_string_free(this->fifo_name);
  124. }
  125. }
  126. /*-------------------------------------------------------- */
  127. clish_shell_t *clish_shell_new(const clish_shell_hooks_t * hooks,
  128. void *cookie,
  129. FILE * istream,
  130. FILE * ostream,
  131. bool_t stop_on_error)
  132. {
  133. clish_shell_t *this = malloc(sizeof(clish_shell_t));
  134. if (this) {
  135. clish_shell_init(this, hooks, cookie,
  136. istream, ostream, stop_on_error);
  137. if (hooks->init_fn) {
  138. /* now call the client initialisation */
  139. if (BOOL_TRUE != hooks->init_fn(this))
  140. this->state = SHELL_STATE_CLOSING;
  141. }
  142. }
  143. return this;
  144. }
  145. /*--------------------------------------------------------- */
  146. void clish_shell_delete(clish_shell_t * this)
  147. {
  148. /* now call the client finalisation */
  149. if (this->client_hooks->fini_fn)
  150. this->client_hooks->fini_fn(this);
  151. clish_shell_fini(this);
  152. free(this);
  153. }
  154. /*-------------------------------------------------------- */