#include <tap/basic.h>
+#define FLAG_OFF 0
+#define FIRST_OFF 1
+#define SECOND_OFF 2
+#define FOURTH_OFF 4
+#define FIFTH_OFF 5
+#define PERHAPS_OFF 6
+#define SANITY_OFF 7
+
static int
testproc(struct cmd_syndesc *as, void *rock)
{
- is_string("foo", as->parms[0].items->data,
+ is_string("foo", as->parms[FIRST_OFF].items->data,
"first option matches");
- is_string("bar", as->parms[1].items->data,
+ is_string("bar", as->parms[SECOND_OFF].items->data,
"second option matches");
- ok(as->parms[2].items != NULL,
+ ok(as->parms[FLAG_OFF].items != NULL,
"flag is set");
return 0;
{
char *tv[100];
struct cmd_syndesc *opts;
+ struct cmd_syndesc *retopts;
int code;
int tc;
+ int retval;
+ char *retstring = NULL;
- plan(29);
+ plan(79);
initialize_CMD_error_table();
opts = cmd_CreateSyntax(NULL, testproc, NULL, NULL);
+ cmd_AddParm(opts, "-flag", CMD_FLAG, CMD_OPTIONAL, "a flag");
cmd_AddParm(opts, "-first", CMD_SINGLE, CMD_REQUIRED, "first option");
cmd_AddParm(opts, "-second", CMD_LIST, CMD_OPTIONAL, "second option");
- cmd_AddParm(opts, "-flag", CMD_FLAG, CMD_OPTIONAL, "a flag");
/* A simple command line */
code = cmd_ParseLine("-first foo -second bar -flag", tv, &tc, 100);
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(0, code, "dispatching simple comamnd line succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "parsing simple command line succeeds");
+ is_string("foo", retopts->parms[FIRST_OFF].items->data, " ... 1st option matches");
+ is_string("bar", retopts->parms[SECOND_OFF].items->data, " ... 2nd option matches");
+ ok(retopts->parms[FLAG_OFF].items != NULL, " ... 3rd option matches");
+ cmd_FreeOptions(&retopts);
cmd_FreeArgv(tv);
/* unknown switch */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_UNKNOWNSWITCH, code, "invalid options fail as expected");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_UNKNOWNSWITCH, code, "and still fail with cmd_Parse");
cmd_FreeArgv(tv);
/* missing parameter */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_TOOFEW, code, "missing parameters fail as expected");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_TOOFEW, code, "and still fail with cmd_Parse");
cmd_FreeArgv(tv);
/* missing option */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_UNKNOWNSWITCH, code, "missing options fail as expected");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_UNKNOWNSWITCH, code, "and still fail with cmd_Parse");
cmd_FreeArgv(tv);
code = cmd_ParseLine("-first foo baz -second bar -third -flag", tv, &tc, 100);
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_NOTLIST, code, "too many parameters fails as expected");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_NOTLIST, code, "and still fail with cmd_Parse");
cmd_FreeArgv(tv);
/* Positional parameters */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(0, code, "dispatching positional parameters succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "and works with cmd_Parse");
+ cmd_FreeOptions(&retopts);
cmd_FreeArgv(tv);
/* Abbreviations */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(0, code, "dispatching abbreviations succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "and works with cmd_Parse");
+ cmd_FreeOptions(&retopts);
+
cmd_FreeArgv(tv);
/* Ambiguous */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_UNKNOWNSWITCH, code, "ambiguous abbreviations correctly fail");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_UNKNOWNSWITCH, code, "and fail with cmd_Parse too");
cmd_FreeArgv(tv);
/* Disable positional commands */
code = cmd_ParseLine("foo bar -flag", tv, &tc, 100);
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
- is_int(3359746, code, "positional parameters can be disabled");
+ is_int(CMD_NOTLIST, code, "positional parameters can be disabled");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_NOTLIST, code, "and fail with cmd_Parse too");
cmd_FreeArgv(tv);
/* Disable abbreviations */
is_int(0, code, "cmd_ParseLine succeeds");
code = cmd_Dispatch(tc, tv);
is_int(CMD_UNKNOWNSWITCH, code, "dispatching abbreviations succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_UNKNOWNSWITCH, code, "and fail with cmd_Parse too");
+
+ cmd_FreeArgv(tv);
+
+ /* Try the new cmd_Parse function with something different*/
+ code = cmd_ParseLine("-first one -second two -flag", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "Parsing with cmd_Parse works");
+ is_string("one", retopts->parms[FIRST_OFF].items->data, " ... 1st option matches");
+ is_string("two", retopts->parms[SECOND_OFF].items->data, " ... 2nd option matches");
+ ok(retopts->parms[FLAG_OFF].items != NULL, " ... 3rd option matches");
+
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+ /* Try adding a couple of parameters at specific positions */
+ cmd_AddParmAtOffset(opts, FIFTH_OFF, "-fifth", CMD_SINGLE, CMD_OPTIONAL,
+ "fifth option");
+ cmd_AddParmAtOffset(opts, FOURTH_OFF, "-fourth", CMD_SINGLE, CMD_OPTIONAL,
+ "fourth option" );
+ code = cmd_ParseLine("-first a -fourth b -fifth c", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "parsing our new options succeeds");
+ is_string("b", retopts->parms[FOURTH_OFF].items->data, " Fourth option in right place");
+ is_string("c", retopts->parms[FIFTH_OFF].items->data, " Fifth option in right place");
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+
+ /* Check Accessors */
+ code = cmd_ParseLine("-first 1 -second second -flag", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+
+ code = cmd_OptionAsInt(retopts, FIRST_OFF, &retval);
+ is_int(0, code, "cmd_OptionsAsInt succeeds");
+ is_int(1, retval, " ... and returns correct value");
+
+ code = cmd_OptionAsString(retopts, SECOND_OFF, &retstring);
+ is_int(0, code, "cmd_OptionsAsString succeeds");
+ is_string("second", retstring, " ... and returns correct value");
+ free(retstring);
+ retstring = NULL;
+
+ code = cmd_OptionAsFlag(retopts, FLAG_OFF, &retval);
+ is_int(0, code, "cmd_OptionsAsFlag succeeds");
+ ok(retval, " ... and flag is correct");
+
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+
+ /* Add an alias */
+ code = cmd_AddParmAlias(opts, SECOND_OFF, "-twa");
+ is_int(0, code, "cmd_AddParmAlias succeeds");
+
+ code = cmd_ParseLine("-first 1 -twa tup", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "cmd_Parse succeeds for alias");
+ cmd_OptionAsString(retopts, SECOND_OFF, &retstring);
+ is_string("tup", retstring, " ... and we have the correct value");
+ free(retstring);
+ retstring = NULL;
+
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+
+ /* Add something that can be a flag or a value, and put something after
+ * it so we can check for parse problems*/
+ cmd_AddParm(opts, "-perhaps", CMD_SINGLE_OR_FLAG, CMD_OPTIONAL,
+ "what am I");
+ cmd_AddParm(opts, "-sanity", CMD_SINGLE, CMD_OPTIONAL, "sanity check");
+
+ /* Try using as an option */
+
+ code = cmd_ParseLine("-first 1 -perhaps 2 -sanity 3", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "cmd_Parse succeeds for option-as-flag as opt");
+ code = cmd_OptionAsInt(retopts, PERHAPS_OFF, &retval);
+ is_int(0, code, "cmd_OptionAsInt succeeds");
+ is_int(2, retval, " ... and we have the correct value");
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+
+ /* And now, as a flag */
+
+ code = cmd_ParseLine("-first 1 -perhaps -sanity 3", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "cmd_Parse succeeds for option-as-flag as flag");
+ code = cmd_OptionAsInt(retopts, PERHAPS_OFF, &retval);
+ is_int(CMD_MISSING, code, " ... pulling out a value fails as expected");
+ cmd_OptionAsFlag(retopts, PERHAPS_OFF, &retval);
+ ok(retval, " ... but parsing as a flag works");
+ cmd_FreeOptions(&retopts);
+ cmd_FreeArgv(tv);
+
+ /* Check that we can produce help output */
+ code = cmd_ParseLine("-help", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(CMD_USAGE, code, "cmd_Parse returns usage error with help output");
+ ok(retopts == NULL, " ... and options is empty");
+
+ /* Check splitting with '=' */
+
+ code = cmd_ParseLine("-first 1 -perhaps=6 -sanity=3", tv, &tc, 100);
+ is_int(0, code, "cmd_ParseLine succeeds");
+ code = cmd_Parse(tc, tv, &retopts);
+ is_int(0, code, "cmd_Parse succeeds for items split with '='");
+ code = cmd_OptionAsInt(retopts, PERHAPS_OFF, &retval);
+ is_int(0, code, "cmd_OptionAsInt succeeds");
+ is_int(6, retval, " ... and we have the correct value once");
+ code = cmd_OptionAsInt(retopts, SANITY_OFF, &retval);
+ is_int(0, code, "cmd_OptionAsInt succeeds");
+ is_int(3, retval, " ... and we have the correct value twice");
+ cmd_FreeOptions(&retopts);
cmd_FreeArgv(tv);
return 0;