/* * shell_tinyrl.c * * This is a specialisation of the tinyrl_t class which maps the readline * functionality to the CLISH envrionment. */ #include "private.h" #include #include #include #include #include "tinyrl/tinyrl.h" #include "tinyrl/history.h" #include "lub/string.h" /* This is used to hold context during tinyrl callbacks */ typedef struct _context context_t; struct _context { clish_shell_t *shell; const clish_command_t *command; clish_pargv_t *pargv; }; /*-------------------------------------------------------- */ static bool_t clish_shell_tinyrl_key_help(tinyrl_t * this, int key) { bool_t result = BOOL_TRUE; if (BOOL_TRUE == tinyrl_is_quoting(this)) { /* if we are in the middle of a quote then simply enter a space */ result = tinyrl_insert_text(this, "?"); } else { /* get the context */ context_t *context = tinyrl__get_context(this); tinyrl_crlf(this); tinyrl_crlf(this); clish_shell_help(context->shell, tinyrl__get_line(this)); tinyrl_crlf(this); tinyrl_reset_line_state(this); } /* keep the compiler happy */ key = key; return result; } /*-------------------------------------------------------- */ /* Generator function for command completion. STATE lets us * know whether to start from scratch; without any state * (i.e. STATE == 0), then we start at the top of the list. */ /*lint -e818 Pointer paramter 'this' could be declared as pointing to const */ static char *clish_shell_tinyrl_word_generator(tinyrl_t * this, const char *line, unsigned offset, unsigned state) { /* get the context */ context_t *context = tinyrl__get_context(this); return clish_shell_word_generator(context->shell, line, offset, state); } /*lint +e818 */ /*-------------------------------------------------------- */ /* * Expand the current line with any history substitutions */ static clish_pargv_status_t clish_shell_tinyrl_expand(tinyrl_t * this) { clish_pargv_status_t status = CLISH_LINE_OK; int rtn; char *buffer; /* first of all perform any history substitutions */ rtn = tinyrl_history_expand(tinyrl__get_history(this), tinyrl__get_line(this), &buffer); switch (rtn) { case -1: /* error in expansion */ status = CLISH_BAD_HISTORY; break; case 0: /*no expansion */ break; case 1: /* expansion occured correctly */ tinyrl_replace_line(this, buffer, 1); break; case 2: /* just display line */ fprintf(tinyrl__get_ostream(this), "\n%s", buffer); free(buffer); buffer = NULL; break; default: break; } free(buffer); return status; } /*-------------------------------------------------------- */ /* * This is a CLISH specific completion function. * If the current prefix is not a recognised prefix then * an error is flagged. * If it is a recognisable prefix then possible completions are displayed * or a unique completion is inserted. */ static tinyrl_match_e clish_shell_tinyrl_complete(tinyrl_t * this) { // context_t *context = tinyrl__get_context(this); tinyrl_match_e status; /* first of all perform any history expansion */ (void)clish_shell_tinyrl_expand(this); /* perform normal completion */ status = tinyrl_complete(this); switch (status) { case TINYRL_NO_MATCH: { if (BOOL_FALSE == tinyrl_is_completion_error_over(this)) { /* The user hasn't even entered a valid prefix!!! */ // tinyrl_crlf(this); // clish_shell_help(context->shell, // tinyrl__get_line(this)); // tinyrl_crlf(this); // tinyrl_reset_line_state(this); } break; } case TINYRL_MATCH: case TINYRL_MATCH_WITH_EXTENSIONS: case TINYRL_COMPLETED_MATCH: case TINYRL_AMBIGUOUS: case TINYRL_COMPLETED_AMBIGUOUS: { /* the default completion function will have prompted for completions as * necessary */ break; } } return status; } /*--------------------------------------------------------- */ static bool_t clish_shell_tinyrl_key_space(tinyrl_t * this, int key) { bool_t result = BOOL_FALSE; tinyrl_match_e status; context_t *context = tinyrl__get_context(this); const char *line = tinyrl__get_line(this); clish_pargv_status_t arg_status; const clish_command_t *cmd = NULL; clish_pargv_t *pargv = NULL; if (BOOL_TRUE == tinyrl_is_quoting(this)) { /* if we are in the middle of a quote then simply enter a space */ result = BOOL_TRUE; } else { arg_status = clish_shell_parse(context->shell, line, &cmd, &pargv); if (pargv) clish_pargv_delete(pargv); switch (arg_status) { case CLISH_LINE_OK: case CLISH_LINE_PARTIAL: if (' ' != line[strlen(line) - 1]) result = BOOL_TRUE; break; default: /* perform word completion */ status = clish_shell_tinyrl_complete(this); switch (status) { case TINYRL_NO_MATCH: case TINYRL_AMBIGUOUS: /* ambiguous result signal an issue */ break; case TINYRL_COMPLETED_AMBIGUOUS: /* perform word completion again in case we just did case modification the first time */ status = clish_shell_tinyrl_complete(this); if (status == TINYRL_MATCH_WITH_EXTENSIONS) { /* all is well with the world just enter a space */ result = BOOL_TRUE; } break; case TINYRL_MATCH: case TINYRL_MATCH_WITH_EXTENSIONS: case TINYRL_COMPLETED_MATCH: /* all is well with the world just enter a space */ result = BOOL_TRUE; break; } break; } } if (result) result = tinyrl_insert_text(this, " "); /* keep compiler happy */ key = key; return result; } /*-------------------------------------------------------- */ static bool_t clish_shell_tinyrl_key_enter(tinyrl_t * this, int key) { context_t *context = tinyrl__get_context(this); const clish_command_t *cmd = NULL; const char *line = tinyrl__get_line(this); bool_t result = BOOL_FALSE; if (*line) { /* try and parse the command */ cmd = clish_shell_resolve_command(context->shell, line); if (NULL == cmd) { tinyrl_match_e status = clish_shell_tinyrl_complete(this); switch (status) { case TINYRL_NO_MATCH: case TINYRL_AMBIGUOUS: case TINYRL_COMPLETED_AMBIGUOUS: { /* failed to get a unique match... */ break; } case TINYRL_MATCH: case TINYRL_MATCH_WITH_EXTENSIONS: case TINYRL_COMPLETED_MATCH: { /* re-fetch the line as it may have changed * due to auto-completion */ line = tinyrl__get_line(this); /* get the command to parse? */ cmd = clish_shell_resolve_command (context->shell, line); if (NULL == cmd) { /* * We have had a match but it is not a command * so add a space so as not to confuse the user */ result = tinyrl_insert_text(this, " "); } break; } } } if (NULL != cmd) { clish_pargv_status_t arg_status; tinyrl_crlf(this); /* we've got a command so check the syntax */ arg_status = clish_shell_parse(context->shell, line, &context->command, &context->pargv); switch (arg_status) { case CLISH_LINE_OK: tinyrl_done(this); result = BOOL_TRUE; break; case CLISH_BAD_HISTORY: fprintf(stderr, "Error: Bad history entry.\n"); break; case CLISH_BAD_CMD: fprintf(stderr, "Error: Illegal command line.\n"); break; case CLISH_BAD_PARAM: fprintf(stderr, "Error: Illegal parameter.\n"); break; case CLISH_LINE_PARTIAL: fprintf(stderr, "Error: The command is not completed.\n"); break; } if (CLISH_LINE_OK != arg_status) tinyrl_reset_line_state(this); } } else { /* nothing to pass simply move down the screen */ tinyrl_crlf(this); tinyrl_reset_line_state(this); result = BOOL_TRUE; } if ((BOOL_FALSE == result) && (BOOL_FALSE == tinyrl__get_isatty(this))) { /* we've found an error in a script */ context->shell->state = SHELL_STATE_SCRIPT_ERROR; } /* keep the compiler happy */ key = key; return result; } /*-------------------------------------------------------- */ /* This is the completion function provided for CLISH */ static tinyrl_completion_func_t clish_shell_tinyrl_completion; static char **clish_shell_tinyrl_completion(tinyrl_t * this, const char *line, unsigned start, unsigned end) { char **matches; /* don't bother to resort to filename completion */ tinyrl_completion_over(this); /* perform the matching */ matches = tinyrl_completion(this, line, start, end, clish_shell_tinyrl_word_generator); return matches; } /*-------------------------------------------------------- */ static void clish_shell_tinyrl_init(tinyrl_t * this) { bool_t status; /* bind the '?' key to the help function */ status = tinyrl_bind_key(this, '?', clish_shell_tinyrl_key_help); assert(BOOL_TRUE == status); /* bind the key to the help function */ status = tinyrl_bind_key(this, '\r', clish_shell_tinyrl_key_enter); assert(BOOL_TRUE == status); status = tinyrl_bind_key(this, '\n', clish_shell_tinyrl_key_enter); assert(BOOL_TRUE == status); /* bind the key to auto-complete if necessary */ status = tinyrl_bind_key(this, ' ', clish_shell_tinyrl_key_space); assert(BOOL_TRUE == status); } /*-------------------------------------------------------- */ /* * Create an instance of the specialised class */ tinyrl_t *clish_shell_tinyrl_new(FILE * istream, FILE * ostream, unsigned stifle) { /* call the parent constructor */ tinyrl_t *this = tinyrl_new(istream, ostream, stifle, clish_shell_tinyrl_completion); if (NULL != this) { /* now call our own constructor */ clish_shell_tinyrl_init(this); } return this; } /*-------------------------------------------------------- */ void clish_shell_tinyrl_fini(tinyrl_t * this) { /* nothing to do... yet */ this = this; } /*-------------------------------------------------------- */ void clish_shell_tinyrl_delete(tinyrl_t * this) { /* call our destructor */ clish_shell_tinyrl_fini(this); /* and call the parent destructor */ tinyrl_delete(this); } /*-------------------------------------------------------- */ bool_t clish_shell_line(clish_shell_t * this, const char *prompt, const clish_command_t ** cmd, clish_pargv_t ** pargv, const char *str) { char *line = NULL; bool_t result = BOOL_FALSE; context_t context; /* set up the context */ context.command = NULL; context.pargv = NULL; context.shell = this; if (SHELL_STATE_CLOSING != this->state) { this->state = SHELL_STATE_READY; if (str) line = tinyrl_forceline(this->tinyrl, prompt, &context, str); else line = tinyrl_readline(this->tinyrl, prompt, &context); if (NULL != line) { tinyrl_history_t *history = tinyrl__get_history(this->tinyrl); if (tinyrl__get_isatty(this->tinyrl)) { /* deal with the history list */ tinyrl_history_add(history, line); } if (this->client_hooks->cmd_line_fn) { /* now let the client know the command line has been entered */ this->client_hooks->cmd_line_fn(this, line); } free(line); result = BOOL_TRUE; *cmd = context.command; *pargv = context.pargv; } } return result; } /*-------------------------------------------------------- */ bool_t clish_shell_execline(clish_shell_t *this, const char *line, char ** out) { const clish_command_t *cmd; char *prompt = NULL; clish_pargv_t *pargv = NULL; const clish_view_t *view; bool_t running = BOOL_TRUE; assert(this); if (!line && !tinyrl__get_istream(this->tinyrl)) return BOOL_FALSE; /* obtain the prompt */ view = clish_shell__get_view(this); assert(view); prompt = clish_view__get_prompt(view, clish_shell__get_viewid(this)); assert(prompt); if (line) { /* push the specified line */ running = clish_shell_line(this, prompt, &cmd, &pargv, line); } else { running = clish_shell_line(this, prompt, &cmd, &pargv, NULL); } lub_string_free(prompt); /* execute the provided command */ if (running && cmd && pargv) { if (BOOL_FALSE == clish_shell_execute(this, cmd, pargv, out)) { if((!this->current_file && line) || (this->current_file && this->current_file->stop_on_error)) { this->state = SHELL_STATE_SCRIPT_ERROR; } } } if (NULL != pargv) clish_pargv_delete(pargv); return running; } /*-------------------------------------------------------- */ bool_t clish_shell_forceline(clish_shell_t *this, const char *line, char ** out) { return clish_shell_execline(this, line, out); } /*-------------------------------------------------------- */ bool_t clish_shell_readline(clish_shell_t *this, char ** out) { return clish_shell_execline(this, NULL, out); } /*-------------------------------------------------------- */ FILE * clish_shell__get_istream(const clish_shell_t * this) { return tinyrl__get_istream(this->tinyrl); } /*-------------------------------------------------------- */ FILE * clish_shell__get_ostream(const clish_shell_t * this) { return tinyrl__get_ostream(this->tinyrl); } /*-------------------------------------------------------- */ void clish_shell__set_interactive(clish_shell_t * this, bool_t interactive) { assert(this); this->interactive = interactive; } /*-------------------------------------------------------- */ bool_t clish_shell__get_interactive(const clish_shell_t * this) { assert(this); return this->interactive; } /*-------------------------------------------------------- */ bool_t clish_shell__get_utf8(const clish_shell_t * this) { assert(this); return tinyrl__get_utf8(this->tinyrl); } /*-------------------------------------------------------- */ void clish_shell__set_utf8(clish_shell_t * this, bool_t utf8) { assert(this); tinyrl__set_utf8(this->tinyrl, utf8); } /*-------------------------------------------------------- */