/* esl_subcmd : utilities for command line programs that take subcommands * * Extends esl_getopts to more complicated programs with subcommands. * * See also: * esl_getopts : command line argument parsing */ #include "esl_config.h" #include #include #include "easel.h" #include "esl_getopts.h" #include "esl_subcmd.h" /* Function: esl_subcmd_CreateDefaultApp() * Synopsis: Process cmdline options for a subcommand * * Purpose: For a subcommand of main program , with * subcommand options , process a command line * /; return a new object on * success. * * If there's a problem with the user's command line, print * informative message and `exit(1)`. If the subcommand * options include `-h`, print help and `exit(0)`. * * is one structure from the main * application's table of subcommands. is a * string, usually the same as . * is a table of for the subcommand. * * If is a path, e.g. "/foo/bar/easel", the * leading path is removed, leaving just "easel" as the * name of the main program. * * Args: topcmd - name of the main command, e.g. "easel" (= argv[0], probably) * sub - ESL_SUBCMD table entry for this subcommand * suboptions - ESL_OPTIONS table for this subcommand * argc - number of args on the commandline * argv - array of args on the commandline * * Returns: ptr to new object * * Throws: NULL on allocation failures. */ ESL_GETOPTS * esl_subcmd_CreateDefaultApp(const char *topcmd, const ESL_SUBCMD *sub, const ESL_OPTIONS *suboptions, int argc, char **argv) { ESL_GETOPTS *go = esl_getopts_Create(suboptions); char *lastslash = strrchr(topcmd, '/'); if (lastslash) topcmd = lastslash+1; if (esl_opt_ProcessCmdline(go, argc, argv) != eslOK || esl_opt_VerifyConfig(go) != eslOK) { if ( esl_printf("Failed to parse command line: %s\n", go->errbuf) != eslOK) goto ERROR; if ( esl_printf("Usage:\n %s %s %s\n", topcmd, sub->subcmd, sub->usage) != eslOK) goto ERROR; if ( esl_printf("\nTo see more help on available options, do `%s %s -h`\n\n", topcmd, sub->subcmd) != eslOK) goto ERROR; exit(1); } if (esl_opt_GetBoolean(go, "-h") == TRUE) { if ( esl_printf("%s %s :: %s\n", topcmd, sub->subcmd, sub->description) != eslOK) goto ERROR; if ( esl_printf("\nUsage:\n %s %s %s\n", topcmd, sub->subcmd, sub->usage) != eslOK) goto ERROR; if ( esl_printf("\nOptions:\n") != eslOK) goto ERROR; if ( esl_opt_DisplayHelp(stdout, go, 0, 2, 80) != eslOK) goto ERROR; exit(0); } if (esl_opt_ArgNumber(go) != sub->nargs) { if ( esl_printf("Incorrect number of command line arguments.") != eslOK) goto ERROR; if ( esl_printf("Usage:\n %s %s %s\n", topcmd, sub->subcmd, sub->usage) != eslOK) goto ERROR; if ( esl_printf("\nTo see more help on available options, do `%s %s -h`\n\n", topcmd, sub->subcmd) != eslOK) goto ERROR; exit(1); } return go; ERROR: esl_getopts_Destroy(go); return NULL; }