// Copyright 2015 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/scheduler.h" #include "gn/test_with_scheduler.h" #include "gn/test_with_scope.h" #include "util/test/test.h" using FunctionForwardVariablesFromTest = TestWithScheduler; TEST_F(FunctionForwardVariablesFromTest, List) { Err err; std::string program = "template(\"a\") {\n" " forward_variables_from(invoker, [\"x\", \"y\", \"z\"])\n" " assert(!defined(z))\n" // "z" should still be undefined. " print(\"$target_name, $x, $y\")\n" "}\n" "a(\"target\") {\n" " x = 1\n" " y = 2\n" "}\n"; { TestWithScope setup; // Defines a template and copy the two x and y, and z values out. TestParseInput input(program); ASSERT_FALSE(input.has_error()); input.parsed()->Execute(setup.scope(), &err); ASSERT_FALSE(err.has_error()) << err.message(); EXPECT_EQ("target, 1, 2\n", setup.print_output()); setup.print_output().clear(); } { TestWithScope setup; // Test that the same input but forwarding a variable with the name of // something in the given scope throws an error rather than clobbering it. // This uses the same known-good program as before, but adds another // variable in the scope before it. TestParseInput clobber("x = 1\n" + program); ASSERT_FALSE(clobber.has_error()); clobber.parsed()->Execute(setup.scope(), &err); ASSERT_TRUE(err.has_error()); // Should thow a clobber error. EXPECT_EQ("Clobbering existing value.", err.message()); } } TEST_F(FunctionForwardVariablesFromTest, LiteralList) { TestWithScope setup; // Forwards all variables from a literal scope into another scope definition. TestParseInput input( "a = {\n" " forward_variables_from({x = 1 y = 2}, \"*\")\n" " z = 3\n" "}\n" "print(\"${a.x} ${a.y} ${a.z}\")\n"); ASSERT_FALSE(input.has_error()); Err err; input.parsed()->Execute(setup.scope(), &err); ASSERT_FALSE(err.has_error()) << err.message(); EXPECT_EQ("1 2 3\n", setup.print_output()); setup.print_output().clear(); } TEST_F(FunctionForwardVariablesFromTest, ListWithExclusion) { TestWithScope setup; // Defines a template and copy the two x and y, and z values out. TestParseInput input( "template(\"a\") {\n" " forward_variables_from(invoker, [\"x\", \"y\", \"z\"], [\"z\"])\n" " assert(!defined(z))\n" // "z" should still be undefined. " print(\"$target_name, $x, $y\")\n" "}\n" "a(\"target\") {\n" " x = 1\n" " y = 2\n" " z = 3\n" " print(\"$z\")\n" "}\n"); ASSERT_FALSE(input.has_error()); Err err; input.parsed()->Execute(setup.scope(), &err); ASSERT_FALSE(err.has_error()) << err.message(); EXPECT_EQ("3\ntarget, 1, 2\n", setup.print_output()); setup.print_output().clear(); } TEST_F(FunctionForwardVariablesFromTest, ErrorCases) { TestWithScope setup; // Type check the source scope. TestParseInput invalid_source( "template(\"a\") {\n" " forward_variables_from(42, [\"x\"])\n" " print(\"$target_name\")\n" // Prevent unused var error. "}\n" "a(\"target\") {\n" "}\n"); ASSERT_FALSE(invalid_source.has_error()); Err err; invalid_source.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("This is not a scope.", err.message()); // Type check the list. We need to use a new template name each time since // all of these invocations are executing in sequence in the same scope. TestParseInput invalid_list( "template(\"b\") {\n" " forward_variables_from(invoker, 42)\n" " print(\"$target_name\")\n" "}\n" "b(\"target\") {\n" "}\n"); ASSERT_FALSE(invalid_list.has_error()); err = Err(); invalid_list.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("Not a valid list of variables to copy.", err.message()); // Type check the exclusion list. TestParseInput invalid_exclusion_list( "template(\"c\") {\n" " forward_variables_from(invoker, \"*\", 42)\n" " print(\"$target_name\")\n" "}\n" "c(\"target\") {\n" "}\n"); ASSERT_FALSE(invalid_exclusion_list.has_error()); err = Err(); invalid_exclusion_list.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("Not a valid list of variables to exclude.", err.message()); // Programmatic values should error. TestParseInput prog( "template(\"d\") {\n" " forward_variables_from(invoker, [\"root_out_dir\"])\n" " print(\"$target_name\")\n" "}\n" "d(\"target\") {\n" "}\n"); ASSERT_FALSE(prog.has_error()); err = Err(); prog.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("This value can't be forwarded.", err.message()); // Not enough arguments. TestParseInput not_enough_arguments( "template(\"e\") {\n" " forward_variables_from(invoker)\n" " print(\"$target_name\")\n" "}\n" "e(\"target\") {\n" "}\n"); ASSERT_FALSE(not_enough_arguments.has_error()); err = Err(); not_enough_arguments.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("Wrong number of arguments.", err.message()); // Too many arguments. TestParseInput too_many_arguments( "template(\"f\") {\n" " forward_variables_from(invoker, \"*\", [], [])\n" " print(\"$target_name\")\n" "}\n" "f(\"target\") {\n" "}\n"); ASSERT_FALSE(too_many_arguments.has_error()); err = Err(); too_many_arguments.parsed()->Execute(setup.scope(), &err); EXPECT_TRUE(err.has_error()); EXPECT_EQ("Wrong number of arguments.", err.message()); } TEST_F(FunctionForwardVariablesFromTest, Star) { TestWithScope setup; // Defines a template and copy the two x and y values out. The "*" behavior // should clobber existing variables with the same name. TestParseInput input( "template(\"a\") {\n" " x = 1000000\n" // Should be clobbered. " forward_variables_from(invoker, \"*\")\n" " print(\"$target_name, $x, $y\")\n" "}\n" "a(\"target\") {\n" " x = 1\n" " y = 2\n" "}\n"); ASSERT_FALSE(input.has_error()); Err err; input.parsed()->Execute(setup.scope(), &err); ASSERT_FALSE(err.has_error()) << err.message(); EXPECT_EQ("target, 1, 2\n", setup.print_output()); setup.print_output().clear(); } TEST_F(FunctionForwardVariablesFromTest, StarWithExclusion) { TestWithScope setup; // Defines a template and copy all values except z value. The "*" behavior // should clobber existing variables with the same name. TestParseInput input( "template(\"a\") {\n" " x = 1000000\n" // Should be clobbered. " forward_variables_from(invoker, \"*\", [\"z\"])\n" " print(\"$target_name, $x, $y\")\n" "}\n" "a(\"target\") {\n" " x = 1\n" " y = 2\n" " z = 3\n" " print(\"$z\")\n" "}\n"); ASSERT_FALSE(input.has_error()); Err err; input.parsed()->Execute(setup.scope(), &err); ASSERT_FALSE(err.has_error()) << err.message(); EXPECT_EQ("3\ntarget, 1, 2\n", setup.print_output()); setup.print_output().clear(); }