/**
* Registered commands.
*/
-command_t cmds[CMD_MAX];
+command_t cmds[MAX_COMMANDS];
+
+/**
+ * active command.
+ */
+static int active = 0;
+
+/**
+ * number of registered commands
+ */
+static int registered = 0;
+
+/**
+ * help command index
+ */
+static int help_idx;
/**
* Global options used by all subcommands
*/
-struct option command_opts[CMD_MAX > MAX_OPTIONS ?: MAX_OPTIONS];
+struct option command_opts[MAX_COMMANDS > MAX_OPTIONS ?: MAX_OPTIONS];
/**
- * Build long_opts for a specific command
+ * Global optstring used by all subcommands
*/
-static void build_opts(command_type_t cmd)
+char command_optstring[(MAX_COMMANDS > MAX_OPTIONS ?: MAX_OPTIONS) * 3];
+
+/**
+ * Build command_opts/command_optstr for the active command
+ */
+static void build_opts()
{
- int i;
+ int i, pos = 0;
memset(command_opts, 0, sizeof(command_opts));
- if (cmd == CMD_HELP)
+ memset(command_optstring, 0, sizeof(command_optstring));
+ if (active == help_idx)
{
- for (i = 0; i < CMD_MAX; i++)
+ for (i = 0; cmds[i].cmd; i++)
{
command_opts[i].name = cmds[i].cmd;
command_opts[i].val = cmds[i].op;
+ command_optstring[i] = cmds[i].op;
}
}
else
{
- for (i = 0; cmds[cmd].options[i].name; i++)
+ for (i = 0; cmds[active].options[i].name; i++)
{
- command_opts[i].name = cmds[cmd].options[i].name;
- command_opts[i].has_arg = cmds[cmd].options[i].arg;
- command_opts[i].val = cmds[cmd].options[i].op;
+ command_opts[i].name = cmds[active].options[i].name;
+ command_opts[i].has_arg = cmds[active].options[i].arg;
+ command_opts[i].val = cmds[active].options[i].op;
+ command_optstring[pos++] = cmds[active].options[i].op;
+ switch (cmds[active].options[i].arg)
+ {
+ case optional_argument:
+ command_optstring[pos++] = ':';
+ /* FALL */
+ case required_argument:
+ command_optstring[pos++] = ':';
+ /* FALL */
+ case no_argument:
+ default:
+ break;
+ }
}
}
}
/**
* Register a command
*/
-void command_register(command_type_t type, command_t command)
+void command_register(command_t command)
{
- cmds[type] = command;
+ cmds[registered++] = command;
}
/**
* Print usage text, with an optional error
*/
-int command_usage(command_type_t cmd, char *error)
+int command_usage(char *error)
{
FILE *out = stdout;
+ char buf[64];
int i;
if (error)
}
fprintf(out, "strongSwan %s PKI tool\n", VERSION);
fprintf(out, "usage:\n");
- if (cmd == CMD_HELP)
+ if (active == help_idx)
{
- for (i = 0; i < CMD_MAX; i++)
+ for (i = 0; cmds[i].cmd; i++)
{
- fprintf(out, " pki --%-6s %s\n", cmds[i].cmd, cmds[i].description);
+ snprintf(buf, sizeof(buf), "--%s (-%c)", cmds[i].cmd, cmds[i].op);
+ fprintf(out, " pki %-14s %s\n", buf, cmds[i].description);
}
}
else
{
- for (i = 0; cmds[cmd].line[i]; i++)
+ for (i = 0; cmds[active].line[i]; i++)
{
if (i == 0)
{
- fprintf(out, " pki --%s %s\n", cmds[cmd].cmd, cmds[cmd].line[i]);
+ fprintf(out, " pki --%s %s\n",
+ cmds[active].cmd, cmds[active].line[i]);
}
else
{
- fprintf(out, " %s\n", cmds[cmd].line[i]);
+ fprintf(out, " %s\n", cmds[active].line[i]);
}
}
- for (i = 0; cmds[cmd].options[i].name; i++)
+ for (i = 0; cmds[active].options[i].name; i++)
{
- fprintf(out, " --%-8s %s\n",
- cmds[cmd].options[i].name, cmds[cmd].options[i].desc);
+ snprintf(buf, sizeof(buf), "--%s (-%c)",
+ cmds[active].options[i].name, cmds[active].options[i].op);
+ fprintf(out, " %-15s %s\n",
+ buf, cmds[active].options[i].desc);
}
}
return error != NULL;
}
-
/**
* Show usage information
*/
static int help(int argc, char *argv[])
{
- return command_usage(CMD_HELP, NULL);
+ return command_usage(NULL);
}
/**
{
int op, i;
- command_register(CMD_HELP, (command_t) {
- help, 'h', "help", "show usage information"});
- build_opts(CMD_HELP);
- op = getopt_long(argc, argv, "", command_opts, NULL);
- for (i = 0; i < CMD_MAX; i++)
+ active = help_idx = registered;
+ command_register((command_t){help, 'h', "help", "show usage information"});
+
+ build_opts();
+ op = getopt_long(argc, argv, command_optstring, command_opts, NULL);
+ for (i = 0; cmds[i].cmd; i++)
{
if (cmds[i].op == op)
{
- build_opts(i);
+ active = i;
+ build_opts();
return cmds[i].call(argc, argv);
}
}
- return command_usage(CMD_HELP, "invalid operation");
+ return command_usage("invalid operation");
}