/*
 * 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