// // Copyright 2019 The ANGLE Project Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // test_utils_unittest.cpp: Unit tests for ANGLE's test utility functions #include "gtest/gtest.h" #include "common/system_utils.h" #include "tests/test_utils/runner/TestSuite.h" #include "util/Timer.h" #include "util/test_utils.h" #include "util/test_utils_unittest_helper.h" using namespace angle; namespace { #if defined(ANGLE_PLATFORM_WINDOWS) constexpr char kRunAppHelperExecutable[] = "test_utils_unittest_helper.exe"; #elif ANGLE_PLATFORM_IOS_FAMILY constexpr char kRunAppHelperExecutable[] = "../test_utils_unittest_helper.app/test_utils_unittest_helper"; #else constexpr char kRunAppHelperExecutable[] = "test_utils_unittest_helper"; #endif // Transforms various line endings into C/Unix line endings: // // - A\nB -> A\nB // - A\rB -> A\nB // - A\r\nB -> A\nB std::string NormalizeNewLines(const std::string &str) { std::string result; for (size_t i = 0; i < str.size(); ++i) { if (str[i] == '\r') { if (i + 1 < str.size() && str[i + 1] == '\n') { ++i; } result += '\n'; } else { result += str[i]; } } return result; } // Tests that Sleep() actually waits some time. TEST(TestUtils, Sleep) { Timer timer; timer.start(); angle::Sleep(500); timer.stop(); // Use a slightly fuzzy range EXPECT_GT(timer.getElapsedWallClockTime(), 0.48); } // TODO: android support. http://anglebug.com/40096559 #if defined(ANGLE_PLATFORM_ANDROID) # define MAYBE_RunApp DISABLED_RunApp # define MAYBE_RunAppAsync DISABLED_RunAppAsync # define MAYBE_RunAppAsyncRedirectStderrToStdout DISABLED_RunAppAsyncRedirectStderrToStdout // TODO: fuchsia support. http://anglebug.com/42265786 #elif defined(ANGLE_PLATFORM_FUCHSIA) # define MAYBE_RunApp DISABLED_RunApp # define MAYBE_RunAppAsync DISABLED_RunAppAsync # define MAYBE_RunAppAsyncRedirectStderrToStdout DISABLED_RunAppAsyncRedirectStderrToStdout // TODO: iOS simulator support. http://anglebug.com/42266562 #elif ANGLE_PLATFORM_IOS_FAMILY_SIMULATOR # define MAYBE_RunApp DISABLED_RunApp # define MAYBE_RunAppAsync DISABLED_RunAppAsync # define MAYBE_RunAppAsyncRedirectStderrToStdout DISABLED_RunAppAsyncRedirectStderrToStdout #else # define MAYBE_RunApp RunApp # define MAYBE_RunAppAsync RunAppAsync # define MAYBE_RunAppAsyncRedirectStderrToStdout RunAppAsyncRedirectStderrToStdout #endif // defined(ANGLE_PLATFORM_ANDROID) std::string GetTestAppExecutablePath() { std::string testExecutableName = angle::TestSuite::GetInstance()->getTestExecutableName(); std::string executablePath = angle::StripFilenameFromPath(testExecutableName); EXPECT_NE(executablePath, ""); executablePath += "/"; executablePath += kRunAppHelperExecutable; return executablePath; } // Test running an external application and receiving its output TEST(TestUtils, MAYBE_RunApp) { std::string executablePath = GetTestAppExecutablePath(); std::vector args = {executablePath.c_str(), kRunAppTestArg1, kRunAppTestArg2}; // Test that the application can be executed. { ProcessHandle process(args, ProcessOutputCapture::StdoutAndStderrSeparately); EXPECT_TRUE(process->started()); EXPECT_TRUE(process->finish()); EXPECT_TRUE(process->finished()); EXPECT_GT(process->getElapsedTimeSeconds(), 0.0); EXPECT_EQ(kRunAppTestStdout, NormalizeNewLines(process->getStdout())); EXPECT_EQ(kRunAppTestStderr, NormalizeNewLines(process->getStderr())); EXPECT_EQ(EXIT_SUCCESS, process->getExitCode()); } // Test that environment variables reach the child. { bool setEnvDone = SetEnvironmentVar(kRunAppTestEnvVarName, kRunAppTestEnvVarValue); EXPECT_TRUE(setEnvDone); ProcessHandle process(LaunchProcess(args, ProcessOutputCapture::StdoutAndStderrSeparately)); EXPECT_TRUE(process->started()); EXPECT_TRUE(process->finish()); EXPECT_GT(process->getElapsedTimeSeconds(), 0.0); EXPECT_EQ("", process->getStdout()); EXPECT_EQ(kRunAppTestEnvVarValue, NormalizeNewLines(process->getStderr())); EXPECT_EQ(EXIT_SUCCESS, process->getExitCode()); // Unset environment var. SetEnvironmentVar(kRunAppTestEnvVarName, ""); } } // Test running an external application and receiving its output asynchronously. TEST(TestUtils, MAYBE_RunAppAsync) { std::string executablePath = GetTestAppExecutablePath(); std::vector args = {executablePath.c_str(), kRunAppTestArg1, kRunAppTestArg2}; // Test that the application can be executed and the output is captured correctly. { ProcessHandle process(args, ProcessOutputCapture::StdoutAndStderrSeparately); EXPECT_TRUE(process->started()); constexpr double kTimeout = 3.0; Timer timer; timer.start(); while (!process->finished() && timer.getElapsedWallClockTime() < kTimeout) { angle::Sleep(1); } EXPECT_TRUE(process->finished()); EXPECT_GT(process->getElapsedTimeSeconds(), 0.0); EXPECT_EQ(kRunAppTestStdout, NormalizeNewLines(process->getStdout())); EXPECT_EQ(kRunAppTestStderr, NormalizeNewLines(process->getStderr())); EXPECT_EQ(EXIT_SUCCESS, process->getExitCode()); } } // Test running an external application and receiving its stdout and stderr output interleaved. TEST(TestUtils, MAYBE_RunAppAsyncRedirectStderrToStdout) { std::string executablePath = GetTestAppExecutablePath(); std::vector args = {executablePath.c_str(), kRunAppTestArg1, kRunAppTestArg2}; // Test that the application can be executed and the output is captured correctly. { ProcessHandle process(args, ProcessOutputCapture::StdoutAndStderrInterleaved); EXPECT_TRUE(process->started()); constexpr double kTimeout = 3.0; Timer timer; timer.start(); while (!process->finished() && timer.getElapsedWallClockTime() < kTimeout) { angle::Sleep(1); } EXPECT_TRUE(process->finished()); EXPECT_GT(process->getElapsedTimeSeconds(), 0.0); EXPECT_EQ(std::string(kRunAppTestStdout) + kRunAppTestStderr, NormalizeNewLines(process->getStdout())); EXPECT_EQ("", process->getStderr()); EXPECT_EQ(EXIT_SUCCESS, process->getExitCode()); } } // Verify that NumberOfProcessors returns something reasonable. TEST(TestUtils, NumberOfProcessors) { int numProcs = angle::NumberOfProcessors(); EXPECT_GT(numProcs, 0); EXPECT_LT(numProcs, 1000); } } // namespace