/* * Copyright (C) 2015, The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <iostream> #include <string> #include <vector> #include <gtest/gtest.h> #include "options.h" using std::cerr; using std::endl; using std::string; using std::unique_ptr; using std::vector; namespace android { namespace aidl { namespace { const char kPreprocessCommandOutputFile[] = "output_file_name"; const char kPreprocessCommandInput1[] = "input1"; const char kPreprocessCommandInput2[] = "input2"; const char kPreprocessCommandInput3[] = "input3"; const char* kPreprocessCommand[] = { "aidl", "--preprocess", kPreprocessCommandOutputFile, kPreprocessCommandInput1, kPreprocessCommandInput2, kPreprocessCommandInput3, nullptr, }; const char kCompileCommandInput[] = "directory/ITool.aidl"; const char kCompileCommandIncludePath[] = "-Iinclude_path"; const char* kCompileJavaCommand[] = { "aidl", "-b", kCompileCommandIncludePath, kCompileCommandInput, nullptr, }; const char kCompileCommandJavaOutput[] = "directory/ITool.java"; const char kCompileDepFileNinja[] = "-ninja"; const char* kCompileJavaCommandNinja[] = { "aidl", "-b", kCompileDepFileNinja, kCompileCommandIncludePath, kCompileCommandInput, nullptr, }; const char kCompileDepFile[] = "-doutput.deps"; const char kCompileCommandHeaderDir[] = "output/dir"; const char kCompileCommandCppOutput[] = "some/file.cpp"; const char* kCompileCppCommand[] = { "aidl-cpp", kCompileCommandIncludePath, kCompileDepFile, kCompileCommandInput, kCompileCommandHeaderDir, kCompileCommandCppOutput, nullptr, }; const char* kCompileCppCommandNinja[] = { "aidl-cpp", kCompileCommandIncludePath, kCompileDepFile, kCompileDepFileNinja, kCompileCommandInput, kCompileCommandHeaderDir, kCompileCommandCppOutput, nullptr, }; template <typename T> unique_ptr<T> GetOptions(const char* command[]) { int argc = 0; const char** command_part = command; for (; *command_part; ++argc, ++command_part) {} unique_ptr<T> options(T::Parse(argc, command)); if (!options) { cerr << "Failed to parse command line:"; for (int i = 0; i < argc; ++i) { cerr << " " << command[i]; cerr << endl; } } EXPECT_NE(options, nullptr) << "Failed to parse options!"; return options; } } // namespace TEST(JavaOptionsTests, ParsesPreprocess) { unique_ptr<JavaOptions> options = GetOptions<JavaOptions>(kPreprocessCommand); EXPECT_EQ(JavaOptions::PREPROCESS_AIDL, options->task); EXPECT_EQ(false, options->fail_on_parcelable_); EXPECT_EQ(0u, options->import_paths_.size()); EXPECT_EQ(0u, options->preprocessed_files_.size()); EXPECT_EQ(string{}, options->input_file_name_); EXPECT_EQ(string{kPreprocessCommandOutputFile}, options->output_file_name_); EXPECT_EQ(false, options->auto_dep_file_); const vector<string> expected_input{kPreprocessCommandInput1, kPreprocessCommandInput2, kPreprocessCommandInput3}; EXPECT_EQ(expected_input, options->files_to_preprocess_); } TEST(JavaOptionsTests, ParsesCompileJava) { unique_ptr<JavaOptions> options = GetOptions<JavaOptions>(kCompileJavaCommand); EXPECT_EQ(JavaOptions::COMPILE_AIDL_TO_JAVA, options->task); EXPECT_EQ(true, options->fail_on_parcelable_); EXPECT_EQ(1u, options->import_paths_.size()); EXPECT_EQ(0u, options->preprocessed_files_.size()); EXPECT_EQ(string{kCompileCommandInput}, options->input_file_name_); EXPECT_EQ(string{kCompileCommandJavaOutput}, options->output_file_name_); EXPECT_EQ(false, options->auto_dep_file_); EXPECT_EQ(false, options->DependencyFileNinja()); } TEST(JavaOptionsTests, ParsesCompileJavaNinja) { unique_ptr<JavaOptions> options = GetOptions<JavaOptions>(kCompileJavaCommandNinja); EXPECT_EQ(JavaOptions::COMPILE_AIDL_TO_JAVA, options->task); EXPECT_EQ(true, options->fail_on_parcelable_); EXPECT_EQ(1u, options->import_paths_.size()); EXPECT_EQ(0u, options->preprocessed_files_.size()); EXPECT_EQ(string{kCompileCommandInput}, options->input_file_name_); EXPECT_EQ(string{kCompileCommandJavaOutput}, options->output_file_name_); EXPECT_EQ(false, options->auto_dep_file_); EXPECT_EQ(true, options->DependencyFileNinja()); } TEST(CppOptionsTests, ParsesCompileCpp) { unique_ptr<CppOptions> options = GetOptions<CppOptions>(kCompileCppCommand); ASSERT_EQ(1u, options->import_paths_.size()); EXPECT_EQ(string{kCompileCommandIncludePath}.substr(2), options->import_paths_[0]); EXPECT_EQ(string{kCompileDepFile}.substr(2), options->dep_file_name_); EXPECT_EQ(false, options->DependencyFileNinja()); EXPECT_EQ(kCompileCommandInput, options->InputFileName()); EXPECT_EQ(kCompileCommandHeaderDir, options->OutputHeaderDir()); EXPECT_EQ(kCompileCommandCppOutput, options->OutputCppFilePath()); } TEST(CppOptionsTests, ParsesCompileCppNinja) { unique_ptr<CppOptions> options = GetOptions<CppOptions>(kCompileCppCommandNinja); ASSERT_EQ(1u, options->import_paths_.size()); EXPECT_EQ(string{kCompileCommandIncludePath}.substr(2), options->import_paths_[0]); EXPECT_EQ(string{kCompileDepFile}.substr(2), options->dep_file_name_); EXPECT_EQ(true, options->DependencyFileNinja()); EXPECT_EQ(kCompileCommandInput, options->InputFileName()); EXPECT_EQ(kCompileCommandHeaderDir, options->OutputHeaderDir()); EXPECT_EQ(kCompileCommandCppOutput, options->OutputCppFilePath()); } TEST(OptionsTests, EndsWith) { EXPECT_TRUE(EndsWith("foo", "")); EXPECT_TRUE(EndsWith("foo", "o")); EXPECT_TRUE(EndsWith("foo", "foo")); EXPECT_FALSE(EndsWith("foo", "fooo")); EXPECT_FALSE(EndsWith("", "o")); EXPECT_TRUE(EndsWith("", "")); } TEST(OptionsTests, ReplaceSuffix) { struct test_case_t { const char* input; const char* old_suffix; const char* new_suffix; const char* result; }; const size_t kNumCases = 3; test_case_t kTestInput[kNumCases] = { {"foo.bar", "bar", "foo", "foo.foo"}, {"whole", "whole", "new", "new"}, {"", "", "", ""}, }; for (const auto& test_case : kTestInput) { string mutated = test_case.input; EXPECT_TRUE(ReplaceSuffix(test_case.old_suffix, test_case.new_suffix, &mutated)); EXPECT_EQ(mutated, test_case.result); } } } // namespace android } // namespace aidl