diff --git a/libdap-app-cli b/libdap-app-cli deleted file mode 160000 index 8c21d84db9762ba3a5c0699599d9f98823bc280b..0000000000000000000000000000000000000000 --- a/libdap-app-cli +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 8c21d84db9762ba3a5c0699599d9f98823bc280b diff --git a/libdap-app-cli/CMakeLists.txt b/libdap-app-cli/CMakeLists.txt new file mode 100644 index 0000000000000000000000000000000000000000..7ca7367a4742324d8576c9a8914d649989bde1a5 --- /dev/null +++ b/libdap-app-cli/CMakeLists.txt @@ -0,0 +1,24 @@ +cmake_minimum_required(VERSION 2.8) +project (dap_app_cli) + +set(DAP_APP_CLI_SRCS + src/dap_app_cli.c + src/dap_app_cli_net.c + src/dap_app_cli_shell.c + ) + +set(DAP_APP_CLI_HEADERS + include/dap_app_cli.h + include/dap_app_cli_net.h + include/dap_app_cli_shell.h + ) + + +if(WIN32) + #include_directories(../3rdparty/curl/include/) +endif() + +add_library(${PROJECT_NAME} STATIC ${DAP_APP_CLI_SRCS} ${DAP_APP_CLI_HEADERS} ) + +target_link_libraries(${PROJECT_NAME} dap_core dap_chain_net m) +target_include_directories(${PROJECT_NAME} PUBLIC include/ ) diff --git a/libdap-app-cli/README.md b/libdap-app-cli/README.md new file mode 100644 index 0000000000000000000000000000000000000000..6de2a0b76165680b8fb3ece91b3012bc18a1312f --- /dev/null +++ b/libdap-app-cli/README.md @@ -0,0 +1,2 @@ +# libdap-chain-cli + diff --git a/libdap-app-cli/include/dap_app_cli.h b/libdap-app-cli/include/dap_app_cli.h new file mode 100644 index 0000000000000000000000000000000000000000..944bb9c96ca46a3d0e3276dd07813e12a5ac7c98 --- /dev/null +++ b/libdap-app-cli/include/dap_app_cli.h @@ -0,0 +1,53 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <kahovski@gmail.com> + * DeM Labs Inc. https://demlabs.net + * DeM Labs Open source community https://github.com/demlabsinc + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + DAP is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ +#pragma once + +#include <stdint.h> +#include <stddef.h> + +// command description +typedef struct dap_app_cli_cmd_state { + char *cmd_name; + char **cmd_param; + int cmd_param_count; + int ret_code; + // for reply + char *cmd_res; + size_t cmd_res_len; + size_t cmd_res_cur; +} dap_app_cli_cmd_state_t; + +#ifdef __cplusplus +extern "C" { +#endif +/** + * Clear and delete memory of structure cmd_state + */ +void dap_app_cli_free_cmd_state(dap_app_cli_cmd_state_t *cmd); + +int dap_app_cli_main(const char * a_app_name, const char * a_socket_path, int argc, char **argv); + +#ifdef __cplusplus +} +#endif diff --git a/libdap-app-cli/include/dap_app_cli_net.h b/libdap-app-cli/include/dap_app_cli_net.h new file mode 100755 index 0000000000000000000000000000000000000000..e22056f9bfb17ddd9d264f41967764c86efc848a --- /dev/null +++ b/libdap-app-cli/include/dap_app_cli_net.h @@ -0,0 +1,53 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <kahovski@gmail.com> + * DeM Labs Inc. https://demlabs.net + * DeM Labs Open source community https://github.com/demlabsinc + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + DAP is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ + +#pragma once + +#include "dap_app_cli.h" +#include "dap_events_socket.h" + +#define DAP_CLI_HTTP_RESPONSE_SIZE_MAX 8192 +#define DAP_CLI_HTTP_TIMEOUT 10 // seconds +#define DAP_CLI_ERROR_FORMAT -1 +#define DAP_CLI_ERROR_TIMEOUT -2 +#define DAP_CLI_ERROR_SOCKET -3 + +// connection description +typedef uint64_t dap_app_cli_connect_param_t; + +/** + * Connect to node unix socket server + * + * return struct connect_param if connect established, else NULL + */ +dap_app_cli_connect_param_t* dap_app_cli_connect(const char * a_socket_path); + +/** + * Send request to kelvin-node + * + * return 0 if OK, else error code + */ +int dap_app_cli_post_command(dap_app_cli_connect_param_t *socket, dap_app_cli_cmd_state_t *cmd); + +int dap_app_cli_disconnect(dap_app_cli_connect_param_t *socket); diff --git a/libdap-app-cli/include/dap_app_cli_shell.h b/libdap-app-cli/include/dap_app_cli_shell.h new file mode 100644 index 0000000000000000000000000000000000000000..5fdfeb4a5bb6bffad4649bf8d4467e8b3df0ae2f --- /dev/null +++ b/libdap-app-cli/include/dap_app_cli_shell.h @@ -0,0 +1,53 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <kahovski@gmail.com> + * DeM Labs Inc. https://demlabs.net + * DeM Labs Open source community https://github.com/demlabsinc + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + DAP is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ + +#pragma once + +#ifndef whitespace +#define whitespace(c) (((c) == ' ') || ((c) == '\t')) +#endif + +/* + * Initialize readline (and terminal if not already). + */ +int rl_initialize(void); + +/** + * Strip whitespace from the start and end of STRING. Return a pointer into STRING. + */ +char * rl_stripwhite(char *string); + +/** + * Read a line of input. Prompt with PROMPT. An empty PROMPT means none. + * A return value of NULL means that EOF was encountered. + */ +char *rl_readline(const char *prompt); + +/** + * Place STRING at the end of the history list. + */ +void add_history(const char *string); + +int parse_shell_options(char **argv, int arg_start, int arg_end); + diff --git a/libdap-app-cli/src/dap_app_cli.c b/libdap-app-cli/src/dap_app_cli.c new file mode 100644 index 0000000000000000000000000000000000000000..923467b1cc275836a86071bd78335f423136139e --- /dev/null +++ b/libdap-app-cli/src/dap_app_cli.c @@ -0,0 +1,218 @@ +/* + * Authors: + * Dmitriy A. Gearasimov <kahovski@gmail.com> + * DeM Labs Inc. https://demlabs.net + * DeM Labs Open source community https://github.com/demlabsinc + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + DAP is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <stddef.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +//#include "dap_client.h" +#include "dap_common.h" +#include "dap_file_utils.h" +#include "dap_strfuncs.h" +#include "dap_chain_node_cli.h" +#include "dap_app_cli.h" +#include "dap_app_cli_net.h" +#include "dap_app_cli_shell.h" + + +#ifdef _WIN32 +#include "registry.h" +#endif + + +/** + * split string to argc and argv + */ +static char** split_word(char *line, int *argc) +{ + if(!line) + { + if(argc) + *argc = 0; + return NULL ; + } + char **argv = calloc(sizeof(char*), strlen(line)); + int n = 0; + char *s, *start = line; + size_t len = strlen(line); + for(s = line; s <= line + len; s++) { + if(whitespace(*s)) { + *s = '\0'; + argv[n] = start; + s++; + // miss spaces + for(; whitespace(*s); s++) + ; + start = s; + n++; + } + } + // last param + if(len) { + argv[n] = start; + n++; + } + if(argc) + *argc = n; + return argv; +} + +/* + * Execute a command line. + */ +int execute_line(dap_app_cli_connect_param_t *cparam, char *line) +{ + register int i; + dap_chain_node_cmd_item_t *command; + char *word; + + /* Isolate the command word. */ + i = 0; + while(line[i] && whitespace(line[i])) + i++; + word = line + i; + + int argc = 0; + char **argv = split_word(word, &argc); + + // Call the function + if(argc > 0) { + dap_app_cli_cmd_state_t cmd; + memset(&cmd, 0, sizeof(dap_app_cli_cmd_state_t)); + cmd.cmd_name = (char *) argv[0]; + cmd.cmd_param_count = argc - 1; + if(cmd.cmd_param_count > 0) + cmd.cmd_param = (char**) (argv + 1); + // Send command + int res = dap_app_cli_post_command(cparam, &cmd); + return res; + } + fprintf(stderr, "No command\n"); + return -1; +} + +/** + * Clear and delete memory of structure cmd_state + */ +void dap_app_cli_free_cmd_state(dap_app_cli_cmd_state_t *cmd) { + if(!cmd->cmd_param) + return; + for(int i = 0; i < cmd->cmd_param_count; i++) + { + DAP_DELETE(cmd->cmd_param[i]); + } + DAP_DELETE(cmd->cmd_res); + DAP_DELETE(cmd); +} + +/** + * Read and execute commands until EOF is reached. This assumes that + * the input source has already been initialized. + */ +int shell_reader_loop(dap_app_cli_connect_param_t *cparam) +{ + char *line, *s; + + rl_initialize(); /* Bind our completer. */ + int done = 0; + // Loop reading and executing lines until the user quits. + for(; done == 0;) { + // Read a line of input + line = rl_readline("> "); + + if(!line) + break; + + /* Remove leading and trailing whitespace from the line. + Then, if there is anything left, add it to the history list + and execute it. */ + s = rl_stripwhite(line); + + if(*s) + { + add_history(s); + execute_line(cparam, s); + } + + DAP_DELETE(line); + } + + return 0; +} + + +/** + * @brief dap_app_cli_main + * @param argc + * @param argv + * @return + */ +int dap_app_cli_main(const char * a_app_name, const char * a_socket_path, int a_argc, char **a_argv) +{ + dap_set_appname(a_app_name); + if (dap_common_init(dap_get_appname(), NULL) != 0) { + printf("Fatal Error: Can't init common functions module"); + return -2; + } + + dap_log_level_set(L_CRITICAL); +#ifdef _WIN32 + WSADATA wsaData; + WSAStartup(MAKEWORD(2,2), &wsaData); +#endif + // connect to node + dap_app_cli_connect_param_t *cparam = dap_app_cli_connect( a_socket_path ); + if(!cparam) + { + printf("Can't connect to %s on socket %s\n",dap_get_appname(), a_socket_path); + exit(-1); + } + + if(a_argc > 1){ + // Call the function + dap_app_cli_cmd_state_t cmd; + memset(&cmd, 0, sizeof(dap_app_cli_cmd_state_t)); + cmd.cmd_name = strdup(a_argv[1]); + cmd.cmd_param_count = a_argc - 2; + if(cmd.cmd_param_count > 0) + cmd.cmd_param = (char**) (a_argv + 2); + // Send command + int res = dap_app_cli_post_command(cparam, &cmd); + dap_app_cli_disconnect(cparam); +#ifdef _WIN32 + WSACleanup(); +#endif + return res; + }else{ + // command not found, start interactive shell + shell_reader_loop(cparam); + dap_app_cli_disconnect(cparam); + } +#ifdef _WIN32 + WSACleanup(); +#endif + return 0; +} + diff --git a/libdap-app-cli/src/dap_app_cli_net.c b/libdap-app-cli/src/dap_app_cli_net.c new file mode 100755 index 0000000000000000000000000000000000000000..2187f7ad2c4f7df47430395f7934fa9f67191dce --- /dev/null +++ b/libdap-app-cli/src/dap_app_cli_net.c @@ -0,0 +1,220 @@ +/* + * Authors: + * Dmitriy A. Gerasimov <kahovski@gmail.com> + * Alexander Lysikov <alexander.lysikov@demlabs.net> + * DeM Labs Inc. https://demlabs.net + * DeM Labs Open source community https://github.com/demlabsinc + * Copyright (c) 2017-2019 + * All rights reserved. + + This file is part of DAP (Deus Applications Prototypes) the open source project + + DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + DAP is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with any DAP based project. If not, see <http://www.gnu.org/licenses/>. + */ + +//#include <dap_client.h> + +#include <stdlib.h> +#include <sys/types.h> +#include <assert.h> +#include <errno.h> + +#ifdef _WIN32 +#include <winsock2.h> +#include <windows.h> +#include <mswsock.h> +#include <ws2tcpip.h> +#include <io.h> +#else +#include <sys/socket.h> +#include <sys/un.h> +#endif + +#include "dap_common.h" +#include "dap_string.h" +#include "dap_strfuncs.h" +#include "dap_chain_node_cli.h" // for UNIX_SOCKET_FILE +#include "dap_app_cli.h" +#include "dap_app_cli_net.h" + +static int s_status; + +//callback function to receive http data +static void dap_app_cli_http_read(uint64_t *socket, dap_app_cli_cmd_state_t *l_cmd) +{ + size_t l_recv_len = recv(*socket, &l_cmd->cmd_res[l_cmd->cmd_res_cur], DAP_CLI_HTTP_RESPONSE_SIZE_MAX, 0); + if (l_recv_len == -1) { + s_status = DAP_CLI_ERROR_SOCKET; + return; + } + l_cmd->cmd_res_cur += l_recv_len; + switch (s_status) { + case 1: { // Find content length + const char *l_cont_len_str = "Content-Length: "; + char *l_str_ptr = strstr(l_cmd->cmd_res, l_cont_len_str); + if (l_str_ptr && strstr(l_str_ptr, "\r\n")) { + l_cmd->cmd_res_len = atoi(l_str_ptr + strlen(l_cont_len_str)); + if (l_cmd->cmd_res_len == 0) { + s_status = DAP_CLI_ERROR_FORMAT; + } else { + s_status++; + } + } else { + break; + } + } + case 2: { // Find header end and throw out header + const char *l_head_end_str = "\r\n\r\n"; + char *l_str_ptr = strstr(l_cmd->cmd_res, l_head_end_str); + if (l_str_ptr) { + l_str_ptr += strlen(l_head_end_str); + size_t l_head_size = l_str_ptr - l_cmd->cmd_res; + strncpy(l_cmd->cmd_res, l_str_ptr, l_cmd->cmd_res_cur - l_head_size); + l_cmd->cmd_res_cur -= l_head_size; + s_status++; + } else { + break; + } + } + default: + case 3: { // Complete command reply + if (l_cmd->cmd_res_cur == l_cmd->cmd_res_len) { + l_cmd->cmd_res[l_cmd->cmd_res_cur] = 0; + s_status = 0; + } + } break; + } +} + +/** + * @brief dap_app_cli_connect + * @details Connect to node unix socket server + * @param a_socket_path + * @return if connect established, else NULL + */ +dap_app_cli_connect_param_t* dap_app_cli_connect(const char *a_socket_path) +{ + // set socket param + int buffsize = DAP_CLI_HTTP_RESPONSE_SIZE_MAX; +#ifdef WIN32 + // TODO connect to the named pipe "\\\\.\\pipe\\node_cli.pipe" + uint16_t l_cli_port = dap_config_get_item_uint16 ( g_config, "conserver", "listen_port_tcp"); + if (!l_cli_port) + return NULL; + SOCKET l_socket = socket(AF_INET, SOCK_STREAM, 0); + setsockopt((SOCKET)l_socket, SOL_SOCKET, SO_SNDBUF, (char *)&buffsize, sizeof(int) ); + setsockopt((SOCKET)l_socket, SOL_SOCKET, SO_RCVBUF, (char *)&buffsize, sizeof(int) ); +#else + if (!a_socket_path) { + return NULL; + } + // create socket + int l_socket = socket(AF_UNIX, SOCK_STREAM, 0); + if (l_socket < 0) { + return NULL; + } + setsockopt(l_socket, SOL_SOCKET, SO_SNDBUF, (void*) &buffsize, sizeof(buffsize)); + setsockopt(l_socket, SOL_SOCKET, SO_RCVBUF, (void*) &buffsize, sizeof(buffsize)); +#endif + // connect + int l_addr_len; +#ifdef WIN32 + struct sockaddr_in l_remote_addr; + l_remote_addr.sin_family = AF_INET; + IN_ADDR _in_addr = { { .S_addr = htonl(INADDR_LOOPBACK) } }; + l_remote_addr.sin_addr = _in_addr; + l_remote_addr.sin_port = l_cli_port; + l_addr_len = sizeof(struct sockaddr_in); +#else + struct sockaddr_un l_remote_addr; + l_remote_addr.sun_family = AF_UNIX; + strcpy(l_remote_addr.sun_path, a_socket_path); + l_addr_len = SUN_LEN(&l_remote_addr); +#endif + if (connect(l_socket, (struct sockaddr *)&l_remote_addr, l_addr_len) == SOCKET_ERROR) { +#ifdef __WIN32 + _set_errno(WSAGetLastError()); +#endif + printf("Socket connection err: %d\n", errno); + closesocket(l_socket); + return NULL; + } + uint64_t *l_ret = DAP_NEW(uint64_t); + *l_ret = l_socket; + return l_ret; +} + +/** + * Send request to kelvin-node + * + * return 0 if OK, else error code + */ +int dap_app_cli_post_command( dap_app_cli_connect_param_t *a_socket, dap_app_cli_cmd_state_t *a_cmd ) +{ + if(!a_socket || !a_cmd || !a_cmd->cmd_name) { + assert(0); + return -1; + } + s_status = 1; + a_cmd->cmd_res = DAP_NEW_Z_SIZE(char, DAP_CLI_HTTP_RESPONSE_SIZE_MAX); + a_cmd->cmd_res_cur = 0; + dap_string_t *l_cmd_data = dap_string_new(a_cmd->cmd_name); + if (a_cmd->cmd_param) { + for (int i = 0; i < a_cmd->cmd_param_count; i++) { + if (a_cmd->cmd_param[i]) { + dap_string_append(l_cmd_data, "\r\n"); + dap_string_append(l_cmd_data, a_cmd->cmd_param[i]); + } + } + } + dap_string_append(l_cmd_data, "\r\n\r\n"); + dap_string_t *l_post_data = dap_string_new(""); + dap_string_printf(l_post_data, "POST /connect HTTP/1.1\r\n" + "Host: localhost\r\n" + "Content-Type: text/text\r\n" + "Content-Length: %d\r\n" + "\r\n" + "%s", l_cmd_data->len, l_cmd_data->str); + send(*a_socket, l_post_data->str, l_post_data->len, 0); + + //wait for command execution + time_t l_start_time = time(NULL); + while(s_status > 0) { + dap_app_cli_http_read(a_socket, a_cmd); + if (time(NULL) - l_start_time > DAP_CLI_HTTP_TIMEOUT) + return DAP_CLI_ERROR_TIMEOUT; + } + // process result + if (a_cmd->cmd_res && !s_status) { + char **l_str = dap_strsplit(a_cmd->cmd_res, "\r\n", 1); + int l_cnt = dap_str_countv(l_str); + char *l_str_reply = NULL; + if (l_cnt == 2) { + long l_err_code = strtol(l_str[0], NULL, 10); + l_str_reply = l_str[1]; + } + printf("%s\n", (l_str_reply) ? l_str_reply : "no response"); + dap_strfreev(l_str); + } + DAP_DELETE(a_cmd->cmd_res); + dap_string_free(l_cmd_data, true); + dap_string_free(l_post_data, true); + return s_status; +} + +int dap_app_cli_disconnect(dap_app_cli_connect_param_t *a_socket) +{ + closesocket(*a_socket); + DAP_DELETE(a_socket); +} diff --git a/libdap-app-cli/src/dap_app_cli_shell.c b/libdap-app-cli/src/dap_app_cli_shell.c new file mode 100644 index 0000000000000000000000000000000000000000..cefbdf196927b8ac323b8a8a475fda318e6a0368 --- /dev/null +++ b/libdap-app-cli/src/dap_app_cli_shell.c @@ -0,0 +1,355 @@ + +#include <stdlib.h> +#include <stdio.h> +#include <time.h> +#include <stdlib.h> +#include <stddef.h> +#include <stdint.h> +#include <unistd.h> +#include <sys/types.h> +#include <getopt.h> +#include <signal.h> +#include <string.h> +#include <assert.h> +#include <setjmp.h> +#include <locale.h> + +#ifdef _WIN32 +#include <winsock2.h> +#include <windows.h> +#include <mswsock.h> +#include <ws2tcpip.h> +#include <io.h> +#include <pthread.h> +#else +#include <sys/ttydefaults.h> +#endif + +#include "dap_common.h" +#include "dap_app_cli.h" +#include "dap_app_cli_shell.h" + +//#include "posixjmp.h" + +#ifndef savestring +#define savestring(x) strcpy ((char *)malloc (1 + strlen (x)), (x)) +#endif + +typedef void rl_voidfunc_t(void); +typedef void rl_vintfunc_t(int); + +/* Current prompt. */ +char *rl_prompt = (char *) NULL; +int rl_visible_prompt_length = 0; + +/* Non-zero means we have been called at least once before. */ +static int rl_initialized; + +/* The stuff that gets printed out before the actual text of the line. + This is usually pointing to rl_prompt. */ +char *rl_display_prompt = (char *) NULL; + +/* Non-zero makes this the next keystroke to read. */ +int rl_pending_input = 0; + +/* Make this non-zero to return the current input_line. */ +int rl_done; +/* Non-zero if the previous command was a kill command. */ +int _rl_last_command_was_kill = 0; + +/* Top level environment for readline_internal (). */ +jmp_buf _rl_top_level; + +/* Length of the current input line. */ +int rl_end; + +/* The character that can generate an EOF. Really read from + the terminal driver... just defaulted here. */ +//int _rl_eof_char = CTRL('D'); + +#define NEWLINE '\n' + +/* Input error; can be returned by (*rl_getc_function) if readline is reading + a top-level command (RL_ISSTATE (RL_STATE_READCMD)). */ +#define READERR (-2) + +/* Possible state values for rl_readline_state */ +#define RL_STATE_NONE 0x000000 /* no state; before first call */ + +#define RL_STATE_INITIALIZING 0x0000001 /* initializing */ +#define RL_STATE_INITIALIZED 0x0000002 /* initialization done */ +#define RL_STATE_READCMD 0x0000008 /* reading a command key */ +#define RL_STATE_INPUTPENDING 0x0020000 /* rl_execute_next called */ +#define RL_STATE_TERMPREPPED 0x0000004 /* terminal is prepped */ + +/* Flags word encapsulating the current readline state. */ +unsigned long rl_readline_state = RL_STATE_NONE; + +#define RL_SETSTATE(x) (rl_readline_state |= (x)) +#define RL_UNSETSTATE(x) (rl_readline_state &= ~(x)) +#define RL_ISSTATE(x) (rl_readline_state & (x)) + +/* The names of the streams that we do input and output to. */ +FILE *rl_instream = (FILE *) NULL; +FILE *rl_outstream = (FILE *) NULL; + +/** + * Read one symbol + */ +unsigned char rl_getc(FILE *stream) +{ + int result; + unsigned char c; + + while(1) + { + +#if defined (__MINGW32__) + if (isatty (fileno (stream))) + return (_getch ()); /* "There is no error return." */ +#endif + result = 0; + if(result >= 0) + result = read(fileno(stream), &c, sizeof(unsigned char)); + if(result == sizeof(unsigned char)) + return (c); + + /* If zero characters are returned, then the file that we are + reading from is empty! Return EOF in that case. */ + if(result == 0) + return (EOF); + } +} + +/** + * Set up the prompt and expand it. Called from readline() + */ +int rl_set_prompt(const char *prompt) +{ + free(rl_prompt); + rl_prompt = prompt ? savestring(prompt) : (char *) NULL; + rl_display_prompt = rl_prompt ? rl_prompt : ""; + fprintf(stdout, "%s", prompt); + fflush(stdout); + //rl_visible_prompt_length = rl_expand_prompt (rl_prompt); + return 0; +} + +/** + * Read a line of input. Prompt with PROMPT. An empty PROMPT means none. + * A return value of NULL means that EOF was encountered. + */ +char *rl_readline(const char *prompt) +{ + int value_size = 3, value_len = 0; + char *value = DAP_NEW_Z_SIZE(char, value_size + 1); + + // Set up the prompt + rl_set_prompt(prompt); + + // Read a line of input from the global rl_instream, doing output on the global rl_outstream. + while(1) + { + unsigned char c = rl_getc(rl_instream); + + if(c == EOF || c == NEWLINE) + break; + value[value_len] = c; + value_len++; + if(value_len == value_size) { + value_size += 32; + value = realloc(value, value_size + 1); + } + } + return (value); +} + +static char* _rl_get_locale_var(const char *v) +{ + char *lspec; + + lspec = getenv("LC_ALL"); + if(lspec == 0 || *lspec == 0) + lspec = getenv(v); + if(lspec == 0 || *lspec == 0) + lspec = getenv("LANG"); + + return lspec; +} + +/* + * Query the right environment variables and call setlocale() to initialize + * the C library locale settings. + */ +static char* _rl_init_locale(void) +{ + char *ret, *lspec; + + /* Set the LC_CTYPE locale category from environment variables. */ + lspec = _rl_get_locale_var("LC_CTYPE"); + /* Since _rl_get_locale_var queries the right environment variables, + we query the current locale settings with setlocale(), and, if + that doesn't return anything, we set lspec to the empty string to + force the subsequent call to setlocale() to define the `native' + environment. */ + if(lspec == 0 || *lspec == 0) + lspec = setlocale(LC_CTYPE, (char *) NULL); + if(lspec == 0) + lspec = ""; + ret = setlocale(LC_CTYPE, lspec); /* ok, since it does not change locale */ + + //_rl_utf8locale = (ret && *ret) ? utf8locale (ret) : 0; + + return ret; +} + +/* + * Initialize readline (and terminal if not already). + */ +int rl_initialize(void) +{ + /* If we have never been called before, initialize the + terminal and data structures. */ + if(rl_initialized == 0) + { + RL_SETSTATE(RL_STATE_INITIALIZING); + rl_instream = (FILE *) stdin; + rl_outstream = (FILE *) stdout; + RL_UNSETSTATE(RL_STATE_INITIALIZING); + rl_initialized++; + RL_SETSTATE(RL_STATE_INITIALIZED); + } + else + (void) _rl_init_locale(); /* check current locale */ + RL_SETSTATE(RL_STATE_INITIALIZING); + rl_instream = (FILE *) stdin; + rl_outstream = (FILE *) stdout; + RL_UNSETSTATE(RL_STATE_INITIALIZING); + return 0; +} + +int parse_shell_options(char **argv, int arg_start, int arg_end) +{ + int arg_index; + int arg_character, on_or_off, next_arg, i; + char *o_option, *arg_string; + + arg_index = arg_start; + while(arg_index != arg_end && (arg_string = argv[arg_index]) && + (*arg_string == '-' || *arg_string == '+')) + { + /* There are flag arguments, so parse them. */ + next_arg = arg_index + 1; + + /* A single `-' signals the end of options. From the 4.3 BSD sh. + An option `--' means the same thing; this is the standard + getopt(3) meaning. */ + if(arg_string[0] == '-' && + (arg_string[1] == '\0' || + (arg_string[1] == '-' && arg_string[2] == '\0'))) + return (next_arg); + + i = 1; + on_or_off = arg_string[0]; + while(arg_character = arg_string[i++]) + { + switch (arg_character) + { + case 'c': + //want_pending_command = 1; + break; + + case 'l': + //make_login_shell = 1; + break; + + case 's': + //read_from_stdin = 1; + break; + + case 'o': + o_option = argv[next_arg]; + if(o_option == 0) + { + //list_minus_o_opts(-1, (on_or_off == '-') ? 0 : 1); + break; + } + //if(set_minus_o_option(on_or_off, o_option) != EXECUTION_SUCCESS) + // exit(EX_BADUSAGE); + next_arg++; + break; + + case 'O': + /* Since some of these can be overridden by the normal + interactive/non-interactive shell initialization or + initializing posix mode, we save the options and process + them after initialization. */ + o_option = argv[next_arg]; + if(o_option == 0) + { + //shopt_listopt(o_option, (on_or_off == '-') ? 0 : 1); + break; + } + //add_shopt_to_alist(o_option, on_or_off); + next_arg++; + break; + + case 'D': + //dump_translatable_strings = 1; + break; + + default: + break; +// if(change_flag(arg_character, on_or_off) == FLAG_ERROR) +// { +// report_error(_("%c%c: invalid option"), on_or_off, arg_character); +// show_shell_usage(stderr, 0); +// exit(EX_BADUSAGE); +// } + } + } + /* Can't do just a simple increment anymore -- what about + "bash -abouo emacs ignoreeof -hP"? */ + arg_index = next_arg; + } + + return (arg_index); +} + +/** + * Strip whitespace from the start and end of STRING. Return a pointer into STRING. + */ +char * rl_stripwhite(char *string) +{ + register char *s, *t; + + for(s = string; whitespace(*s); s++) + ; + + if(*s == 0) + return (s); + + t = s + strlen(s) - 1; + while(t > s && whitespace(*t)) + t--; + *++t = '\0'; + + return s; +} + +/* The structure used to store a history entry. */ +typedef struct _hist_entry { + char *line; + char *timestamp; /* char * rather than time_t for read/write */ + char *data; +} HIST_ENTRY; + +/** + * Place STRING at the end of the history list. + */ +void add_history(const char *string) +{ + HIST_ENTRY *temp; + // The data field is set to NULL + // TODO +}