910e62b5创建于 1月15日历史提交
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "tools/json_schema_compiler/test/error_generation.h"

#include <memory>
#include <vector>

#include "base/json/json_writer.h"
#include "base/logging.h"
#include "base/strings/utf_string_conversions.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "tools/json_schema_compiler/test/test_util.h"

namespace errors = test::api::error_generation;
using base::Value;
using json_schema_compiler::test_util::Dictionary;
using json_schema_compiler::test_util::List;

template <typename T, typename Value>
std::u16string GetPopulateError(const Value& value) {
  return T::FromValue(value).error_or(std::u16string());
}

testing::AssertionResult EqualsUtf16(const std::string& expected,
                                     const std::u16string& actual) {
  if (base::ASCIIToUTF16(expected) == actual)
    return testing::AssertionSuccess();
  return testing::AssertionFailure() << "\n    actual:     " << actual
                                     << "\n    expected:   " << expected;
}

// GenerateTypePopulate errors

TEST(JsonSchemaCompilerErrorTest, RequiredPropertyPopulate) {
  {
    base::Value value = Dictionary("string", Value("bling"));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::TestType>(value.GetDict())));
  }
}

TEST(JsonSchemaCompilerErrorTest, UnexpectedTypePopulation) {
  {
    base::Value value(Value::Type::LIST);
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::ChoiceType::Integers>(value)));
  }
  {
    base::Value value(Value::Type::BINARY);
    EXPECT_TRUE(
        EqualsUtf16("expected integers or integer, got binary",
                    GetPopulateError<errors::ChoiceType::Integers>(value)));
  }
}

// GenerateTypePopulateProperty errors

TEST(JsonSchemaCompilerErrorTest, TypeIsRequired) {
  {
    base::Value value = Dictionary("integers", Value(5));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::ChoiceType>(value.GetDict())));
  }
  {
    base::Value::Dict value;
    EXPECT_TRUE(EqualsUtf16("'integers' is required",
                            GetPopulateError<errors::ChoiceType>(value)));
  }
}

// GenerateParamsCheck errors

TEST(JsonSchemaCompilerErrorTest, TooManyParameters) {
  {
    base::Value::List params_value;
    params_value.Append(5);
    EXPECT_TRUE(errors::TestFunction::Params::Create(params_value).has_value());
  }
  {
    base::Value::List params_value;
    params_value.Append(5);
    params_value.Append(5);
    EXPECT_TRUE(EqualsUtf16("expected 1 arguments, got 2",
                            errors::TestFunction::Params::Create(params_value)
                                .error_or(std::u16string())));
  }
}

// GenerateFunctionParamsCreate errors

TEST(JsonSchemaCompilerErrorTest, ParamIsRequired) {
  {
    base::Value::List params_value;
    params_value.Append(5);
    EXPECT_TRUE(errors::TestFunction::Params::Create(params_value).has_value());
  }
  {
    base::Value::List params_value;
    params_value.Append(base::Value());
    EXPECT_TRUE(EqualsUtf16("'num' is required",
                            errors::TestFunction::Params::Create(params_value)
                                .error_or(std::u16string())));
  }
}

// GeneratePopulateVariableFromValue errors

TEST(JsonSchemaCompilerErrorTest, WrongPropertyValueType) {
  {
    base::Value value = Dictionary("string", Value("yes"));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::TestType>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("string", Value(1.1));
    EXPECT_TRUE(
        EqualsUtf16("'string': expected string, got double",
                    GetPopulateError<errors::TestType>(value.GetDict())));
  }
}

TEST(JsonSchemaCompilerErrorTest, WrongParameterCreationType) {
  {
    std::u16string error;
    base::Value::List params_value;
    params_value.Append("Yeah!");
    EXPECT_TRUE(errors::TestString::Params::Create(params_value).has_value());
  }
  {
    base::Value::List params_value;
    params_value.Append(5);
    std::u16string error;
    EXPECT_TRUE(
        EqualsUtf16("'paramObject': expected dictionary, got integer",
                    errors::TestTypeInObject::Params::Create(params_value)
                        .error_or(std::u16string())));
  }
}

TEST(JsonSchemaCompilerErrorTest, WrongTypeValueType) {
  {
    base::Value::Dict value;
    EXPECT_TRUE(EqualsUtf16("", GetPopulateError<errors::ObjectType>(value)));
  }
  {
    base::Value value = Dictionary("otherType", Value(1.1));
    auto out = errors::ObjectType::FromValue(value.GetDict());
    ASSERT_FALSE(out.has_value());
    EXPECT_TRUE(EqualsUtf16("'otherType': expected dictionary, got double",
                            out.error()));
  }
}

