/* * src/utils.c Utilities * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation version 2.1 * of the License. * * Copyright (c) 2003-2009 Thomas Graf <tgraf@suug.ch> */ /** * @defgroup cli Command Line Interface API * * @{ * * These modules provide an interface for text based applications. The * functions provided are wrappers for their libnl equivalent with * added error handling. The functions check for allocation failures, * invalid input, and unknown types and will print error messages * accordingly via nl_cli_fatal(). */ #include <netlink/cli/utils.h> /** * Parse a text based 32 bit unsigned integer argument * @arg arg Integer in text form. * * Tries to convert the number provided in arg to a uint32_t. Will call * nl_cli_fatal() if the conversion fails. * * @return 32bit unsigned integer. */ uint32_t nl_cli_parse_u32(const char *arg) { unsigned long lval; char *endptr; lval = strtoul(arg, &endptr, 0); if (endptr == arg || lval == ULONG_MAX) nl_cli_fatal(EINVAL, "Unable to parse \"%s\", not a number.", arg); return (uint32_t) lval; } void nl_cli_print_version(void) { printf("libnl tools version %s\n", LIBNL_VERSION); printf( "Copyright (C) 2003-2010 Thomas Graf <tgraf@redhat.com>\n" "\n" "This program comes with ABSOLUTELY NO WARRANTY. This is free \n" "software, and you are welcome to redistribute it under certain\n" "conditions. See the GNU General Public License for details.\n" ); exit(0); } /** * Print error message and quit application * @arg err Error code. * @arg fmt Error message. * * Prints the formatted error message to stderr and quits the application * using the provided error code. */ void nl_cli_fatal(int err, const char *fmt, ...) { va_list ap; char buf[256]; fprintf(stderr, "Error: "); if (fmt) { va_start(ap, fmt); vfprintf(stderr, fmt, ap); va_end(ap); fprintf(stderr, "\n"); } else fprintf(stderr, "%s\n", strerror_r(err, buf, sizeof(buf))); exit(abs(err)); } int nl_cli_connect(struct nl_sock *sk, int protocol) { int err; if ((err = nl_connect(sk, protocol)) < 0) nl_cli_fatal(err, "Unable to connect netlink socket: %s", nl_geterror(err)); return err; } struct nl_sock *nl_cli_alloc_socket(void) { struct nl_sock *sock; if (!(sock = nl_socket_alloc())) nl_cli_fatal(ENOBUFS, "Unable to allocate netlink socket"); return sock; } struct nl_addr *nl_cli_addr_parse(const char *str, int family) { struct nl_addr *addr; int err; if ((err = nl_addr_parse(str, family, &addr)) < 0) nl_cli_fatal(err, "Unable to parse address \"%s\": %s", str, nl_geterror(err)); return addr; } int nl_cli_parse_dumptype(const char *str) { if (!strcasecmp(str, "brief")) return NL_DUMP_LINE; else if (!strcasecmp(str, "details") || !strcasecmp(str, "detailed")) return NL_DUMP_DETAILS; else if (!strcasecmp(str, "stats")) return NL_DUMP_STATS; else nl_cli_fatal(EINVAL, "Invalid dump type \"%s\".\n", str); return 0; } int nl_cli_confirm(struct nl_object *obj, struct nl_dump_params *params, int default_yes) { nl_object_dump(obj, params); for (;;) { char buf[32] = { 0 }; int answer; printf("Delete? (%c/%c) ", default_yes ? 'Y' : 'y', default_yes ? 'n' : 'N'); if (!fgets(buf, sizeof(buf), stdin)) { fprintf(stderr, "Error while reading\n."); continue; } switch ((answer = tolower(buf[0]))) { case '\n': answer = default_yes ? 'y' : 'n'; case 'y': case 'n': return answer == 'y'; } fprintf(stderr, "Invalid input, try again.\n"); } return 0; } struct nl_cache *nl_cli_alloc_cache(struct nl_sock *sock, const char *name, int (*ac)(struct nl_sock *, struct nl_cache **)) { struct nl_cache *cache; int err; if ((err = ac(sock, &cache)) < 0) nl_cli_fatal(err, "Unable to allocate %s cache: %s", name, nl_geterror(err)); nl_cache_mngt_provide(cache); return cache; } void nl_cli_load_module(const char *prefix, const char *name) { char path[FILENAME_MAX+1]; void *handle; snprintf(path, sizeof(path), "%s/%s/%s.so", PKGLIBDIR, prefix, name); if (!(handle = dlopen(path, RTLD_NOW))) nl_cli_fatal(ENOENT, "Unable to load module \"%s\": %s\n", path, dlerror()); } /** @} */