| // Protocol Buffers - Google's data interchange format |
| // Copyright 2014 Google Inc. All rights reserved. |
| // https://developers.google.com/protocol-buffers/ |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are |
| // met: |
| // |
| // * Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above |
| // copyright notice, this list of conditions and the following disclaimer |
| // in the documentation and/or other materials provided with the |
| // distribution. |
| // * Neither the name of Google Inc. nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include "google/protobuf/compiler/objectivec/helpers.h" |
| #include "google/protobuf/io/zero_copy_stream_impl_lite.h" |
| #include <gtest/gtest.h> |
| |
| namespace google { |
| namespace protobuf { |
| namespace compiler { |
| namespace objectivec { |
| namespace { |
| |
| TEST(ObjCHelper, TextFormatDecodeData_DecodeDataForString_RawStrings) { |
| std::string input_for_decode("abcdefghIJ"); |
| std::string desired_output_for_decode; |
| std::string expected; |
| std::string result; |
| |
| // Different data, can't transform. |
| |
| desired_output_for_decode = "zbcdefghIJ"; |
| expected = std::string("\0zbcdefghIJ\0", 12); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| desired_output_for_decode = "abcdezghIJ"; |
| expected = std::string("\0abcdezghIJ\0", 12); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| // Shortened data, can't transform. |
| |
| desired_output_for_decode = "abcdefghI"; |
| expected = std::string("\0abcdefghI\0", 11); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| // Extra data, can't transform. |
| |
| desired_output_for_decode = "abcdefghIJz"; |
| expected = std::string("\0abcdefghIJz\0", 13); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| } |
| |
| TEST(ObjCHelper, TextFormatDecodeData_DecodeDataForString_ByteCodes) { |
| std::string input_for_decode("abcdefghIJ"); |
| std::string desired_output_for_decode; |
| std::string expected; |
| std::string result; |
| |
| desired_output_for_decode = "abcdefghIJ"; |
| expected = std::string("\x0A\x0", 2); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| desired_output_for_decode = "_AbcdefghIJ"; |
| expected = std::string("\xCA\x0", 2); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| desired_output_for_decode = "ABCD__EfghI_j"; |
| expected = std::string("\x64\x80\xC5\xA1\x0", 5); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| |
| // Long name so multiple decode ops are needed. |
| |
| // clang-format off |
| input_for_decode = |
| "longFieldNameIsLooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong1000"; |
| desired_output_for_decode = |
| "long_field_name_is_looooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong_1000"; |
| // clang-format on |
| expected = std::string("\x04\xA5\xA4\xA2\xBF\x1F\x0E\x84\x0", 9); |
| result = TextFormatDecodeData::DecodeDataForString(input_for_decode, |
| desired_output_for_decode); |
| EXPECT_EQ(expected, result); |
| } |
| |
| // Death tests do not work on Windows as of yet. |
| #ifdef PROTOBUF_HAS_DEATH_TEST |
| TEST(ObjCHelperDeathTest, TextFormatDecodeData_DecodeDataForString_Failures) { |
| // Empty inputs. |
| |
| EXPECT_EXIT(TextFormatDecodeData::DecodeDataForString("", ""), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| EXPECT_EXIT(TextFormatDecodeData::DecodeDataForString("a", ""), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| EXPECT_EXIT(TextFormatDecodeData::DecodeDataForString("", "a"), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| |
| // Null char in the string. |
| |
| std::string str_with_null_char("ab\0c", 4); |
| EXPECT_EXIT( |
| TextFormatDecodeData::DecodeDataForString(str_with_null_char, "def"), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got a null char in a string for making TextFormat data, input:"); |
| EXPECT_EXIT( |
| TextFormatDecodeData::DecodeDataForString("def", str_with_null_char), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got a null char in a string for making TextFormat data, input:"); |
| } |
| #endif // PROTOBUF_HAS_DEATH_TEST |
| |
| TEST(ObjCHelper, TextFormatDecodeData_RawStrings) { |
| TextFormatDecodeData decode_data; |
| |
| // Different data, can't transform. |
| decode_data.AddString(1, "abcdefghIJ", "zbcdefghIJ"); |
| decode_data.AddString(3, "abcdefghIJ", "abcdezghIJ"); |
| // Shortened data, can't transform. |
| decode_data.AddString(2, "abcdefghIJ", "abcdefghI"); |
| // Extra data, can't transform. |
| decode_data.AddString(4, "abcdefghIJ", "abcdefghIJz"); |
| |
| EXPECT_EQ(4, decode_data.num_entries()); |
| |
| uint8_t expected_data[] = { |
| // clang-format off |
| 0x4, |
| 0x1, 0x0, 'z', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'I', 'J', 0x0, |
| 0x3, 0x0, 'a', 'b', 'c', 'd', 'e', 'z', 'g', 'h', 'I', 'J', 0x0, |
| 0x2, 0x0, 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'I', 0x0, |
| 0x4, 0x0, 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'I', 'J', 'z', 0x0, |
| // clang-format on |
| }; |
| std::string expected((const char*)expected_data, sizeof(expected_data)); |
| |
| EXPECT_EQ(expected, decode_data.Data()); |
| } |
| |
| TEST(ObjCHelper, TextFormatDecodeData_ByteCodes) { |
| TextFormatDecodeData decode_data; |
| |
| decode_data.AddString(1, "abcdefghIJ", "abcdefghIJ"); |
| decode_data.AddString(3, "abcdefghIJ", "_AbcdefghIJ"); |
| decode_data.AddString(2, "abcdefghIJ", "Abcd_EfghIJ"); |
| decode_data.AddString(4, "abcdefghIJ", "ABCD__EfghI_j"); |
| // clang-format off |
| decode_data.AddString(1000, |
| "longFieldNameIsLooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong1000", |
| "long_field_name_is_looooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong_1000"); |
| // clang-format on |
| |
| EXPECT_EQ(5, decode_data.num_entries()); |
| |
| // clang-format off |
| uint8_t expected_data[] = { |
| 0x5, |
| // All as is (00 op) |
| 0x1, 0x0A, 0x0, |
| // Underscore, upper + 9 (10 op) |
| 0x3, 0xCA, 0x0, |
| // Upper + 3 (10 op), underscore, upper + 5 (10 op) |
| 0x2, 0x44, 0xC6, 0x0, |
| // All Upper for 4 (11 op), underscore, underscore, upper + 5 (10 op), |
| // underscore, lower + 0 (01 op) |
| 0x4, 0x64, 0x80, 0xC5, 0xA1, 0x0, |
| // 2 byte key: as is + 3 (00 op), underscore, lower + 4 (01 op), |
| // underscore, lower + 3 (01 op), underscore, lower + 1 (01 op), |
| // underscore, lower + 30 (01 op), as is + 30 (00 op), as is + 13 (00 |
| // op), |
| // underscore, as is + 3 (00 op) |
| 0xE8, 0x07, 0x04, 0xA5, 0xA4, 0xA2, 0xBF, 0x1F, 0x0E, 0x84, 0x0, |
| }; |
| // clang-format on |
| std::string expected((const char*)expected_data, sizeof(expected_data)); |
| |
| EXPECT_EQ(expected, decode_data.Data()); |
| } |
| |
| |
| // Death tests do not work on Windows as of yet. |
| #ifdef PROTOBUF_HAS_DEATH_TEST |
| TEST(ObjCHelperDeathTest, TextFormatDecodeData_Failures) { |
| TextFormatDecodeData decode_data; |
| |
| // Empty inputs. |
| |
| EXPECT_EXIT(decode_data.AddString(1, "", ""), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| EXPECT_EXIT(decode_data.AddString(1, "a", ""), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| EXPECT_EXIT(decode_data.AddString(1, "", "a"), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got empty string for making TextFormat data, input:"); |
| |
| // Null char in the string. |
| |
| std::string str_with_null_char("ab\0c", 4); |
| EXPECT_EXIT( |
| decode_data.AddString(1, str_with_null_char, "def"), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got a null char in a string for making TextFormat data, input:"); |
| EXPECT_EXIT( |
| decode_data.AddString(1, "def", str_with_null_char), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: got a null char in a string for making TextFormat data, input:"); |
| |
| // Duplicate keys |
| |
| decode_data.AddString(1, "abcdefghIJ", "abcdefghIJ"); |
| decode_data.AddString(3, "abcdefghIJ", "_AbcdefghIJ"); |
| decode_data.AddString(2, "abcdefghIJ", "Abcd_EfghIJ"); |
| EXPECT_EXIT(decode_data.AddString(2, "xyz", "x_yz"), |
| ::testing::KilledBySignal(SIGABRT), |
| "error: duplicate key \\(2\\) making TextFormat data, input:"); |
| } |
| #endif // PROTOBUF_HAS_DEATH_TEST |
| |
| class TestLineCollector : public LineConsumer { |
| public: |
| explicit TestLineCollector(std::vector<std::string>* inout_lines, |
| const std::string* reject_line = nullptr, |
| bool skip_msg = false) |
| : lines_(inout_lines), reject_(reject_line), skip_msg_(skip_msg) {} |
| |
| bool ConsumeLine(const absl::string_view& line, std::string* out_error) override { |
| if (reject_ && *reject_ == line) { |
| if (!skip_msg_) { |
| *out_error = std::string("Rejected '") + *reject_ + "'"; |
| } |
| return false; |
| } |
| if (lines_) { |
| lines_->emplace_back(line); |
| } |
| return true; |
| } |
| |
| private: |
| std::vector<std::string>* lines_; |
| const std::string* reject_; |
| bool skip_msg_; |
| }; |
| |
| const int kBlockSizes[] = {-1, 1, 2, 5, 64}; |
| const int kBlockSizeCount = ABSL_ARRAYSIZE(kBlockSizes); |
| |
| TEST(ObjCHelper, ParseSimple_BasicsSuccess) { |
| const std::vector<std::pair<std::string, std::vector<std::string>>> tests = { |
| {"", {}}, |
| {"a", {"a"}}, |
| {"a c", {"a c"}}, |
| {" a c ", {"a c"}}, |
| {"\ta c ", {"a c"}}, |
| {"abc\n", {"abc"}}, |
| {"abc\nd f", {"abc", "d f"}}, |
| {"\n abc \n def \n\n", {"abc", "def"}}, |
| }; |
| |
| for (const auto& test : tests) { |
| for (int i = 0; i < kBlockSizeCount; i++) { |
| io::ArrayInputStream input(test.first.data(), test.first.size(), kBlockSizes[i]); |
| std::string err_str; |
| std::vector<std::string> lines; |
| TestLineCollector collector(&lines); |
| EXPECT_TRUE(ParseSimpleStream(input, "dummy", &collector, &err_str)); |
| EXPECT_EQ(lines, test.second); |
| EXPECT_TRUE(err_str.empty()); |
| } |
| } |
| } |
| |
| TEST(ObjCHelper, ParseSimple_DropsComments) { |
| const std::vector<std::pair<std::string, std::vector<std::string>>> tests = { |
| {"# nothing", {}}, |
| {"#", {}}, |
| {"##", {}}, |
| {"\n# nothing\n", {}}, |
| {"a # same line", {"a"}}, |
| {"a # same line\n", {"a"}}, |
| {"a\n# line\nc", {"a", "c"}}, |
| {"# n o t # h i n g #", {}}, |
| {"## n o # t h i n g #", {}}, |
| {"a# n o t # h i n g #", {"a"}}, |
| {"a\n## n o # t h i n g #", {"a"}}, |
| }; |
| |
| for (const auto& test : tests) { |
| for (int i = 0; i < kBlockSizeCount; i++) { |
| io::ArrayInputStream input(test.first.data(), test.first.size(), kBlockSizes[i]); |
| std::string err_str; |
| std::vector<std::string> lines; |
| TestLineCollector collector(&lines); |
| EXPECT_TRUE(ParseSimpleStream(input, "dummy", &collector, &err_str)); |
| EXPECT_EQ(lines, test.second); |
| EXPECT_TRUE(err_str.empty()); |
| } |
| } |
| } |
| |
| TEST(ObjCHelper, ParseSimple_RejectLines) { |
| const std::vector<std::tuple<std::string, std::string, int>> tests = { |
| std::make_tuple("a\nb\nc", "a", 1), |
| std::make_tuple("a\nb\nc", "b", 2), |
| std::make_tuple("a\nb\nc", "c", 3), |
| std::make_tuple("a\nb\nc\n", "c", 3), |
| }; |
| |
| for (const auto& test : tests) { |
| for (int i = 0; i < kBlockSizeCount; i++) { |
| io::ArrayInputStream input(std::get<0>(test).data(), std::get<0>(test).size(), |
| kBlockSizes[i]); |
| std::string err_str; |
| TestLineCollector collector(nullptr, &std::get<1>(test)); |
| EXPECT_FALSE(ParseSimpleStream(input, "dummy", &collector, &err_str)); |
| std::string expected_err = |
| absl::StrCat("error: dummy Line ", std::get<2>(test), ", Rejected '", std::get<1>(test), "'"); |
| EXPECT_EQ(err_str, expected_err); |
| } |
| } |
| } |
| |
| TEST(ObjCHelper, ParseSimple_RejectLinesNoMessage) { |
| const std::vector<std::tuple<std::string, std::string, int>> tests = { |
| std::make_tuple("a\nb\nc", "a", 1), |
| std::make_tuple("a\nb\nc", "b", 2), |
| std::make_tuple("a\nb\nc", "c", 3), |
| std::make_tuple("a\nb\nc\n", "c", 3), |
| }; |
| |
| for (const auto& test : tests) { |
| for (int i = 0; i < kBlockSizeCount; i++) { |
| io::ArrayInputStream input(std::get<0>(test).data(), std::get<0>(test).size(), |
| kBlockSizes[i]); |
| std::string err_str; |
| TestLineCollector collector(nullptr, &std::get<1>(test), true /* skip msg */); |
| EXPECT_FALSE(ParseSimpleStream(input, "dummy", &collector, &err_str)); |
| std::string expected_err = |
| absl::StrCat("error: dummy Line ", std::get<2>(test), |
| ", ConsumeLine failed without setting an error."); |
| EXPECT_EQ(err_str, expected_err); |
| } |
| } |
| } |
| |
| TEST(ObjCHelper, IsRetainedName) { |
| EXPECT_TRUE(IsRetainedName("new")); |
| EXPECT_TRUE(IsRetainedName("alloc")); |
| EXPECT_TRUE(IsRetainedName("copy")); |
| EXPECT_TRUE(IsRetainedName("mutableCopy")); |
| EXPECT_TRUE(IsRetainedName("newFoo")); |
| EXPECT_TRUE(IsRetainedName("allocFoo")); |
| EXPECT_TRUE(IsRetainedName("copyFoo")); |
| EXPECT_TRUE(IsRetainedName("mutableCopyFoo")); |
| EXPECT_TRUE(IsRetainedName("new_foo")); |
| EXPECT_TRUE(IsRetainedName("alloc_foo")); |
| EXPECT_TRUE(IsRetainedName("copy_foo")); |
| EXPECT_TRUE(IsRetainedName("mutableCopy_foo")); |
| |
| EXPECT_FALSE(IsRetainedName("")); |
| EXPECT_FALSE(IsRetainedName("ne")); |
| EXPECT_FALSE(IsRetainedName("all")); |
| EXPECT_FALSE(IsRetainedName("co")); |
| EXPECT_FALSE(IsRetainedName("mutable")); |
| EXPECT_FALSE(IsRetainedName("New")); |
| EXPECT_FALSE(IsRetainedName("Alloc")); |
| EXPECT_FALSE(IsRetainedName("Copy")); |
| EXPECT_FALSE(IsRetainedName("MutableCopy")); |
| EXPECT_FALSE(IsRetainedName("newer")); |
| EXPECT_FALSE(IsRetainedName("alloced")); |
| EXPECT_FALSE(IsRetainedName("copying")); |
| EXPECT_FALSE(IsRetainedName("mutableCopying")); |
| |
| EXPECT_FALSE(IsRetainedName("init")); |
| EXPECT_FALSE(IsRetainedName("Create")); |
| EXPECT_FALSE(IsRetainedName("Copy")); |
| } |
| |
| TEST(ObjCHelper, IsInitName) { |
| EXPECT_TRUE(IsInitName("init")); |
| EXPECT_TRUE(IsInitName("initFoo")); |
| EXPECT_TRUE(IsInitName("init_foo")); |
| |
| EXPECT_FALSE(IsInitName("")); |
| EXPECT_FALSE(IsInitName("in")); |
| EXPECT_FALSE(IsInitName("Init")); |
| EXPECT_FALSE(IsInitName("inIt")); |
| EXPECT_FALSE(IsInitName("initial")); |
| EXPECT_FALSE(IsInitName("initiAl")); |
| EXPECT_FALSE(IsInitName("fooInit")); |
| EXPECT_FALSE(IsInitName("foo_init")); |
| |
| EXPECT_FALSE(IsInitName("new")); |
| EXPECT_FALSE(IsInitName("alloc")); |
| EXPECT_FALSE(IsInitName("copy")); |
| EXPECT_FALSE(IsInitName("mutableCopy")); |
| EXPECT_FALSE(IsInitName("Create")); |
| EXPECT_FALSE(IsInitName("Copy")); |
| } |
| |
| TEST(ObjCHelper, IsCreateName) { |
| EXPECT_TRUE(IsCreateName("Create")); |
| EXPECT_TRUE(IsCreateName("Copy")); |
| EXPECT_TRUE(IsCreateName("CreateFoo")); |
| EXPECT_TRUE(IsCreateName("CopyFoo")); |
| EXPECT_TRUE(IsCreateName("Create_foo")); |
| EXPECT_TRUE(IsCreateName("Copy_foo")); |
| EXPECT_TRUE(IsCreateName("ReCreate")); |
| EXPECT_TRUE(IsCreateName("ReCopy")); |
| EXPECT_TRUE(IsCreateName("FOOCreate")); |
| EXPECT_TRUE(IsCreateName("FOOCopy")); |
| EXPECT_TRUE(IsCreateName("CreateWithCopy")); |
| |
| EXPECT_FALSE(IsCreateName("")); |
| EXPECT_FALSE(IsCreateName("Crea")); |
| EXPECT_FALSE(IsCreateName("Co")); |
| EXPECT_FALSE(IsCreateName("create")); |
| EXPECT_FALSE(IsCreateName("recreate")); |
| EXPECT_FALSE(IsCreateName("recopy")); |
| EXPECT_FALSE(IsCreateName("ReCreated")); |
| EXPECT_FALSE(IsCreateName("ReCopying")); |
| |
| EXPECT_FALSE(IsCreateName("init")); |
| EXPECT_FALSE(IsCreateName("new")); |
| EXPECT_FALSE(IsCreateName("alloc")); |
| EXPECT_FALSE(IsCreateName("copy")); |
| EXPECT_TRUE(IsCreateName("mutableCopy")); |
| } |
| |
| // TODO(thomasvl): Should probably add some unittests for all the special cases |
| // of name mangling (class name, field name, enum names). Rather than doing |
| // this with an ObjC test in the objectivec directory, we should be able to |
| // use src/google/protobuf/compiler/importer* (like other tests) to support a |
| // virtual file system to feed in protos, once we have the Descriptor tree, the |
| // tests could use the helper methods for generating names and validate the |
| // right things are happening. |
| |
| } // namespace |
| } // namespace objectivec |
| } // namespace compiler |
| } // namespace protobuf |
| } // namespace google |