Skip to content

Commit 697e9fa

Browse files
committed
[#2788] exhaust options before throwing error
Prior to this change, if parseArgs() was called twice during the same program lifetime and it stumbled on an unsupported option and throwed an exception on the first call, the previous set of arguments lived on to be parsed by the second call. This is a situation that likely arises only in unit tests, but let us fix it properly to at least silence the unit test failure on alpine, which was happening because of different implementation of getopt from musl, and which motivated looking into how getopt behaves. To make the bug evident even in a non-alpine environment, add an EXPECT_THROW_MSG in DStubControllerTest.commandLineArgs when parsing argv3, and see that it outputs "unsupported option: [s]" instead of "extraneous command line information".
1 parent f6bf9f9 commit 697e9fa

File tree

5 files changed

+24
-12
lines changed

5 files changed

+24
-12
lines changed

src/bin/agent/tests/ca_controller_unittests.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -216,7 +216,7 @@ TEST_F(CtrlAgentControllerTest, commandLineArgs) {
216216
char* argv2[] = { const_cast<char*>("progName"),
217217
const_cast<char*>("-x") };
218218
argc = 2;
219-
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: [x] ");
219+
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: -x");
220220
}
221221

222222
// Tests application process creation and initialization.

src/bin/d2/tests/d2_controller_unittests.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@ TEST_F(D2ControllerTest, commandLineArgs) {
132132
char* argv2[] = { const_cast<char*>("progName"),
133133
const_cast<char*>("-x") };
134134
argc = 2;
135-
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: [x] ");
135+
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: -x");
136136
}
137137

138138
/// @brief Tests application process creation and initialization.

src/bin/netconf/tests/netconf_controller_unittests.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -131,7 +131,7 @@ TEST_F(NetconfControllerTest, commandLineArgs) {
131131
char* argv2[] = { const_cast<char*>("progName"),
132132
const_cast<char*>("-x") };
133133
argc = 2;
134-
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: [x] ");
134+
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: -x");
135135
}
136136

137137
// Tests application process creation and initialization.

src/lib/process/d_controller.cc

+19-7
Original file line numberDiff line numberDiff line change
@@ -293,21 +293,33 @@ DControllerBase::parseArgs(int argc, char* argv[]) {
293293
break;
294294

295295
case '?': {
296+
char const saved_optopt(optopt);
297+
std::string const saved_optarg(optarg ? optarg : std::string());
298+
299+
// Exhaust all remaining options in case parseArgs() is called again.
300+
while (getopt(argc, argv, opts.c_str()) != -1) {
301+
}
302+
296303
// We hit an invalid option.
297-
isc_throw(InvalidUsage, "unsupported option: ["
298-
<< static_cast<char>(optopt) << "] "
299-
<< (!optarg ? "" : optarg));
304+
isc_throw(InvalidUsage, "unsupported option: -" << saved_optopt <<
305+
(saved_optarg.empty() ? std::string() : " " + saved_optarg));
300306

301307
break;
302308
}
303309

304310
default:
305311
// We hit a valid custom option
306312
if (!customOption(ch, optarg)) {
307-
// This would be a programmatic error.
308-
isc_throw(InvalidUsage, " Option listed but implemented?: ["
309-
<< static_cast<char>(ch) << "] "
310-
<< (!optarg ? "" : optarg));
313+
char const saved_optopt(optopt);
314+
std::string const saved_optarg(optarg ? optarg : std::string());
315+
316+
// Exhaust all remaining options in case parseArgs() is called again.
317+
while (getopt(argc, argv, opts.c_str()) != -1) {
318+
}
319+
320+
// We hit an invalid option.
321+
isc_throw(InvalidUsage, "unsupported option: -" << saved_optopt <<
322+
(saved_optarg.empty() ? std::string() : " " + saved_optarg));
311323
}
312324
break;
313325
}

src/lib/process/tests/d_controller_unittests.cc

+2-2
Original file line numberDiff line numberDiff line change
@@ -99,14 +99,14 @@ TEST_F(DStubControllerTest, commandLineArgs) {
9999
char* argv2[] = { const_cast<char*>("progName"),
100100
const_cast<char*>("-bs") };
101101
argc = 2;
102-
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: [b] ");
102+
EXPECT_THROW_MSG(parseArgs(argc, argv2), InvalidUsage, "unsupported option: -b");
103103

104104
// Verify that extraneous information is detected.
105105
char* argv3[] = { const_cast<char*>("progName"),
106106
const_cast<char*>("extra"),
107107
const_cast<char*>("information") };
108108
argc = 3;
109-
EXPECT_THROW_MSG(parseArgs(argc, argv3), InvalidUsage, "unsupported option: [s] ");
109+
EXPECT_THROW_MSG(parseArgs(argc, argv3), InvalidUsage, "extraneous command line information");
110110
}
111111

112112
/// @brief Tests application process creation and initialization.

0 commit comments

Comments
 (0)