2 * This file is part of the GROMACS molecular simulation package.
4 * Copyright (c) 2015,2016,2019,2020, by the GROMACS development team, led by
5 * Mark Abraham, David van der Spoel, Berk Hess, and Erik Lindahl,
6 * and including many others, as listed in the AUTHORS file in the
7 * top-level source directory and at http://www.gromacs.org.
9 * GROMACS is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public License
11 * as published by the Free Software Foundation; either version 2.1
12 * of the License, or (at your option) any later version.
14 * GROMACS is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with GROMACS; if not, see
21 * http://www.gnu.org/licenses, or write to the Free Software Foundation,
22 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 * If you want to redistribute modifications to GROMACS, please
25 * consider that scientific software is very special. Version
26 * control is crucial - bugs must be traceable. We will be happy to
27 * consider code for inclusion in the official distribution, but
28 * derived work must not be called official GROMACS. Details are found
29 * in the README & COPYING files - if they are missing, get the
30 * official version at http://www.gromacs.org.
32 * To help us fund GROMACS development, we humbly ask that you cite
33 * the research papers on the package. Check out http://www.gromacs.org.
37 * Self-tests for interactive test helpers.
39 * \author Teemu Murtola <teemu.murtola@gmail.com>
40 * \ingroup module_testutils
44 #include "testutils/interactivetest.h"
48 #include <gtest/gtest.h>
49 #include <gtest/gtest-spi.h>
51 #include "gromacs/utility/textstream.h"
53 #include "testutils/refdata.h"
62 class InteractiveSession
65 explicit InteractiveSession(ReferenceDataMode mode
) :
67 helper_(data_
.rootChecker()),
72 void addOutput(const char* output
) { events_
.emplace_back(WriteOutput
, output
); }
73 void addInputLine(const char* inputLine
) { inputLines_
.push_back(inputLine
); }
74 void addReadInput() { events_
.emplace_back(ReadInput
, ""); }
75 void addInput(const char* inputLine
)
77 addInputLine(inputLine
);
80 void addInputNoNewline(const char* inputLine
)
82 addInputLine(inputLine
);
83 helper_
.setLastNewline(false);
84 events_
.emplace_back(ReadInputNoNewline
, "");
89 gmx::TextInputStream
& input
= helper_
.inputStream();
90 gmx::TextOutputStream
& output
= helper_
.outputStream();
91 helper_
.setInputLines(inputLines_
);
92 std::vector
<Event
>::const_iterator event
;
93 for (event
= events_
.begin(); event
!= events_
.end(); ++event
)
95 if (event
->first
== WriteOutput
)
97 output
.write(event
->second
);
101 std::string expectedLine
;
102 const bool bInputRemaining
= (nextInputLine_
< inputLines_
.size());
105 expectedLine
= inputLines_
[nextInputLine_
];
106 if (event
->first
!= ReadInputNoNewline
)
108 expectedLine
.append("\n");
113 EXPECT_EQ(bInputRemaining
, input
.readLine(&line
));
114 EXPECT_EQ(expectedLine
, line
);
117 helper_
.checkSession();
127 // The latter is the output string.
128 typedef std::pair
<EventType
, const char*> Event
;
130 TestReferenceData data_
;
131 InteractiveTestHelper helper_
;
132 std::vector
<const char*> inputLines_
;
133 size_t nextInputLine_
;
134 std::vector
<Event
> events_
;
137 TEST(InteractiveTestHelperTest
, ChecksSimpleSession
)
140 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
141 session
.addOutput("First line\n");
142 session
.addOutput("> ");
143 session
.addInput("input");
144 session
.addOutput("Second line\n");
145 session
.addOutput("> ");
146 session
.addReadInput();
147 session
.addOutput("\n");
148 session
.addOutput(".\n");
152 InteractiveSession
session(ReferenceDataMode::Compare
);
153 session
.addOutput("First line\n");
154 session
.addOutput("> ");
155 session
.addInput("input");
156 session
.addOutput("Second line\n");
157 session
.addOutput("> ");
158 session
.addReadInput();
159 session
.addOutput("\n");
160 session
.addOutput(".\n");
165 TEST(InteractiveTestHelperTest
, ChecksSessionWithoutLastNewline
)
168 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
169 session
.addOutput("First line\n");
170 session
.addOutput("> ");
171 session
.addInput("input");
172 session
.addOutput("Second line\n");
173 session
.addOutput("> ");
174 session
.addInputNoNewline("input2");
175 session
.addOutput("\n");
176 session
.addOutput(".\n");
180 InteractiveSession
session(ReferenceDataMode::Compare
);
181 session
.addOutput("First line\n");
182 session
.addOutput("> ");
183 session
.addInput("input");
184 session
.addOutput("Second line\n");
185 session
.addOutput("> ");
186 session
.addInputNoNewline("input2");
187 session
.addOutput("\n");
188 session
.addOutput(".\n");
193 TEST(InteractiveTestHelperTest
, ChecksSessionWithMissingOutput
)
196 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
197 session
.addOutput("First line\n> ");
198 session
.addInput("input");
199 session
.addInput("input2");
200 session
.addOutput("Second line\n> ");
201 session
.addReadInput();
202 session
.addOutput("\n.\n");
206 InteractiveSession
session(ReferenceDataMode::Compare
);
207 session
.addOutput("First line\n> ");
208 session
.addInput("input");
209 session
.addInput("input2");
210 session
.addOutput("Second line\n> ");
211 session
.addReadInput();
212 session
.addOutput("\n.\n");
217 TEST(InteractiveTestHelperTest
, ChecksSessionWithEquivalentOutput
)
220 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
221 session
.addOutput("First line\n");
222 session
.addOutput("> ");
223 session
.addInput("input");
224 session
.addOutput("Second line\n> ");
225 session
.addReadInput();
226 session
.addOutput("\n");
227 session
.addOutput(".\n");
231 InteractiveSession
session(ReferenceDataMode::Compare
);
232 session
.addOutput("First line\n> ");
233 session
.addInput("input");
234 session
.addOutput("Second line\n");
235 session
.addOutput("> ");
236 session
.addReadInput();
237 session
.addOutput("\n.\n");
242 TEST(InteractiveTestHelperTest
, DetectsIncorrectOutput
)
245 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
246 session
.addOutput("First line\n> ");
247 session
.addInput("input");
248 session
.addOutput("Second line\n> ");
249 session
.addReadInput();
250 session
.addOutput("\n.\n");
254 InteractiveSession
session(ReferenceDataMode::Compare
);
255 session
.addOutput("First line\n> ");
256 session
.addInput("input");
257 session
.addOutput("Incorrect line\n> ");
258 session
.addReadInput();
259 session
.addOutput("\n.\n");
260 EXPECT_NONFATAL_FAILURE(session
.run(), "");
264 TEST(InteractiveTestHelperTest
, DetectsMissingOutput
)
267 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
268 session
.addOutput("First line\n> ");
269 session
.addInput("input");
270 session
.addOutput("Second line\n> ");
271 session
.addInput("input2");
272 session
.addOutput("Third line\n> ");
273 session
.addReadInput();
274 session
.addOutput("\n.\n");
278 InteractiveSession
session(ReferenceDataMode::Compare
);
279 session
.addOutput("First line\n> ");
280 session
.addInput("input");
281 session
.addInput("input2");
282 session
.addOutput("Third line\n> ");
283 session
.addReadInput();
284 session
.addOutput("\n.\n");
285 EXPECT_NONFATAL_FAILURE(session
.run(), "");
289 TEST(InteractiveTestHelperTest
, DetectsMissingFinalOutput
)
292 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
293 session
.addOutput("First line\n> ");
294 session
.addInput("input");
295 session
.addOutput("Second line\n> ");
296 session
.addReadInput();
297 session
.addOutput("\n.\n");
301 InteractiveSession
session(ReferenceDataMode::Compare
);
302 session
.addOutput("First line\n> ");
303 session
.addInput("input");
304 session
.addOutput("Second line\n> ");
305 session
.addReadInput();
306 EXPECT_NONFATAL_FAILURE(session
.run(), "");
310 TEST(InteractiveTestHelperTest
, DetectsExtraOutput
)
313 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
314 session
.addOutput("First line\n> ");
315 session
.addInput("input");
316 session
.addInput("input2");
317 session
.addOutput("More output\n> ");
318 session
.addReadInput();
319 session
.addOutput("\n.\n");
323 InteractiveSession
session(ReferenceDataMode::Compare
);
324 session
.addOutput("First line\n> ");
325 session
.addInput("input");
326 session
.addOutput("Extra output\n> ");
327 session
.addInput("input2");
328 session
.addOutput("More output\n> ");
329 session
.addReadInput();
330 session
.addOutput("\n.\n");
331 EXPECT_NONFATAL_FAILURE(session
.run(), "");
335 TEST(InteractiveTestHelperTest
, DetectsMissingInput
)
338 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
339 session
.addInput("input");
340 session
.addInput("input2");
341 session
.addReadInput();
345 InteractiveSession
session(ReferenceDataMode::Compare
);
346 session
.addInputLine("input");
347 session
.addInputLine("input2");
348 session
.addReadInput();
349 session
.addReadInput();
350 EXPECT_NONFATAL_FAILURE(session
.run(), "");
354 TEST(InteractiveTestHelperTest
, DetectsExtraInput
)
357 InteractiveSession
session(ReferenceDataMode::UpdateAll
);
358 session
.addInput("input");
359 session
.addInput("input2");
360 session
.addReadInput();
364 InteractiveSession
session(ReferenceDataMode::Compare
);
365 session
.addInputLine("input");
366 session
.addInputLine("input2");
367 session
.addReadInput();
368 session
.addReadInput();
369 session
.addReadInput();
370 session
.addReadInput();
371 EXPECT_NONFATAL_FAILURE(session
.run(), "");