blob: 553385d58b55d460374778b680f3977ddd1bd261 [file] [log] [blame]
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +00001// Copyright 2006-2015 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14package com.google.devtools.build.lib.syntax;
15
16import static com.google.common.truth.Truth.assertThat;
17import static org.junit.Assert.fail;
18
Lukacs Berki4e503752015-09-17 11:52:33 +000019import com.google.common.base.Joiner;
Florian Weikert28da3652015-07-01 14:52:30 +000020import com.google.common.truth.Ordered;
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000021import com.google.devtools.build.lib.events.Event;
22import com.google.devtools.build.lib.events.EventCollector;
23import com.google.devtools.build.lib.events.EventHandler;
24import com.google.devtools.build.lib.events.EventKind;
25import com.google.devtools.build.lib.events.util.EventCollectionApparatus;
Florian Weikert28da3652015-07-01 14:52:30 +000026import com.google.devtools.build.lib.packages.PackageFactory;
27import com.google.devtools.build.lib.testutil.TestMode;
28import com.google.devtools.build.lib.testutil.TestRuleClassProvider;
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000029
30import org.junit.Before;
31
Florian Weikert28da3652015-07-01 14:52:30 +000032import java.util.LinkedList;
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000033import java.util.List;
34
35/**
36 * Base class for test cases that use parsing and evaluation services.
37 */
38public class EvaluationTestCase {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000039
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000040 private EventCollectionApparatus eventCollectionApparatus;
Florian Weikert28da3652015-07-01 14:52:30 +000041 private PackageFactory factory;
42 private TestMode testMode = TestMode.SKYLARK;
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000043 protected Environment env;
44 protected Mutability mutability = Mutability.create("test");
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000045
Florian Weikert28da3652015-07-01 14:52:30 +000046 public EvaluationTestCase() {
47 createNewInfrastructure();
48 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000049
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000050 @Before
51 public void setUp() throws Exception {
Florian Weikert28da3652015-07-01 14:52:30 +000052 createNewInfrastructure();
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000053 env = newEnvironment();
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +000054 }
55
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000056 /**
57 * Creates a standard Environment for tests in the BUILD language.
58 * No PythonPreprocessing, mostly empty mutable Environment.
59 */
60 public Environment newBuildEnvironment() {
61 return Environment.builder(mutability)
62 .setGlobals(Environment.BUILD)
63 .setEventHandler(getEventHandler())
64 .setLoadingPhase()
65 .build();
66 }
67
68 /**
69 * Creates an Environment for Skylark with a mostly empty initial environment.
70 * For internal initialization or tests.
71 */
72 public Environment newSkylarkEnvironment() {
73 return Environment.builder(mutability)
74 .setSkylark()
75 .setGlobals(Environment.SKYLARK)
76 .setEventHandler(getEventHandler())
77 .build();
78 }
79
80 /**
81 * Creates a new Environment suitable for the test case. Subclasses may override it
82 * to fit their purpose and e.g. call newBuildEnvironment or newSkylarkEnvironment;
83 * or they may play with the testMode to run tests in either or both kinds of Environment.
84 * Note that all Environment-s may share the same Mutability, so don't close it.
85 * @return a fresh Environment.
86 */
87 public Environment newEnvironment() throws Exception {
Florian Weikert28da3652015-07-01 14:52:30 +000088 if (testMode == null) {
89 throw new IllegalArgumentException(
90 "TestMode is null. Please set a Testmode via setMode() or set the "
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000091 + "Environment manually by overriding newEnvironment()");
Florian Weikert28da3652015-07-01 14:52:30 +000092 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +000093 return testMode.createEnvironment(getEventHandler(), null);
Florian Weikert28da3652015-07-01 14:52:30 +000094 }
95
96 protected void createNewInfrastructure() {
97 eventCollectionApparatus = new EventCollectionApparatus(EventKind.ALL_EVENTS);
98 factory = new PackageFactory(TestRuleClassProvider.getRuleClassProvider());
99 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000100
Florian Weikert28da3652015-07-01 14:52:30 +0000101 /**
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000102 * Sets the specified {@code TestMode} and tries to create the appropriate {@code Environment}
Florian Weikert28da3652015-07-01 14:52:30 +0000103 * @param testMode
104 * @throws Exception
105 */
106 protected void setMode(TestMode testMode) throws Exception {
107 this.testMode = testMode;
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000108 env = newEnvironment();
Florian Weikert28da3652015-07-01 14:52:30 +0000109 }
110
111 protected void enableSkylarkMode() throws Exception {
112 setMode(TestMode.SKYLARK);
113 }
114
115 protected void enableBuildMode() throws Exception {
116 setMode(TestMode.BUILD);
117 }
118
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000119 protected EventHandler getEventHandler() {
120 return eventCollectionApparatus.reporter();
121 }
122
Florian Weikert28da3652015-07-01 14:52:30 +0000123 protected PackageFactory getFactory() {
124 return factory;
125 }
126
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000127 public Environment getEnvironment() {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000128 return env;
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000129 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000130
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000131 public boolean isSkylark() {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000132 return env.isSkylark();
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000133 }
134
135 protected List<Statement> parseFile(String... input) {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000136 return env.parseFile(input);
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000137 }
138
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000139 /** Parses an Expression from string without a supporting file */
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000140 Expression parseExpression(String... input) {
Lukacs Berki4e503752015-09-17 11:52:33 +0000141 return Parser.parseExpression(
142 ParserInputSource.create(Joiner.on("\n").join(input), null),
143 getEventHandler());
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000144 }
145
146 public EvaluationTestCase update(String varname, Object value) throws Exception {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000147 env.update(varname, value);
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000148 return this;
149 }
150
151 public Object lookup(String varname) throws Exception {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000152 return env.lookup(varname);
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000153 }
154
155 public Object eval(String... input) throws Exception {
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000156 return env.eval(input);
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000157 }
158
159 public void checkEvalError(String msg, String... input) throws Exception {
160 setFailFast(true);
161 try {
162 eval(input);
Laurent Le Brun092f13b2015-08-24 14:50:00 +0000163 fail("Expected error '" + msg + "' but got no error");
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000164 } catch (IllegalArgumentException | EvalException e) {
165 assertThat(e).hasMessage(msg);
166 }
167 }
168
169 public void checkEvalErrorContains(String msg, String... input) throws Exception {
170 try {
171 eval(input);
Laurent Le Brun092f13b2015-08-24 14:50:00 +0000172 fail("Expected error containing '" + msg + "' but got no error");
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000173 } catch (IllegalArgumentException | EvalException e) {
174 assertThat(e.getMessage()).contains(msg);
175 }
176 }
177
178 public void checkEvalErrorStartsWith(String msg, String... input) throws Exception {
179 try {
180 eval(input);
Laurent Le Brun092f13b2015-08-24 14:50:00 +0000181 fail("Expected error starting with '" + msg + "' but got no error");
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000182 } catch (IllegalArgumentException | EvalException e) {
183 assertThat(e.getMessage()).startsWith(msg);
184 }
185 }
186
187 // Forward relevant methods to the EventCollectionApparatus
188 public EvaluationTestCase setFailFast(boolean failFast) {
189 eventCollectionApparatus.setFailFast(failFast);
190 return this;
191 }
192 public EvaluationTestCase assertNoEvents() {
193 eventCollectionApparatus.assertNoEvents();
194 return this;
195 }
196 public EventCollector getEventCollector() {
197 return eventCollectionApparatus.collector();
198 }
199 public Event assertContainsEvent(String expectedMessage) {
200 return eventCollectionApparatus.assertContainsEvent(expectedMessage);
201 }
202 public List<Event> assertContainsEventWithFrequency(String expectedMessage,
203 int expectedFrequency) {
204 return eventCollectionApparatus.assertContainsEventWithFrequency(
205 expectedMessage, expectedFrequency);
206 }
207 public Event assertContainsEventWithWordsInQuotes(String... words) {
208 return eventCollectionApparatus.assertContainsEventWithWordsInQuotes(words);
209 }
210 public EvaluationTestCase clearEvents() {
211 eventCollectionApparatus.collector().clear();
212 return this;
213 }
Florian Weikert28da3652015-07-01 14:52:30 +0000214
215 /**
216 * Encapsulates a separate test which can be executed by a {@code TestMode}
217 */
218 protected interface Testable {
219 public void run() throws Exception;
220 }
221
222 /**
223 * Base class for test cases that run in specific modes (e.g. Build and/or Skylark)
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000224 */
Florian Weikert28da3652015-07-01 14:52:30 +0000225 protected abstract class ModalTestCase {
226 private final SetupActions setup;
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000227
228 protected ModalTestCase() {
Florian Weikert28da3652015-07-01 14:52:30 +0000229 setup = new SetupActions();
230 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000231
Florian Weikert28da3652015-07-01 14:52:30 +0000232 /**
233 * Allows the execution of several statements before each following test
234 * @param statements The statement(s) to be executed
235 * @return This {@code ModalTestCase}
236 */
237 public ModalTestCase setUp(String... statements) {
238 setup.registerEval(statements);
239 return this;
240 }
241
242 /**
243 * Allows the update of the specified variable before each following test
244 * @param name The name of the variable that should be updated
245 * @param value The new value of the variable
246 * @return This {@code ModalTestCase}
247 */
248 public ModalTestCase update(String name, Object value) {
249 setup.registerUpdate(name, value);
250 return this;
251 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000252
Florian Weikert28da3652015-07-01 14:52:30 +0000253 /**
254 * Evaluates two parameters and compares their results.
255 * @param statement The statement to be evaluated
256 * @param expectedEvalString The expression of the expected result
257 * @return This {@code ModalTestCase}
258 * @throws Exception
259 */
260 public ModalTestCase testEval(String statement, String expectedEvalString) throws Exception {
261 runTest(createComparisonTestable(statement, expectedEvalString, true));
262 return this;
263 }
264
265 /**
266 * Evaluates the given statement and compares its result to the expected object
267 * @param statement
268 * @param expected
269 * @return This {@code ModalTestCase}
270 * @throws Exception
271 */
272 public ModalTestCase testStatement(String statement, Object expected) throws Exception {
273 runTest(createComparisonTestable(statement, expected, false));
274 return this;
275 }
276
277 /**
278 * Evaluates the given statement and compares its result to the collection of expected objects
279 * without considering their order
280 * @param statement The statement to be evaluated
281 * @param items The expected items
282 * @return This {@code ModalTestCase}
283 * @throws Exception
284 */
285 public ModalTestCase testCollection(String statement, Object... items) throws Exception {
286 runTest(collectionTestable(statement, false, items));
287 return this;
288 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000289
Florian Weikert28da3652015-07-01 14:52:30 +0000290 /**
291 * Evaluates the given statement and compares its result to the collection of expected objects
292 * while considering their order
293 * @param statement The statement to be evaluated
294 * @param items The expected items, in order
295 * @return This {@code ModalTestCase}
296 * @throws Exception
297 */
298 public ModalTestCase testExactOrder(String statement, Object... items) throws Exception {
299 runTest(collectionTestable(statement, true, items));
300 return this;
301 }
302
303 /**
304 * Evaluates the given statement and checks whether the given error message appears
305 * @param expectedError The expected error message
306 * @param statements The statement(s) to be evaluated
307 * @return This ModalTestCase
308 * @throws Exception
309 */
310 public ModalTestCase testIfExactError(String expectedError, String... statements)
311 throws Exception {
312 runTest(errorTestable(true, expectedError, statements));
313 return this;
314 }
315
316 /**
317 * Evaluates the given statement and checks whether an error that contains the expected message
318 * occurs
319 * @param expectedError
320 * @param statements
321 * @return This ModalTestCase
322 * @throws Exception
323 */
324 public ModalTestCase testIfErrorContains(String expectedError, String... statements)
325 throws Exception {
326 runTest(errorTestable(false, expectedError, statements));
327 return this;
328 }
329
330 /**
331 * Looks up the value of the specified variable and compares it to the expected value
332 * @param name
333 * @param expected
334 * @return This ModalTestCase
335 * @throws Exception
336 */
337 public ModalTestCase testLookup(String name, Object expected) throws Exception {
338 runTest(createLookUpTestable(name, expected));
339 return this;
340 }
341
342 /**
343 * Creates a Testable that checks whether the evaluation of the given statement leads to the
344 * expected error
345 * @param statements
346 * @param error
347 * @param exactMatch If true, the error message has to be identical to the expected error
348 * @return An instance of Testable that runs the error check
349 */
350 protected Testable errorTestable(final boolean exactMatch, final String error,
351 final String... statements) {
352 return new Testable() {
353 @Override
354 public void run() throws Exception {
355 if (exactMatch) {
356 checkEvalError(error, statements);
357 } else {
358 checkEvalErrorContains(error, statements);
359 }
360 }
361 };
362 }
363
364 /**
365 * Creates a testable that checks whether the evaluation of the given statement leads to a list
366 * that contains exactly the expected objects
367 * @param statement The statement to be evaluated
368 * @param ordered Determines whether the order of the elements is checked as well
369 * @param expected Expected objects
370 * @return An instance of Testable that runs the check
371 */
372 protected Testable collectionTestable(
373 final String statement, final boolean ordered, final Object... expected) {
374 return new Testable() {
375 @Override
376 public void run() throws Exception {
377 Ordered tmp = assertThat((Iterable<?>) eval(statement)).containsExactly(expected);
378
379 if (ordered) {
380 tmp.inOrder();
381 }
382 }
383 };
384 }
385
386 /**
387 * Creates a testable that compares the evaluation of the given statement to a specified result
388 *
389 * @param statement The statement to be evaluated
390 * @param expected Either the expected object or an expression whose evaluation leads to the
391 * expected object
392 * @param expectedIsExpression Signals whether {@code expected} is an object or an expression
393 * @return An instance of Testable that runs the comparison
394 */
395 protected Testable createComparisonTestable(
396 final String statement, final Object expected, final boolean expectedIsExpression) {
397 return new Testable() {
398 @Override
399 public void run() throws Exception {
400 Object actual = eval(statement);
Florian Weikertf31b9472015-08-04 16:36:58 +0000401 Object realExpected = expected;
Florian Weikert28da3652015-07-01 14:52:30 +0000402
Florian Weikertf31b9472015-08-04 16:36:58 +0000403 // We could also print the actual object and compare the string to the expected
404 // expression, but then the order of elements would matter.
Florian Weikert28da3652015-07-01 14:52:30 +0000405 if (expectedIsExpression) {
Florian Weikertf31b9472015-08-04 16:36:58 +0000406 realExpected = eval((String) expected);
Florian Weikert28da3652015-07-01 14:52:30 +0000407 }
408
Florian Weikertf31b9472015-08-04 16:36:58 +0000409 assertThat(actual).isEqualTo(realExpected);
Florian Weikert28da3652015-07-01 14:52:30 +0000410 }
411 };
412 }
413
414 /**
415 * Creates a Testable that looks up the given variable and compares its value to the expected
416 * value
417 * @param name
418 * @param expected
419 * @return An instance of Testable that does both lookup and comparison
420 */
421 protected Testable createLookUpTestable(final String name, final Object expected) {
422 return new Testable() {
423 @Override
424 public void run() throws Exception {
425 assertThat(lookup(name)).isEqualTo(expected);
426 }
427 };
428 }
429
430 /**
431 * Executes the given Testable
432 * @param testable
433 * @throws Exception
434 */
435 protected void runTest(Testable testable) throws Exception {
436 run(new TestableDecorator(setup, testable));
437 }
438
439 protected abstract void run(Testable testable) throws Exception;
440 }
441
442 /**
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000443 * A simple decorator that allows the execution of setup actions before running
Florian Weikert28da3652015-07-01 14:52:30 +0000444 * a {@code Testable}
445 */
446 class TestableDecorator implements Testable {
447 private final SetupActions setup;
448 private final Testable decorated;
449
450 public TestableDecorator(SetupActions setup, Testable decorated) {
451 this.setup = setup;
452 this.decorated = decorated;
453 }
454
455 /**
456 * Executes all stored actions and updates plus the actual {@code Testable}
457 */
458 @Override
459 public void run() throws Exception {
460 setup.executeAll();
461 decorated.run();
462 }
463 }
464
465 /**
466 * A container for collection actions that should be executed before a test
467 */
468 class SetupActions {
469 private List<Testable> setup;
470
471 public SetupActions() {
472 setup = new LinkedList<>();
473 }
474
475 /**
476 * Registers a variable that has to be updated before a test
477 *
478 * @param name
479 * @param value
480 */
481 public void registerUpdate(final String name, final Object value) {
482 setup.add(new Testable() {
483 @Override
484 public void run() throws Exception {
485 EvaluationTestCase.this.update(name, value);
486 }
487 });
488 }
489
490 /**
491 * Registers a statement for evaluation prior to a test
492 *
493 * @param statements
494 */
495 public void registerEval(final String... statements) {
496 setup.add(new Testable() {
497 @Override
498 public void run() throws Exception {
499 EvaluationTestCase.this.eval(statements);
500 }
501 });
502 }
503
504 /**
505 * Executes all stored actions and updates
506 * @throws Exception
507 */
508 public void executeAll() throws Exception {
509 for (Testable testable : setup) {
510 testable.run();
511 }
512 }
513 }
Francois-Rene Rideau89312fb2015-09-10 18:53:03 +0000514
Florian Weikert28da3652015-07-01 14:52:30 +0000515 /**
516 * A class that executes each separate test in both modes (Build and Skylark)
517 */
518 protected class BothModesTest extends ModalTestCase {
519 public BothModesTest() {}
520
521 /**
522 * Executes the given Testable in both Build and Skylark mode
523 */
524 @Override
525 protected void run(Testable testable) throws Exception {
526 enableSkylarkMode();
527 try {
528 testable.run();
529 } catch (Exception e) {
530 throw new Exception("While in Skylark mode", e);
531 }
532
533 enableBuildMode();
534 try {
535 testable.run();
536 } catch (Exception e) {
537 throw new Exception("While in Build mode", e);
538 }
539 }
540 }
541
542 /**
543 * A class that runs all tests in Build mode
544 */
545 protected class BuildTest extends ModalTestCase {
546 public BuildTest() {}
547
548 @Override
549 protected void run(Testable testable) throws Exception {
550 enableBuildMode();
551 testable.run();
552 }
553 }
554
555 /**
556 * A class that runs all tests in Skylark mode
557 */
558 protected class SkylarkTest extends ModalTestCase {
559 public SkylarkTest() {}
560
561 @Override
562 protected void run(Testable testable) throws Exception {
563 enableSkylarkMode();
564 testable.run();
565 }
566 }
Francois-Rene Rideau5f3e30c2015-04-10 19:08:39 +0000567}