]>
Commit | Line | Data |
---|---|---|
1 | /////////////////////////////////////////////////////////////////////////////// | |
2 | // Name: tests/cmdline/cmdlinetest.cpp | |
3 | // Purpose: wxCmdLineParser unit test | |
4 | // Author: Vadim Zeitlin | |
5 | // Created: 2008-04-12 | |
6 | // RCS-ID: $Id$ | |
7 | // Copyright: (c) 2008 Vadim Zeitlin <vadim@wxwidgets.org> | |
8 | /////////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | // ---------------------------------------------------------------------------- | |
11 | // headers | |
12 | // ---------------------------------------------------------------------------- | |
13 | ||
14 | #include "testprec.h" | |
15 | ||
16 | #ifdef __BORLANDC__ | |
17 | #pragma hdrstop | |
18 | #endif | |
19 | ||
20 | #ifndef WX_PRECOMP | |
21 | #endif // WX_PRECOMP | |
22 | ||
23 | #include "wx/cmdline.h" | |
24 | #include "wx/msgout.h" | |
25 | #include "wx/scopeguard.h" | |
26 | ||
27 | // -------------------------------------------------------------------------- | |
28 | // test class | |
29 | // -------------------------------------------------------------------------- | |
30 | ||
31 | class CmdLineTestCase : public CppUnit::TestCase | |
32 | { | |
33 | public: | |
34 | CmdLineTestCase() {} | |
35 | ||
36 | private: | |
37 | CPPUNIT_TEST_SUITE( CmdLineTestCase ); | |
38 | CPPUNIT_TEST( ConvertStringTestCase ); | |
39 | CPPUNIT_TEST( ParseSwitches ); | |
40 | CPPUNIT_TEST( Usage ); | |
41 | CPPUNIT_TEST_SUITE_END(); | |
42 | ||
43 | void ConvertStringTestCase(); | |
44 | void ParseSwitches(); | |
45 | void Usage(); | |
46 | ||
47 | DECLARE_NO_COPY_CLASS(CmdLineTestCase) | |
48 | }; | |
49 | ||
50 | // register in the unnamed registry so that these tests are run by default | |
51 | CPPUNIT_TEST_SUITE_REGISTRATION( CmdLineTestCase ); | |
52 | ||
53 | // also include in its own registry so that these tests can be run alone | |
54 | CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( CmdLineTestCase, "CmdLineTestCase" ); | |
55 | ||
56 | // ============================================================================ | |
57 | // implementation | |
58 | // ============================================================================ | |
59 | ||
60 | void CmdLineTestCase::ConvertStringTestCase() | |
61 | { | |
62 | #define WX_ASSERT_DOS_ARGS_EQUAL(s, args) \ | |
63 | { \ | |
64 | const wxArrayString \ | |
65 | argsDOS(wxCmdLineParser::ConvertStringToArgs(args, \ | |
66 | wxCMD_LINE_SPLIT_DOS)); \ | |
67 | WX_ASSERT_STRARRAY_EQUAL(s, argsDOS); \ | |
68 | } | |
69 | ||
70 | #define WX_ASSERT_UNIX_ARGS_EQUAL(s, args) \ | |
71 | { \ | |
72 | const wxArrayString \ | |
73 | argsUnix(wxCmdLineParser::ConvertStringToArgs(args, \ | |
74 | wxCMD_LINE_SPLIT_UNIX)); \ | |
75 | WX_ASSERT_STRARRAY_EQUAL(s, argsUnix); \ | |
76 | } | |
77 | ||
78 | #define WX_ASSERT_ARGS_EQUAL(s, args) \ | |
79 | WX_ASSERT_DOS_ARGS_EQUAL(s, args) \ | |
80 | WX_ASSERT_UNIX_ARGS_EQUAL(s, args) | |
81 | ||
82 | // normal cases | |
83 | WX_ASSERT_ARGS_EQUAL( "foo", "foo" ) | |
84 | WX_ASSERT_ARGS_EQUAL( "foo bar", "\"foo bar\"" ) | |
85 | WX_ASSERT_ARGS_EQUAL( "foo|bar", "foo bar" ) | |
86 | WX_ASSERT_ARGS_EQUAL( "foo|bar|baz", "foo bar baz" ) | |
87 | WX_ASSERT_ARGS_EQUAL( "foo|bar baz", "foo \"bar baz\"" ) | |
88 | ||
89 | // special cases | |
90 | WX_ASSERT_ARGS_EQUAL( "", "" ) | |
91 | WX_ASSERT_ARGS_EQUAL( "foo", "foo " ) | |
92 | WX_ASSERT_ARGS_EQUAL( "foo", "foo \t " ) | |
93 | WX_ASSERT_ARGS_EQUAL( "foo|bar", "foo bar " ) | |
94 | WX_ASSERT_ARGS_EQUAL( "foo|bar|", "foo bar \"" ) | |
95 | WX_ASSERT_DOS_ARGS_EQUAL( "foo|bar|\\", "foo bar \\" ) | |
96 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo|bar|", "foo bar \\" ) | |
97 | ||
98 | WX_ASSERT_ARGS_EQUAL( "12 34", "1\"2 3\"4" ); | |
99 | WX_ASSERT_ARGS_EQUAL( "1|2 34", "1 \"2 3\"4" ); | |
100 | WX_ASSERT_ARGS_EQUAL( "1|2 3|4", "1 \"2 3\" 4" ); | |
101 | ||
102 | // check for (broken) Windows semantics: backslash doesn't escape spaces | |
103 | WX_ASSERT_DOS_ARGS_EQUAL( "\\\\foo\\\\|/bar", "\"\\\\foo\\\\\" /bar" ); | |
104 | WX_ASSERT_DOS_ARGS_EQUAL( "foo|bar\\|baz", "foo bar\\ baz" ); | |
105 | WX_ASSERT_DOS_ARGS_EQUAL( "foo|bar\\\"baz", "foo \"bar\\\"baz\"" ); | |
106 | ||
107 | // check for more sane Unix semantics: backslash does escape spaces and | |
108 | // quotes | |
109 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo|bar baz", "foo bar\\ baz" ); | |
110 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo|bar\"baz", "foo \"bar\\\"baz\"" ); | |
111 | ||
112 | // check that single quotes work too with Unix semantics | |
113 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo bar", "'foo bar'" ) | |
114 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo|bar baz", "foo 'bar baz'" ) | |
115 | WX_ASSERT_UNIX_ARGS_EQUAL( "foo|bar baz", "foo 'bar baz'" ) | |
116 | WX_ASSERT_UNIX_ARGS_EQUAL( "O'Henry", "\"O'Henry\"" ) | |
117 | WX_ASSERT_UNIX_ARGS_EQUAL( "O'Henry", "O\\'Henry" ) | |
118 | ||
119 | #undef WX_ASSERT_DOS_ARGS_EQUAL | |
120 | #undef WX_ASSERT_UNIX_ARGS_EQUAL | |
121 | #undef WX_ASSERT_ARGS_EQUAL | |
122 | } | |
123 | ||
124 | void CmdLineTestCase::ParseSwitches() | |
125 | { | |
126 | // install a dummy message output object just suppress error messages from | |
127 | // wxCmdLineParser::Parse() | |
128 | class NoMessageOutput : public wxMessageOutput | |
129 | { | |
130 | public: | |
131 | virtual void Output(const wxString& WXUNUSED(str)) { } | |
132 | } noMessages; | |
133 | ||
134 | wxMessageOutput * const old = wxMessageOutput::Set(&noMessages); | |
135 | wxON_BLOCK_EXIT1( wxMessageOutput::Set, old ); | |
136 | ||
137 | wxCmdLineParser p; | |
138 | p.AddSwitch("a"); | |
139 | p.AddSwitch("b"); | |
140 | p.AddSwitch("c"); | |
141 | p.AddSwitch("d"); | |
142 | p.AddSwitch("n", "neg", "Switch that can be negated", | |
143 | wxCMD_LINE_SWITCH_NEGATABLE); | |
144 | ||
145 | p.SetCmdLine(""); | |
146 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
147 | CPPUNIT_ASSERT( !p.Found("a") ); | |
148 | ||
149 | p.SetCmdLine("-z"); | |
150 | CPPUNIT_ASSERT( p.Parse(false) != 0 ); | |
151 | ||
152 | p.SetCmdLine("-a"); | |
153 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
154 | CPPUNIT_ASSERT( p.Found("a") ); | |
155 | CPPUNIT_ASSERT( !p.Found("b") ); | |
156 | ||
157 | p.SetCmdLine("-a -d"); | |
158 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
159 | CPPUNIT_ASSERT( p.Found("a") ); | |
160 | CPPUNIT_ASSERT( !p.Found("b") ); | |
161 | CPPUNIT_ASSERT( !p.Found("c") ); | |
162 | CPPUNIT_ASSERT( p.Found("d") ); | |
163 | ||
164 | p.SetCmdLine("-abd"); | |
165 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
166 | CPPUNIT_ASSERT( p.Found("a") ); | |
167 | CPPUNIT_ASSERT( p.Found("b") ); | |
168 | CPPUNIT_ASSERT( !p.Found("c") ); | |
169 | CPPUNIT_ASSERT( p.Found("d") ); | |
170 | ||
171 | p.SetCmdLine("-abdz"); | |
172 | CPPUNIT_ASSERT( p.Parse(false) != 0 ); | |
173 | ||
174 | p.SetCmdLine("-ab -cd"); | |
175 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
176 | CPPUNIT_ASSERT( p.Found("a") ); | |
177 | CPPUNIT_ASSERT( p.Found("b") ); | |
178 | CPPUNIT_ASSERT( p.Found("c") ); | |
179 | CPPUNIT_ASSERT( p.Found("d") ); | |
180 | ||
181 | p.SetCmdLine("-da"); | |
182 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
183 | CPPUNIT_ASSERT( p.Found("a") ); | |
184 | CPPUNIT_ASSERT( !p.Found("b") ); | |
185 | CPPUNIT_ASSERT( !p.Found("c") ); | |
186 | CPPUNIT_ASSERT( p.Found("d") ); | |
187 | ||
188 | p.SetCmdLine("-n"); | |
189 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
190 | CPPUNIT_ASSERT_EQUAL(wxCMD_SWITCH_NOT_FOUND, p.FoundSwitch("a") ); | |
191 | CPPUNIT_ASSERT_EQUAL(wxCMD_SWITCH_ON, p.FoundSwitch("n") ); | |
192 | ||
193 | p.SetCmdLine("-n-"); | |
194 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
195 | CPPUNIT_ASSERT_EQUAL(wxCMD_SWITCH_OFF, p.FoundSwitch("neg") ); | |
196 | ||
197 | p.SetCmdLine("--neg"); | |
198 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
199 | CPPUNIT_ASSERT_EQUAL(wxCMD_SWITCH_ON, p.FoundSwitch("neg") ); | |
200 | ||
201 | p.SetCmdLine("--neg-"); | |
202 | CPPUNIT_ASSERT_EQUAL(0, p.Parse(false) ); | |
203 | CPPUNIT_ASSERT_EQUAL(wxCMD_SWITCH_OFF, p.FoundSwitch("n") ); | |
204 | } | |
205 | ||
206 | void CmdLineTestCase::Usage() | |
207 | { | |
208 | // check that Usage() returns roughly what we expect (don't check all the | |
209 | // details, its format can change in the future) | |
210 | static const wxCmdLineEntryDesc desc[] = | |
211 | { | |
212 | { wxCMD_LINE_USAGE_TEXT, NULL, NULL, "Verbosity options" }, | |
213 | { wxCMD_LINE_SWITCH, "v", "verbose", "be verbose" }, | |
214 | { wxCMD_LINE_SWITCH, "q", "quiet", "be quiet" }, | |
215 | ||
216 | { wxCMD_LINE_USAGE_TEXT, NULL, NULL, "Output options" }, | |
217 | { wxCMD_LINE_OPTION, "o", "output", "output file" }, | |
218 | { wxCMD_LINE_OPTION, "s", "size", "output block size", wxCMD_LINE_VAL_NUMBER }, | |
219 | { wxCMD_LINE_OPTION, "d", "date", "output file date", wxCMD_LINE_VAL_DATE }, | |
220 | { wxCMD_LINE_OPTION, "f", "double", "output double", wxCMD_LINE_VAL_DOUBLE }, | |
221 | ||
222 | { wxCMD_LINE_PARAM, NULL, NULL, "input file", }, | |
223 | ||
224 | { wxCMD_LINE_USAGE_TEXT, NULL, NULL, "\nEven more usage text" }, | |
225 | { wxCMD_LINE_NONE } | |
226 | }; | |
227 | ||
228 | wxCmdLineParser p(desc); | |
229 | const wxArrayString usageLines = wxSplit(p.GetUsageString(), '\n'); | |
230 | ||
231 | enum | |
232 | { | |
233 | Line_Synopsis, | |
234 | Line_Text_Verbosity, | |
235 | Line_Verbose, | |
236 | Line_Quiet, | |
237 | Line_Text_Output, | |
238 | Line_Output_File, | |
239 | Line_Output_Size, | |
240 | Line_Output_Date, | |
241 | Line_Output_Double, | |
242 | Line_Text_Dummy1, | |
243 | Line_Text_Dummy2, | |
244 | Line_Last, | |
245 | Line_Max | |
246 | }; | |
247 | ||
248 | CPPUNIT_ASSERT_EQUAL(Line_Max, usageLines.size()); | |
249 | CPPUNIT_ASSERT_EQUAL("Verbosity options", usageLines[Line_Text_Verbosity]); | |
250 | CPPUNIT_ASSERT_EQUAL("", usageLines[Line_Text_Dummy1]); | |
251 | CPPUNIT_ASSERT_EQUAL("Even more usage text", usageLines[Line_Text_Dummy2]); | |
252 | CPPUNIT_ASSERT_EQUAL("", usageLines[Line_Last]); | |
253 | } |