TEST(JsonSchemaCompilerErrorTest, UnableToPopulateArray) {
  {
    base::Value params_value = List(Value(5));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::ChoiceType::Integers>(params_value)));
  }
  {
    base::Value params_value = List(Value(5), Value(false));
    EXPECT_TRUE(EqualsUtf16(
        "Error at key 'integers': Parsing array failed at index 1: expected "
        "integer, got boolean",
        GetPopulateError<errors::ChoiceType::Integers>(params_value)));
  }
}

TEST(JsonSchemaCompilerErrorTest, BinaryTypeExpected) {
  {
    base::Value value = Dictionary("data", Value(Value::Type::BINARY));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::BinaryData>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("data", Value(1.1));
    EXPECT_TRUE(
        EqualsUtf16("'data': expected binary, got double",
                    GetPopulateError<errors::BinaryData>(value.GetDict())));
  }
}

TEST(JsonSchemaCompilerErrorTest, ListExpected) {
  {
    base::Value value =
        Dictionary("TheArray", base::Value(base::Value::Type::LIST));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::ArrayObject>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("TheArray", Value(5));
    EXPECT_TRUE(
        EqualsUtf16("'TheArray': expected list, got integer",
                    GetPopulateError<errors::ArrayObject>(value.GetDict())));
  }
}

// GenerateStringToEnumConversion errors

TEST(JsonSchemaCompilerErrorTest, BadEnumValue) {
  {
    base::Value value = Dictionary("enumeration", Value("one"));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::HasEnumeration>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("enumeration", Value("bad sauce"));
    EXPECT_TRUE(
        EqualsUtf16("'Enumeration': expected \"one\" or \"two\" "
                    "or \"three\", got \"bad sauce\"",
                    GetPopulateError<errors::HasEnumeration>(value.GetDict())));
  }
}

TEST(JsonSchemaCompilerErrorTest, ErrorOnOptionalFailure) {
  {
    base::Value value = Dictionary("string", Value("bling"));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::OptionalTestType>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("string", Value(1));

    auto out = errors::OptionalTestType::FromValue(value.GetDict());
    ASSERT_FALSE(out.has_value());
    EXPECT_TRUE(
        EqualsUtf16("'string': expected string, got integer", out.error()));
  }
}

TEST(JsonSchemaCompilerErrorTest, OptionalBinaryTypeFailure) {
  {
    base::Value value = Dictionary("data", Value(Value::Type::BINARY));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::OptionalBinaryData>(value.GetDict())));
  }
  {
    // There's a bug with silent failures if the key doesn't exist.
    base::Value value = Dictionary("data", Value(1));

    auto out = errors::OptionalBinaryData::FromValue(value.GetDict());
    ASSERT_FALSE(out.has_value());
    EXPECT_TRUE(
        EqualsUtf16("'data': expected binary, got integer", out.error()));
  }
}

TEST(JsonSchemaCompilerErrorTest, OptionalArrayTypeFailure) {
  {
    base::Value value =
        Dictionary("TheArray", base::Value(base::Value::Type::LIST));
    EXPECT_TRUE(EqualsUtf16(
        "", GetPopulateError<errors::ArrayObject>(value.GetDict())));
  }
  {
    base::Value value = Dictionary("TheArray", Value(5));
    auto out = errors::ArrayObject::FromValue(value.GetDict());
    EXPECT_FALSE(out.has_value());
    EXPECT_TRUE(
        EqualsUtf16("'TheArray': expected list, got integer", out.error()));
  }
}

TEST(JsonSchemaCompilerErrorTest, OptionalUnableToPopulateArray) {
  {
    base::Value params_value = List(Value(5));
    EXPECT_TRUE(EqualsUtf16(
        "",
        GetPopulateError<errors::OptionalChoiceType::Integers>(params_value)));
  }
  {
    base::Value params_value = List(Value(5), Value(false));
    auto out = errors::OptionalChoiceType::Integers::FromValue(params_value);
    EXPECT_FALSE(out.has_value());
    EXPECT_TRUE(
        EqualsUtf16("Error at key 'integers': Parsing array failed at index 1: "
                    "expected integer, got boolean",
                    out.error()));
  }
}

TEST(JsonSchemaCompilerErrorTest, TooManyKeys) {
  {
    base::Value value = Dictionary("string", Value("yes"));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::TestType>(value.GetDict())));
  }
  {
    // We simply ignore extra keys.
    base::Value value =
        Dictionary("string", Value("yes"), "ohno", Value("many values"));
    EXPECT_TRUE(
        EqualsUtf16("", GetPopulateError<errors::TestType>(value.GetDict())));
  }
}