[BACK]Return to bgplgsh.c CVS log [TXT][DIR] Up to [local] / src / usr.bin / bgplg

Annotation of src/usr.bin/bgplg/bgplgsh.c, Revision 1.8

1.8     ! mmcc        1: /*     $OpenBSD: bgplgsh.c,v 1.7 2015/01/16 06:40:06 deraadt Exp $     */
1.1       reyk        2:
                      3: /*
1.4       reyk        4:  * Copyright (c) 2005, 2006 Reyk Floeter <reyk@openbsd.org>
1.1       reyk        5:  *
                      6:  * Permission to use, copy, modify, and distribute this software for any
                      7:  * purpose with or without fee is hereby granted, provided that the above
                      8:  * copyright notice and this permission notice appear in all copies.
                      9:  *
                     10:  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
                     11:  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
                     12:  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
                     13:  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
                     14:  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
                     15:  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
                     16:  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
                     17:  */
                     18:
1.7       deraadt    19: #include <sys/types.h>
1.1       reyk       20: #include <sys/stat.h>
                     21:
                     22: #include <stdio.h>
                     23: #include <stdlib.h>
                     24: #include <signal.h>
                     25: #include <string.h>
                     26: #include <unistd.h>
1.7       deraadt    27: #include <limits.h>
1.1       reyk       28: #include <ctype.h>
                     29: #include <errno.h>
                     30: #include <fcntl.h>
                     31:
                     32: #include <readline/readline.h>
                     33: #include <readline/history.h>
                     34:
                     35: #include "bgplg.h"
                     36:
1.5       florian    37: #define BGPDSOCK       "/var/www/run/bgpd.rsock"
1.1       reyk       38: #define BGPCTL         "/usr/sbin/bgpctl", "-s", BGPDSOCK
                     39: #define PING           "/sbin/ping"
                     40: #define TRACEROUTE     "/usr/sbin/traceroute"
1.3       sthen      41: #define PING6          "/sbin/ping6"
                     42: #define TRACEROUTE6    "/usr/sbin/traceroute6"
