edit_readline.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. /*
  2. * Command line editing and history wrapper for readline
  3. * Copyright (c) 2010, Jouni Malinen <j@w1.fi>
  4. *
  5. * This software may be distributed under the terms of the BSD license.
  6. * See README for more details.
  7. */
  8. #include "includes.h"
  9. #include <readline/readline.h>
  10. #include <readline/history.h>
  11. #include "common.h"
  12. #include "eloop.h"
  13. #include "edit.h"
  14. static void *edit_cb_ctx;
  15. static void (*edit_cmd_cb)(void *ctx, char *cmd);
  16. static void (*edit_eof_cb)(void *ctx);
  17. static char ** (*edit_completion_cb)(void *ctx, const char *cmd, int pos) =
  18. NULL;
  19. static char **pending_completions = NULL;
  20. static void readline_free_completions(void)
  21. {
  22. int i;
  23. if (pending_completions == NULL)
  24. return;
  25. for (i = 0; pending_completions[i]; i++)
  26. os_free(pending_completions[i]);
  27. os_free(pending_completions);
  28. pending_completions = NULL;
  29. }
  30. static char * readline_completion_func(const char *text, int state)
  31. {
  32. static int pos = 0;
  33. static size_t len = 0;
  34. if (pending_completions == NULL) {
  35. rl_attempted_completion_over = 1;
  36. return NULL;
  37. }
  38. if (state == 0) {
  39. pos = 0;
  40. len = os_strlen(text);
  41. }
  42. for (; pending_completions[pos]; pos++) {
  43. if (strncmp(pending_completions[pos], text, len) == 0)
  44. return strdup(pending_completions[pos++]);
  45. }
  46. rl_attempted_completion_over = 1;
  47. return NULL;
  48. }
  49. static char ** readline_completion(const char *text, int start, int end)
  50. {
  51. readline_free_completions();
  52. if (edit_completion_cb)
  53. pending_completions = edit_completion_cb(edit_cb_ctx,
  54. rl_line_buffer, end);
  55. return rl_completion_matches(text, readline_completion_func);
  56. }
  57. static void edit_read_char(int sock, void *eloop_ctx, void *sock_ctx)
  58. {
  59. rl_callback_read_char();
  60. }
  61. static void trunc_nl(char *str)
  62. {
  63. char *pos = str;
  64. while (*pos != '\0') {
  65. if (*pos == '\n') {
  66. *pos = '\0';
  67. break;
  68. }
  69. pos++;
  70. }
  71. }
  72. static void readline_cmd_handler(char *cmd)
  73. {
  74. if (cmd && *cmd) {
  75. HIST_ENTRY *h;
  76. while (next_history())
  77. ;
  78. h = previous_history();
  79. if (h == NULL || os_strcmp(cmd, h->line) != 0)
  80. add_history(cmd);
  81. next_history();
  82. }
  83. if (cmd == NULL) {
  84. edit_eof_cb(edit_cb_ctx);
  85. return;
  86. }
  87. trunc_nl(cmd);
  88. edit_cmd_cb(edit_cb_ctx, cmd);
  89. }
  90. int edit_init(void (*cmd_cb)(void *ctx, char *cmd),
  91. void (*eof_cb)(void *ctx),
  92. char ** (*completion_cb)(void *ctx, const char *cmd, int pos),
  93. void *ctx, const char *history_file, const char *ps)
  94. {
  95. edit_cb_ctx = ctx;
  96. edit_cmd_cb = cmd_cb;
  97. edit_eof_cb = eof_cb;
  98. edit_completion_cb = completion_cb;
  99. rl_attempted_completion_function = readline_completion;
  100. if (history_file) {
  101. read_history(history_file);
  102. stifle_history(100);
  103. }
  104. eloop_register_read_sock(STDIN_FILENO, edit_read_char, NULL, NULL);
  105. if (ps) {
  106. size_t blen = os_strlen(ps) + 3;
  107. char *ps2 = os_malloc(blen);
  108. if (ps2) {
  109. os_snprintf(ps2, blen, "%s> ", ps);
  110. rl_callback_handler_install(ps2, readline_cmd_handler);
  111. os_free(ps2);
  112. return 0;
  113. }
  114. }
  115. rl_callback_handler_install("> ", readline_cmd_handler);
  116. return 0;
  117. }
  118. void edit_deinit(const char *history_file,
  119. int (*filter_cb)(void *ctx, const char *cmd))
  120. {
  121. rl_set_prompt("");
  122. rl_replace_line("", 0);
  123. rl_redisplay();
  124. rl_callback_handler_remove();
  125. readline_free_completions();
  126. eloop_unregister_read_sock(STDIN_FILENO);
  127. if (history_file) {
  128. /* Save command history, excluding lines that may contain
  129. * passwords. */
  130. HIST_ENTRY *h;
  131. history_set_pos(0);
  132. while ((h = current_history())) {
  133. char *p = h->line;
  134. while (*p == ' ' || *p == '\t')
  135. p++;
  136. if (filter_cb && filter_cb(edit_cb_ctx, p)) {
  137. h = remove_history(where_history());
  138. if (h) {
  139. free(h->line);
  140. free(h->data);
  141. free(h);
  142. } else
  143. next_history();
  144. } else
  145. next_history();
  146. }
  147. write_history(history_file);
  148. }
  149. }
  150. void edit_clear_line(void)
  151. {
  152. }
  153. void edit_redraw(void)
  154. {
  155. rl_on_new_line();
  156. rl_redisplay();
  157. }