1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "base/command_line.h"
6
7 #include <memory>
8 #include <string>
9 #include <vector>
10
11 #include "base/files/file_path.h"
12 #include "base/macros.h"
13 #include "base/strings/utf_string_conversions.h"
14 #include "build/build_config.h"
15 #include "testing/gtest/include/gtest/gtest.h"
16
17 namespace base {
18
19 // To test Windows quoting behavior, we use a string that has some backslashes
20 // and quotes.
21 // Consider the command-line argument: q\"bs1\bs2\\bs3q\\\"
22 // Here it is with C-style escapes.
23 static const CommandLine::StringType kTrickyQuoted =
24 FILE_PATH_LITERAL("q\\\"bs1\\bs2\\\\bs3q\\\\\\\"");
25 // It should be parsed by Windows as: q"bs1\bs2\\bs3q\"
26 // Here that is with C-style escapes.
27 static const CommandLine::StringType kTricky =
28 FILE_PATH_LITERAL("q\"bs1\\bs2\\\\bs3q\\\"");
29
TEST(CommandLineTest,CommandLineConstructor)30 TEST(CommandLineTest, CommandLineConstructor) {
31 const CommandLine::CharType* argv[] = {
32 FILE_PATH_LITERAL("program"),
33 FILE_PATH_LITERAL("--foo="),
34 FILE_PATH_LITERAL("-bAr"),
35 FILE_PATH_LITERAL("-spaetzel=pierogi"),
36 FILE_PATH_LITERAL("-baz"),
37 FILE_PATH_LITERAL("flim"),
38 FILE_PATH_LITERAL("--other-switches=--dog=canine --cat=feline"),
39 FILE_PATH_LITERAL("-spaetzle=Crepe"),
40 FILE_PATH_LITERAL("-=loosevalue"),
41 FILE_PATH_LITERAL("-"),
42 FILE_PATH_LITERAL("FLAN"),
43 FILE_PATH_LITERAL("a"),
44 FILE_PATH_LITERAL("--input-translation=45--output-rotation"),
45 FILE_PATH_LITERAL("--"),
46 FILE_PATH_LITERAL("--"),
47 FILE_PATH_LITERAL("--not-a-switch"),
48 FILE_PATH_LITERAL("\"in the time of submarines...\""),
49 FILE_PATH_LITERAL("unquoted arg-with-space")};
50 CommandLine cl(arraysize(argv), argv);
51
52 EXPECT_FALSE(cl.GetCommandLineString().empty());
53 EXPECT_FALSE(cl.HasSwitch("cruller"));
54 EXPECT_FALSE(cl.HasSwitch("flim"));
55 EXPECT_FALSE(cl.HasSwitch("program"));
56 EXPECT_FALSE(cl.HasSwitch("dog"));
57 EXPECT_FALSE(cl.HasSwitch("cat"));
58 EXPECT_FALSE(cl.HasSwitch("output-rotation"));
59 EXPECT_FALSE(cl.HasSwitch("not-a-switch"));
60 EXPECT_FALSE(cl.HasSwitch("--"));
61
62 EXPECT_EQ(FilePath(FILE_PATH_LITERAL("program")).value(),
63 cl.GetProgram().value());
64
65 EXPECT_TRUE(cl.HasSwitch("foo"));
66 #if defined(OS_WIN)
67 EXPECT_TRUE(cl.HasSwitch("bar"));
68 #else
69 EXPECT_FALSE(cl.HasSwitch("bar"));
70 #endif
71 EXPECT_TRUE(cl.HasSwitch("baz"));
72 EXPECT_TRUE(cl.HasSwitch("spaetzle"));
73 EXPECT_TRUE(cl.HasSwitch("other-switches"));
74 EXPECT_TRUE(cl.HasSwitch("input-translation"));
75
76 EXPECT_EQ("Crepe", cl.GetSwitchValueASCII("spaetzle"));
77 EXPECT_EQ("", cl.GetSwitchValueASCII("foo"));
78 EXPECT_EQ("", cl.GetSwitchValueASCII("bar"));
79 EXPECT_EQ("", cl.GetSwitchValueASCII("cruller"));
80 EXPECT_EQ("--dog=canine --cat=feline", cl.GetSwitchValueASCII(
81 "other-switches"));
82 EXPECT_EQ("45--output-rotation", cl.GetSwitchValueASCII("input-translation"));
83
84 const CommandLine::StringVector& args = cl.GetArgs();
85 ASSERT_EQ(8U, args.size());
86
87 std::vector<CommandLine::StringType>::const_iterator iter = args.begin();
88 EXPECT_EQ(FILE_PATH_LITERAL("flim"), *iter);
89 ++iter;
90 EXPECT_EQ(FILE_PATH_LITERAL("-"), *iter);
91 ++iter;
92 EXPECT_EQ(FILE_PATH_LITERAL("FLAN"), *iter);
93 ++iter;
94 EXPECT_EQ(FILE_PATH_LITERAL("a"), *iter);
95 ++iter;
96 EXPECT_EQ(FILE_PATH_LITERAL("--"), *iter);
97 ++iter;
98 EXPECT_EQ(FILE_PATH_LITERAL("--not-a-switch"), *iter);
99 ++iter;
100 EXPECT_EQ(FILE_PATH_LITERAL("\"in the time of submarines...\""), *iter);
101 ++iter;
102 EXPECT_EQ(FILE_PATH_LITERAL("unquoted arg-with-space"), *iter);
103 ++iter;
104 EXPECT_TRUE(iter == args.end());
105 }
106
TEST(CommandLineTest,CommandLineFromString)107 TEST(CommandLineTest, CommandLineFromString) {
108 #if defined(OS_WIN)
109 CommandLine cl = CommandLine::FromString(
110 L"program --foo= -bAr /Spaetzel=pierogi /Baz flim "
111 L"--other-switches=\"--dog=canine --cat=feline\" "
112 L"-spaetzle=Crepe -=loosevalue FLAN "
113 L"--input-translation=\"45\"--output-rotation "
114 L"--quotes=" + kTrickyQuoted + L" "
115 L"-- -- --not-a-switch "
116 L"\"in the time of submarines...\"");
117
118 EXPECT_FALSE(cl.GetCommandLineString().empty());
119 EXPECT_FALSE(cl.HasSwitch("cruller"));
120 EXPECT_FALSE(cl.HasSwitch("flim"));
121 EXPECT_FALSE(cl.HasSwitch("program"));
122 EXPECT_FALSE(cl.HasSwitch("dog"));
123 EXPECT_FALSE(cl.HasSwitch("cat"));
124 EXPECT_FALSE(cl.HasSwitch("output-rotation"));
125 EXPECT_FALSE(cl.HasSwitch("not-a-switch"));
126 EXPECT_FALSE(cl.HasSwitch("--"));
127
128 EXPECT_EQ(FilePath(FILE_PATH_LITERAL("program")).value(),
129 cl.GetProgram().value());
130
131 EXPECT_TRUE(cl.HasSwitch("foo"));
132 EXPECT_TRUE(cl.HasSwitch("bar"));
133 EXPECT_TRUE(cl.HasSwitch("baz"));
134 EXPECT_TRUE(cl.HasSwitch("spaetzle"));
135 EXPECT_TRUE(cl.HasSwitch("other-switches"));
136 EXPECT_TRUE(cl.HasSwitch("input-translation"));
137 EXPECT_TRUE(cl.HasSwitch("quotes"));
138
139 EXPECT_EQ("Crepe", cl.GetSwitchValueASCII("spaetzle"));
140 EXPECT_EQ("", cl.GetSwitchValueASCII("foo"));
141 EXPECT_EQ("", cl.GetSwitchValueASCII("bar"));
142 EXPECT_EQ("", cl.GetSwitchValueASCII("cruller"));
143 EXPECT_EQ("--dog=canine --cat=feline", cl.GetSwitchValueASCII(
144 "other-switches"));
145 EXPECT_EQ("45--output-rotation", cl.GetSwitchValueASCII("input-translation"));
146 EXPECT_EQ(kTricky, cl.GetSwitchValueNative("quotes"));
147
148 const CommandLine::StringVector& args = cl.GetArgs();
149 ASSERT_EQ(5U, args.size());
150
151 std::vector<CommandLine::StringType>::const_iterator iter = args.begin();
152 EXPECT_EQ(FILE_PATH_LITERAL("flim"), *iter);
153 ++iter;
154 EXPECT_EQ(FILE_PATH_LITERAL("FLAN"), *iter);
155 ++iter;
156 EXPECT_EQ(FILE_PATH_LITERAL("--"), *iter);
157 ++iter;
158 EXPECT_EQ(FILE_PATH_LITERAL("--not-a-switch"), *iter);
159 ++iter;
160 EXPECT_EQ(FILE_PATH_LITERAL("in the time of submarines..."), *iter);
161 ++iter;
162 EXPECT_TRUE(iter == args.end());
163
164 // Check that a generated string produces an equivalent command line.
165 CommandLine cl_duplicate = CommandLine::FromString(cl.GetCommandLineString());
166 EXPECT_EQ(cl.GetCommandLineString(), cl_duplicate.GetCommandLineString());
167 #endif
168 }
169
170 // Tests behavior with an empty input string.
TEST(CommandLineTest,EmptyString)171 TEST(CommandLineTest, EmptyString) {
172 #if defined(OS_WIN)
173 CommandLine cl_from_string = CommandLine::FromString(L"");
174 EXPECT_TRUE(cl_from_string.GetCommandLineString().empty());
175 EXPECT_TRUE(cl_from_string.GetProgram().empty());
176 EXPECT_EQ(1U, cl_from_string.argv().size());
177 EXPECT_TRUE(cl_from_string.GetArgs().empty());
178 #endif
179 CommandLine cl_from_argv(0, nullptr);
180 EXPECT_TRUE(cl_from_argv.GetCommandLineString().empty());
181 EXPECT_TRUE(cl_from_argv.GetProgram().empty());
182 EXPECT_EQ(1U, cl_from_argv.argv().size());
183 EXPECT_TRUE(cl_from_argv.GetArgs().empty());
184 }
185
TEST(CommandLineTest,GetArgumentsString)186 TEST(CommandLineTest, GetArgumentsString) {
187 static const FilePath::CharType kPath1[] =
188 FILE_PATH_LITERAL("C:\\Some File\\With Spaces.ggg");
189 static const FilePath::CharType kPath2[] =
190 FILE_PATH_LITERAL("C:\\no\\spaces.ggg");
191
192 static const char kFirstArgName[] = "first-arg";
193 static const char kSecondArgName[] = "arg2";
194 static const char kThirdArgName[] = "arg with space";
195 static const char kFourthArgName[] = "nospace";
196 static const char kFifthArgName[] = "%1";
197
198 CommandLine cl(CommandLine::NO_PROGRAM);
199 cl.AppendSwitchPath(kFirstArgName, FilePath(kPath1));
200 cl.AppendSwitchPath(kSecondArgName, FilePath(kPath2));
201 cl.AppendArg(kThirdArgName);
202 cl.AppendArg(kFourthArgName);
203 cl.AppendArg(kFifthArgName);
204
205 #if defined(OS_WIN)
206 CommandLine::StringType expected_first_arg(UTF8ToUTF16(kFirstArgName));
207 CommandLine::StringType expected_second_arg(UTF8ToUTF16(kSecondArgName));
208 CommandLine::StringType expected_third_arg(UTF8ToUTF16(kThirdArgName));
209 CommandLine::StringType expected_fourth_arg(UTF8ToUTF16(kFourthArgName));
210 CommandLine::StringType expected_fifth_arg(UTF8ToUTF16(kFifthArgName));
211 #elif defined(OS_POSIX) || defined(OS_FUCHSIA)
212 CommandLine::StringType expected_first_arg(kFirstArgName);
213 CommandLine::StringType expected_second_arg(kSecondArgName);
214 CommandLine::StringType expected_third_arg(kThirdArgName);
215 CommandLine::StringType expected_fourth_arg(kFourthArgName);
216 CommandLine::StringType expected_fifth_arg(kFifthArgName);
217 #endif
218
219 #if defined(OS_WIN)
220 #define QUOTE_ON_WIN FILE_PATH_LITERAL("\"")
221 #else
222 #define QUOTE_ON_WIN FILE_PATH_LITERAL("")
223 #endif // OS_WIN
224
225 CommandLine::StringType expected_str;
226 expected_str.append(FILE_PATH_LITERAL("--"))
227 .append(expected_first_arg)
228 .append(FILE_PATH_LITERAL("="))
229 .append(QUOTE_ON_WIN)
230 .append(kPath1)
231 .append(QUOTE_ON_WIN)
232 .append(FILE_PATH_LITERAL(" "))
233 .append(FILE_PATH_LITERAL("--"))
234 .append(expected_second_arg)
235 .append(FILE_PATH_LITERAL("="))
236 .append(QUOTE_ON_WIN)
237 .append(kPath2)
238 .append(QUOTE_ON_WIN)
239 .append(FILE_PATH_LITERAL(" "))
240 .append(QUOTE_ON_WIN)
241 .append(expected_third_arg)
242 .append(QUOTE_ON_WIN)
243 .append(FILE_PATH_LITERAL(" "))
244 .append(expected_fourth_arg)
245 .append(FILE_PATH_LITERAL(" "));
246
247 CommandLine::StringType expected_str_no_quote_placeholders(expected_str);
248 expected_str_no_quote_placeholders.append(expected_fifth_arg);
249 EXPECT_EQ(expected_str_no_quote_placeholders, cl.GetArgumentsString());
250
251 #if defined(OS_WIN)
252 CommandLine::StringType expected_str_quote_placeholders(expected_str);
253 expected_str_quote_placeholders.append(QUOTE_ON_WIN)
254 .append(expected_fifth_arg)
255 .append(QUOTE_ON_WIN);
256 EXPECT_EQ(expected_str_quote_placeholders,
257 cl.GetArgumentsStringWithPlaceholders());
258 #endif
259 }
260
261 // Test methods for appending switches to a command line.
TEST(CommandLineTest,AppendSwitches)262 TEST(CommandLineTest, AppendSwitches) {
263 std::string switch1 = "switch1";
264 std::string switch2 = "switch2";
265 std::string value2 = "value";
266 std::string switch3 = "switch3";
267 std::string value3 = "a value with spaces";
268 std::string switch4 = "switch4";
269 std::string value4 = "\"a value with quotes\"";
270 std::string switch5 = "quotes";
271 CommandLine::StringType value5 = kTricky;
272
273 CommandLine cl(FilePath(FILE_PATH_LITERAL("Program")));
274
275 cl.AppendSwitch(switch1);
276 cl.AppendSwitchASCII(switch2, value2);
277 cl.AppendSwitchASCII(switch3, value3);
278 cl.AppendSwitchASCII(switch4, value4);
279 cl.AppendSwitchASCII(switch5, value4);
280 cl.AppendSwitchNative(switch5, value5);
281
282 EXPECT_TRUE(cl.HasSwitch(switch1));
283 EXPECT_TRUE(cl.HasSwitch(switch2));
284 EXPECT_EQ(value2, cl.GetSwitchValueASCII(switch2));
285 EXPECT_TRUE(cl.HasSwitch(switch3));
286 EXPECT_EQ(value3, cl.GetSwitchValueASCII(switch3));
287 EXPECT_TRUE(cl.HasSwitch(switch4));
288 EXPECT_EQ(value4, cl.GetSwitchValueASCII(switch4));
289 EXPECT_TRUE(cl.HasSwitch(switch5));
290 EXPECT_EQ(value5, cl.GetSwitchValueNative(switch5));
291
292 #if defined(OS_WIN)
293 EXPECT_EQ(L"Program "
294 L"--switch1 "
295 L"--switch2=value "
296 L"--switch3=\"a value with spaces\" "
297 L"--switch4=\"\\\"a value with quotes\\\"\" "
298 // Even though the switches are unique, appending can add repeat
299 // switches to argv.
300 L"--quotes=\"\\\"a value with quotes\\\"\" "
301 L"--quotes=\"" + kTrickyQuoted + L"\"",
302 cl.GetCommandLineString());
303 #endif
304 }
305
TEST(CommandLineTest,AppendSwitchesDashDash)306 TEST(CommandLineTest, AppendSwitchesDashDash) {
307 const CommandLine::CharType* raw_argv[] = { FILE_PATH_LITERAL("prog"),
308 FILE_PATH_LITERAL("--"),
309 FILE_PATH_LITERAL("--arg1") };
310 CommandLine cl(arraysize(raw_argv), raw_argv);
311
312 cl.AppendSwitch("switch1");
313 cl.AppendSwitchASCII("switch2", "foo");
314
315 cl.AppendArg("--arg2");
316
317 EXPECT_EQ(FILE_PATH_LITERAL("prog --switch1 --switch2=foo -- --arg1 --arg2"),
318 cl.GetCommandLineString());
319 CommandLine::StringVector cl_argv = cl.argv();
320 EXPECT_EQ(FILE_PATH_LITERAL("prog"), cl_argv[0]);
321 EXPECT_EQ(FILE_PATH_LITERAL("--switch1"), cl_argv[1]);
322 EXPECT_EQ(FILE_PATH_LITERAL("--switch2=foo"), cl_argv[2]);
323 EXPECT_EQ(FILE_PATH_LITERAL("--"), cl_argv[3]);
324 EXPECT_EQ(FILE_PATH_LITERAL("--arg1"), cl_argv[4]);
325 EXPECT_EQ(FILE_PATH_LITERAL("--arg2"), cl_argv[5]);
326 }
327
328 // Tests that when AppendArguments is called that the program is set correctly
329 // on the target CommandLine object and the switches from the source
330 // CommandLine are added to the target.
TEST(CommandLineTest,AppendArguments)331 TEST(CommandLineTest, AppendArguments) {
332 CommandLine cl1(FilePath(FILE_PATH_LITERAL("Program")));
333 cl1.AppendSwitch("switch1");
334 cl1.AppendSwitchASCII("switch2", "foo");
335
336 CommandLine cl2(CommandLine::NO_PROGRAM);
337 cl2.AppendArguments(cl1, true);
338 EXPECT_EQ(cl1.GetProgram().value(), cl2.GetProgram().value());
339 EXPECT_EQ(cl1.GetCommandLineString(), cl2.GetCommandLineString());
340
341 CommandLine c1(FilePath(FILE_PATH_LITERAL("Program1")));
342 c1.AppendSwitch("switch1");
343 CommandLine c2(FilePath(FILE_PATH_LITERAL("Program2")));
344 c2.AppendSwitch("switch2");
345
346 c1.AppendArguments(c2, true);
347 EXPECT_EQ(c1.GetProgram().value(), c2.GetProgram().value());
348 EXPECT_TRUE(c1.HasSwitch("switch1"));
349 EXPECT_TRUE(c1.HasSwitch("switch2"));
350 }
351
352 #if defined(OS_WIN)
353 // Make sure that the command line string program paths are quoted as necessary.
354 // This only makes sense on Windows and the test is basically here to guard
355 // against regressions.
TEST(CommandLineTest,ProgramQuotes)356 TEST(CommandLineTest, ProgramQuotes) {
357 // Check that quotes are not added for paths without spaces.
358 const FilePath kProgram(L"Program");
359 CommandLine cl_program(kProgram);
360 EXPECT_EQ(kProgram.value(), cl_program.GetProgram().value());
361 EXPECT_EQ(kProgram.value(), cl_program.GetCommandLineString());
362
363 const FilePath kProgramPath(L"Program Path");
364
365 // Check that quotes are not returned from GetProgram().
366 CommandLine cl_program_path(kProgramPath);
367 EXPECT_EQ(kProgramPath.value(), cl_program_path.GetProgram().value());
368
369 // Check that quotes are added to command line string paths containing spaces.
370 CommandLine::StringType cmd_string(cl_program_path.GetCommandLineString());
371 EXPECT_EQ(L"\"Program Path\"", cmd_string);
372
373 // Check the optional quoting of placeholders in programs.
374 CommandLine cl_quote_placeholder(FilePath(L"%1"));
375 EXPECT_EQ(L"%1", cl_quote_placeholder.GetCommandLineString());
376 EXPECT_EQ(L"\"%1\"",
377 cl_quote_placeholder.GetCommandLineStringWithPlaceholders());
378 }
379 #endif
380
381 // Calling Init multiple times should not modify the previous CommandLine.
TEST(CommandLineTest,Init)382 TEST(CommandLineTest, Init) {
383 // Call Init without checking output once so we know it's been called
384 // whether or not the test runner does so.
385 CommandLine::Init(0, nullptr);
386 CommandLine* initial = CommandLine::ForCurrentProcess();
387 EXPECT_FALSE(CommandLine::Init(0, nullptr));
388 CommandLine* current = CommandLine::ForCurrentProcess();
389 EXPECT_EQ(initial, current);
390 }
391
392 // Test that copies of CommandLine have a valid StringPiece map.
TEST(CommandLineTest,Copy)393 TEST(CommandLineTest, Copy) {
394 std::unique_ptr<CommandLine> initial(
395 new CommandLine(CommandLine::NO_PROGRAM));
396 initial->AppendSwitch("a");
397 initial->AppendSwitch("bbbbbbbbbbbbbbb");
398 initial->AppendSwitch("c");
399 CommandLine copy_constructed(*initial);
400 CommandLine assigned = *initial;
401 CommandLine::SwitchMap switch_map = initial->GetSwitches();
402 initial.reset();
403 for (const auto& pair : switch_map)
404 EXPECT_TRUE(copy_constructed.HasSwitch(pair.first));
405 for (const auto& pair : switch_map)
406 EXPECT_TRUE(assigned.HasSwitch(pair.first));
407 }
408
TEST(CommandLineTest,PrependSimpleWrapper)409 TEST(CommandLineTest, PrependSimpleWrapper) {
410 CommandLine cl(FilePath(FILE_PATH_LITERAL("Program")));
411 cl.AppendSwitch("a");
412 cl.AppendSwitch("b");
413 cl.PrependWrapper(FILE_PATH_LITERAL("wrapper --foo --bar"));
414
415 EXPECT_EQ(6u, cl.argv().size());
416 EXPECT_EQ(FILE_PATH_LITERAL("wrapper"), cl.argv()[0]);
417 EXPECT_EQ(FILE_PATH_LITERAL("--foo"), cl.argv()[1]);
418 EXPECT_EQ(FILE_PATH_LITERAL("--bar"), cl.argv()[2]);
419 EXPECT_EQ(FILE_PATH_LITERAL("Program"), cl.argv()[3]);
420 EXPECT_EQ(FILE_PATH_LITERAL("--a"), cl.argv()[4]);
421 EXPECT_EQ(FILE_PATH_LITERAL("--b"), cl.argv()[5]);
422 }
423
TEST(CommandLineTest,PrependComplexWrapper)424 TEST(CommandLineTest, PrependComplexWrapper) {
425 CommandLine cl(FilePath(FILE_PATH_LITERAL("Program")));
426 cl.AppendSwitch("a");
427 cl.AppendSwitch("b");
428 cl.PrependWrapper(
429 FILE_PATH_LITERAL("wrapper --foo='hello world' --bar=\"let's go\""));
430
431 EXPECT_EQ(6u, cl.argv().size());
432 EXPECT_EQ(FILE_PATH_LITERAL("wrapper"), cl.argv()[0]);
433 EXPECT_EQ(FILE_PATH_LITERAL("--foo='hello world'"), cl.argv()[1]);
434 EXPECT_EQ(FILE_PATH_LITERAL("--bar=\"let's go\""), cl.argv()[2]);
435 EXPECT_EQ(FILE_PATH_LITERAL("Program"), cl.argv()[3]);
436 EXPECT_EQ(FILE_PATH_LITERAL("--a"), cl.argv()[4]);
437 EXPECT_EQ(FILE_PATH_LITERAL("--b"), cl.argv()[5]);
438 }
439
440 } // namespace base
441