| // Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "gn/test_with_scope.h" |
| #include "util/test/test.h" |
| |
| TEST(FunctionForeach, CollisionOnLoopVar) { |
| TestWithScope setup; |
| TestParseInput input( |
| "a = 5\n" |
| "i = 6\n" |
| "foreach(i, [1, 2, 3]) {\n" // Use same loop var name previously defined. |
| " print(\"$a $i\")\n" |
| " a = a + 1\n" // Test for side effects inside loop. |
| "}\n" |
| "print(\"$a $i\")"); // Make sure that i goes back to original value. |
| ASSERT_FALSE(input.has_error()); |
| |
| Err err; |
| input.parsed()->Execute(setup.scope(), &err); |
| ASSERT_FALSE(err.has_error()) << err.message(); |
| |
| EXPECT_EQ("5 1\n6 2\n7 3\n8 6\n", setup.print_output()); |
| } |
| |
| TEST(FunctionForeach, UniqueLoopVar) { |
| TestWithScope setup; |
| TestParseInput input_good( |
| "foreach(i, [1, 2, 3]) {\n" |
| " print(i)\n" |
| "}\n"); |
| ASSERT_FALSE(input_good.has_error()); |
| |
| Err err; |
| input_good.parsed()->Execute(setup.scope(), &err); |
| ASSERT_FALSE(err.has_error()) << err.message(); |
| |
| EXPECT_EQ("1\n2\n3\n", setup.print_output()); |
| setup.print_output().clear(); |
| |
| // Same thing but try to use the loop var after loop is done. It should be |
| // undefined and throw an error. |
| TestParseInput input_bad( |
| "foreach(i, [1, 2, 3]) {\n" |
| " print(i)\n" |
| "}\n" |
| "print(i)"); |
| ASSERT_FALSE(input_bad.has_error()); // Should parse OK. |
| |
| input_bad.parsed()->Execute(setup.scope(), &err); |
| ASSERT_TRUE(err.has_error()); // Shouldn't actually run. |
| } |
| |
| // Checks that the identifier used as the list is marked as "used". |
| TEST(FunctionForeach, MarksIdentAsUsed) { |
| TestWithScope setup; |
| TestParseInput input_good( |
| "a = [1, 2]\n" |
| "foreach(i, a) {\n" |
| " print(i)\n" |
| "}\n"); |
| ASSERT_FALSE(input_good.has_error()); |
| |
| Err err; |
| input_good.parsed()->Execute(setup.scope(), &err); |
| ASSERT_FALSE(err.has_error()) << err.message(); |
| |
| EXPECT_EQ("1\n2\n", setup.print_output()); |
| setup.print_output().clear(); |
| |
| // Check for unused vars. |
| EXPECT_TRUE(setup.scope()->CheckForUnusedVars(&err)); |
| EXPECT_FALSE(err.has_error()); |
| } |
| |
| // Checks that the list can be modified during iteration without crashing. |
| TEST(FunctionForeach, ListModification) { |
| TestWithScope setup; |
| TestParseInput input_grow( |
| "a = [1, 2]\n" |
| "foreach(i, a) {\n" |
| " print(i)\n" |
| " if (i <= 8) {\n" |
| " a += [ i + 2 ]\n" |
| " }\n" |
| "}\n" |
| "print(a)"); |
| ASSERT_FALSE(input_grow.has_error()); |
| |
| Err err; |
| input_grow.parsed()->Execute(setup.scope(), &err); |
| ASSERT_FALSE(err.has_error()) << err.message(); |
| |
| // The result of the loop should have been unaffected by the mutations of |
| // the list variable inside the loop, but the modifications made to it |
| // should have been persisted. |
| EXPECT_EQ("1\n2\n[1, 2, 3, 4]\n", setup.print_output()); |
| setup.print_output().clear(); |
| } |