#define _GNU_SOURCE /* getopt_long */ #include #include #include #include #include #define VERSION "1.0.0" void print_usage(FILE *output) { fprintf(output, "Usage:\n"); fprintf(output, "rpt [-n | --count COUNT] [-f | --force] COMMAND\n"); fprintf(output, "rpt [-h | --help]\n"); fprintf(output, "rpt [-V | --version]\n"); } void print_help() { puts("Repeat : repeat a shell command"); puts(""); print_usage(stdout); } void print_version() { puts(VERSION); } /* Exit whenever a subprocess fails. */ void exit_on_error(int status) { if (!status) return; fprintf(stderr, "Process exited with status %d, aborting.\n", status); exit(1); } /* Ignore subprocesses' errors. */ void continue_on_error(int status) { (void) status; } int main(int argc, char* argv[]) { struct option const opts[] = { {"count", required_argument, NULL, 'n'}, {"force", no_argument, NULL, 'f'}, {"version", no_argument, NULL, 'V'}, {"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0} }; opterr = 0; /* dismiss getopt error message */ int opt; char *strcount = NULL; long count = 1; void (*handle_exit_f)(int status) = exit_on_error; while ((opt = getopt_long(argc, argv, "+:n:fVh", opts, NULL)) >= 0) { switch(opt) { case 'n': strcount = optarg; break; case 'f': handle_exit_f = continue_on_error; break; case 'h': print_help(); return 0; case 'V': print_version(); return 0; case ':': fprintf(stderr, "COUNT value missing.\n"); print_usage(stderr); return 1; default: print_usage(stderr); return 1; } } if (strcount) { char *err = NULL; count = strtol(strcount, &err, 10); if (err && (*err != '\0')) { fprintf(stderr, "Failed to read COUNT : '%s'.\n", strcount); return 2; } } return 0; }