1.1       reyk       43:
                     44: static volatile int quit;
                     45:
                     46: static struct cmd cmds[] = CMDS;
                     47:
                     48: char           **lg_arg2argv(char *, int *);
                     49: char           **lg_argextra(char **, int, int, struct cmd *);
                     50: int              lg_checkarg(char *);
                     51: int              lg_checkcmd(int, char **, int *, struct cmd *);
                     52: char            *lg_completion(const char *, int);
                     53:
                     54: int
                     55: lg_checkarg(char *arg)
                     56: {
                     57:        size_t len;
                     58:        u_int i;
                     59:
                     60:        if (!(len = strlen(arg)))
                     61:                return (0);
                     62:
                     63: #define allowed_in_string(_x)                                           \
1.6       deraadt    64:        ((isalnum((unsigned char)_x) || isprint((unsigned char)_x)) &&  \
1.2       reyk       65:        (_x != '%' && _x != '\\' && _x != ';' && _x != '&' && _x != '|'))
1.1       reyk       66:
                     67:        for (i = 0; i < len; i++) {
                     68:                if (!allowed_in_string(arg[i])) {
                     69:                        fprintf(stderr, "invalid character in input\n");
                     70:                        return (EPERM);
                     71:                }
                     72:        }
1.6       deraadt    73: #undef allowed_in_string
1.1       reyk       74:        return (0);
                     75: }
                     76:
                     77: char **
                     78: lg_arg2argv(char *arg, int *argc)
                     79: {
                     80:        char **argv, *ptr = arg;
                     81:        size_t len;
                     82:        u_int i, c = 1;
                     83:
                     84:        if (lg_checkarg(arg) != 0)
                     85:                return (NULL);
                     86:        if (!(len = strlen(arg)))
                     87:                return (NULL);
                     88:
                     89:        /* Count elements */
                     90:        for (i = 0; i < len; i++) {
1.6       deraadt    91:                if (isspace((unsigned char)arg[i])) {
1.1       reyk       92:                        /* filter out additional options */
                     93:                        if (arg[i + 1] == '-') {
                     94:                                printf("invalid input\n");
                     95:                                return (NULL);
                     96:                        }
                     97:                        arg[i] = '\0';
                     98:                        c++;
                     99:                }
                    100:        }
                    101:        if (arg[0] == '\0')
                    102:                return (NULL);
                    103:
                    104:        /* Generate array */
                    105:        if ((argv = calloc(c + 1, sizeof(char *))) == NULL) {
                    106:                printf("fatal error: %s\n", strerror(errno));
                    107:                return (NULL);
                    108:        }
                    109:
                    110:        argv[c] = NULL;
                    111:        *argc = c;
                    112:
                    113:        /* Fill array */
                    114:        for (i = c = 0; i < len; i++) {
                    115:                if (arg[i] == '\0' || i == 0) {
                    116:                        if (i != 0)
                    117:                                ptr = &arg[i + 1];
                    118:                        argv[c++] = ptr;
                    119:                }
                    120:        }
                    121:
                    122:        return (argv);
                    123: }
                    124:
                    125: char **
                    126: lg_argextra(char **argv, int argc, int off, struct cmd *cmdp)
                    127: {
                    128:        char **new_argv;
                    129:        int i, c = 0, n;
                    130:
                    131:        if ((n = argc - off) < 0)
                    132:                return (NULL);
                    133:
                    134:        /* Count elements */
                    135:        for (i = 0; cmdp->earg[i] != NULL; i++)
                    136:                c++;
                    137:
                    138:        /* Generate array */
                    139:        if ((new_argv = calloc(c + n + 1, sizeof(char *))) == NULL) {
                    140:                printf("fatal error: %s\n", strerror(errno));
                    141:                return (NULL);
                    142:        }
                    143:
                    144:        /* Fill array */
                    145:        for (i = c = 0; cmdp->earg[i] != NULL; i++)
                    146:                new_argv[c++] = cmdp->earg[i];
                    147:
                    148:        /* Append old array */
                    149:        for (i = n; i < argc; i++)
                    150:                new_argv[c++] = argv[i];
                    151:
                    152:        new_argv[c] = NULL;
                    153:
1.8     ! mmcc      154:        free(argv);
1.1       reyk      155:
                    156:        return (new_argv);
                    157: }
                    158:
                    159: int
                    160: lg_checkcmd(int argc, char **argv, int *off, struct cmd *cmd)
                    161: {
                    162:        char **cmdp = NULL, *cmdstr = NULL;
                    163:        int i, ncmd, v, ret = -1;
                    164:
                    165:        if ((cmdstr = strdup(cmd->name)) == NULL)
                    166:                goto done;
                    167:        if ((cmdp = lg_arg2argv(cmdstr, &ncmd)) == NULL)
                    168:                goto done;
                    169:        if (ncmd > argc || argc > (ncmd + cmd->maxargs))
                    170:                goto done;
                    171:
                    172:        for (i = 0; i < ncmd; i++)
                    173:                if (strcmp(argv[i], cmdp[i]) != 0)
                    174:                        goto done;
                    175:
                    176:        if ((v = argc - ncmd) < 0 ||
                    177:            (*off != -1 && *off < v))
                    178:                goto done;
                    179:        if (cmd->minargs && v < cmd->minargs) {
                    180:                ret = EINVAL;
                    181:                goto done;
                    182:        }
                    183:        *off = v;
                    184:        ret = 0;
                    185:
                    186:  done:
1.8     ! mmcc      187:        free(cmdp);
        !           188:        free(cmdstr);
1.1       reyk      189:        return (ret);
                    190: }
                    191:
                    192: char *
                    193: lg_completion(const char *str, int state)
                    194: {
                    195:        static int lg_complidx, len;
                    196:        const char *name;
                    197:
                    198:        if (state == 0) {
                    199:                len = strlen(str);
                    200:                lg_complidx = 0;
                    201:        }
                    202:        while ((name = cmds[lg_complidx].name) != NULL) {
                    203:                lg_complidx++;
                    204:                if (strncmp(name, str, len) == 0)
                    205:                        return (strdup(name));
                    206:        }
                    207:
                    208:        return (NULL);
                    209: }
                    210:
                    211: int
                    212: main(void)
                    213: {
                    214:        struct cmd *cmd = NULL;
1.7       deraadt   215:        char prompt[HOST_NAME_MAX+1], *line, **argp = NULL;
1.1       reyk      216:        int ncmd, ret, v = -1;
                    217:        u_int i;
                    218:
                    219:        rl_readline_name = NAME;
                    220:        rl_completion_entry_function = lg_completion;
                    221:
                    222:        /* Ignore the whitespace character */
                    223:        rl_basic_word_break_characters = "\t\n\"\\'`@$><=;|&{(";
                    224:
                    225:        while (!quit) {
                    226:                v = -1;
                    227:                gethostname(prompt, sizeof(prompt) - 2);
                    228:                strlcat(prompt, "> ", sizeof(prompt));
                    229:
                    230:                if ((line = readline(prompt)) == NULL) {
                    231:                        printf("\n");
                    232:                        lg_help(cmds, NULL);
                    233:                        continue;
                    234:                }
                    235:                if (!lg_strip(line))
                    236:                        goto next;
                    237:                if (strcmp(line, "exit") == 0) {
                    238:                        quit = 1;
                    239:                        goto next;
                    240:                }
                    241:
                    242:                add_history(line);
                    243:
                    244:                if ((argp = lg_arg2argv(line, &ncmd)) == NULL)
                    245:                        goto next;
                    246:
                    247:                for (i = 0; cmds[i].name != NULL; i++) {
                    248:                        ret = lg_checkcmd(ncmd, argp, &v, &cmds[i]);
                    249:                        if (ret == 0)
                    250:                                cmd = &cmds[i];
                    251:                        else if (ret == EINVAL) {
                    252:                                printf("invalid number of arguments\n");
                    253:                                goto next;
                    254:                        }
                    255:                }
                    256:
                    257:                if (cmd == NULL) {
                    258:                        printf("invalid command\n");
                    259:                } else if (cmd->func != NULL) {
                    260:                        cmd->func(cmds, argp);
                    261:                } else {
                    262:                        if ((argp = lg_argextra(argp, ncmd, v, cmd)) == NULL)
                    263:                                goto next;
                    264:                        lg_exec(cmd->earg[0], argp);
                    265:                }
                    266:
                    267:  next:
1.8     ! mmcc      268:                free(argp);
        !           269:                argp = NULL;
        !           270:                free(line);
        !           271:                line = NULL;
1.1       reyk      272:                cmd = NULL;
                    273:        }
                    274:
                    275:        return (0);
                    276: }