David Reiss | ea2cba8 | 2009-03-30 21:35:00 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Licensed to the Apache Software Foundation (ASF) under one |
| 3 | * or more contributor license agreements. See the NOTICE file |
| 4 | * distributed with this work for additional information |
| 5 | * regarding copyright ownership. The ASF licenses this file |
| 6 | * to you under the Apache License, Version 2.0 (the |
| 7 | * "License"); you may not use this file except in compliance |
| 8 | * with the License. You may obtain a copy of the License at |
| 9 | * |
| 10 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 11 | * |
| 12 | * Unless required by applicable law or agreed to in writing, |
| 13 | * software distributed under the License is distributed on an |
| 14 | * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| 15 | * KIND, either express or implied. See the License for the |
| 16 | * specific language governing permissions and limitations |
| 17 | * under the License. |
| 18 | */ |
| 19 | |
Bryan Duxbury | f7d9f8e | 2010-05-02 23:04:03 +0000 | [diff] [blame] | 20 | #include <boost/test/auto_unit_test.hpp> |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 21 | #include <iostream> |
| 22 | #include <climits> |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 23 | #include <vector> |
Roger Meier | 49ff8b1 | 2012-04-13 09:12:31 +0000 | [diff] [blame] | 24 | #include <thrift/protocol/TBinaryProtocol.h> |
James E. King, III | 82ae957 | 2017-08-05 12:23:54 -0400 | [diff] [blame] | 25 | #include <thrift/stdcxx.h> |
| 26 | #include <thrift/transport/TBufferTransports.h> |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 27 | #include "gen-cpp/ThriftTest_types.h" |
| 28 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 29 | BOOST_AUTO_TEST_SUITE(TMemoryBufferTest) |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 30 | |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 31 | using apache::thrift::protocol::TBinaryProtocol; |
| 32 | using apache::thrift::transport::TMemoryBuffer; |
| 33 | using apache::thrift::transport::TTransportException; |
James E. King, III | 82ae957 | 2017-08-05 12:23:54 -0400 | [diff] [blame] | 34 | using apache::thrift::stdcxx::shared_ptr; |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 35 | using std::cout; |
| 36 | using std::endl; |
| 37 | using std::string; |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 38 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 39 | BOOST_AUTO_TEST_CASE(test_read_write_grow) { |
Roger Meier | a6b6633 | 2015-05-15 15:21:50 +0200 | [diff] [blame] | 40 | // Added to test the fix for THRIFT-1248 |
| 41 | TMemoryBuffer uut; |
| 42 | const int maxSize = 65536; |
| 43 | uint8_t verify[maxSize]; |
| 44 | std::vector<uint8_t> buf; |
| 45 | buf.resize(maxSize); |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 46 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 47 | for (uint32_t i = 0; i < maxSize; ++i) { |
Roger Meier | a6b6633 | 2015-05-15 15:21:50 +0200 | [diff] [blame] | 48 | buf[i] = static_cast<uint8_t>(i); |
| 49 | } |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 50 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 51 | for (uint32_t i = 1; i < maxSize; i *= 2) { |
Roger Meier | a6b6633 | 2015-05-15 15:21:50 +0200 | [diff] [blame] | 52 | uut.write(&buf[0], i); |
| 53 | } |
| 54 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 55 | for (uint32_t i = 1; i < maxSize; i *= 2) { |
Roger Meier | a6b6633 | 2015-05-15 15:21:50 +0200 | [diff] [blame] | 56 | uut.read(verify, i); |
| 57 | BOOST_CHECK_EQUAL(0, ::memcmp(verify, &buf[0], i)); |
| 58 | } |
Jim King | 6077481 | 2015-05-10 08:08:18 -0400 | [diff] [blame] | 59 | } |
| 60 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 61 | BOOST_AUTO_TEST_CASE(test_roundtrip) { |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 62 | shared_ptr<TMemoryBuffer> strBuffer(new TMemoryBuffer()); |
| 63 | shared_ptr<TBinaryProtocol> binaryProtcol(new TBinaryProtocol(strBuffer)); |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 64 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 65 | thrift::test::Xtruct a; |
| 66 | a.i32_thing = 10; |
| 67 | a.i64_thing = 30; |
| 68 | a.string_thing = "holla back a"; |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 69 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 70 | a.write(binaryProtcol.get()); |
| 71 | std::string serialized = strBuffer->getBufferAsString(); |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 72 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 73 | shared_ptr<TMemoryBuffer> strBuffer2(new TMemoryBuffer()); |
| 74 | shared_ptr<TBinaryProtocol> binaryProtcol2(new TBinaryProtocol(strBuffer2)); |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 75 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 76 | strBuffer2->resetBuffer((uint8_t*)serialized.data(), static_cast<uint32_t>(serialized.length())); |
| 77 | thrift::test::Xtruct a2; |
| 78 | a2.read(binaryProtcol2.get()); |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 79 | |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 80 | BOOST_CHECK(a == a2); |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 81 | } |
| 82 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 83 | BOOST_AUTO_TEST_CASE(test_copy) { |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 84 | string* str1 = new string("abcd1234"); |
| 85 | const char* data1 = str1->data(); |
| 86 | TMemoryBuffer buf((uint8_t*)str1->data(), |
| 87 | static_cast<uint32_t>(str1->length()), |
| 88 | TMemoryBuffer::COPY); |
| 89 | delete str1; |
| 90 | string* str2 = new string("plsreuse"); |
| 91 | bool obj_reuse = (str1 == str2); |
| 92 | bool dat_reuse = (data1 == str2->data()); |
Konrad Grochowski | e9bdb41 | 2015-09-25 20:17:36 +0200 | [diff] [blame] | 93 | BOOST_TEST_MESSAGE("Object reuse: " << obj_reuse << " Data reuse: " << dat_reuse |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 94 | << ((obj_reuse && dat_reuse) ? " YAY!" : "")); |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 95 | delete str2; |
| 96 | |
| 97 | string str3 = "wxyz", str4 = "6789"; |
| 98 | buf.readAppendToString(str3, 4); |
| 99 | buf.readAppendToString(str4, INT_MAX); |
| 100 | |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 101 | BOOST_CHECK(str3 == "wxyzabcd"); |
| 102 | BOOST_CHECK(str4 == "67891234"); |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 103 | } |
| 104 | |
Konrad Grochowski | 1f6e380 | 2015-05-18 18:10:06 +0200 | [diff] [blame] | 105 | BOOST_AUTO_TEST_CASE(test_exceptions) { |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 106 | char data[] = "foo\0bar"; |
| 107 | |
| 108 | TMemoryBuffer buf1((uint8_t*)data, 7, TMemoryBuffer::OBSERVE); |
| 109 | string str = buf1.getBufferAsString(); |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 110 | BOOST_CHECK(str.length() == 7); |
| 111 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 112 | buf1.resetBuffer(); |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 113 | |
| 114 | BOOST_CHECK_THROW(buf1.write((const uint8_t*)"foo", 3), TTransportException); |
David Reiss | 2dc72c3 | 2007-08-21 23:59:34 +0000 | [diff] [blame] | 115 | |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 116 | TMemoryBuffer buf2((uint8_t*)data, 7, TMemoryBuffer::COPY); |
Claudius Heine | 5ef662b | 2015-06-24 10:03:50 +0200 | [diff] [blame] | 117 | BOOST_CHECK_NO_THROW(buf2.write((const uint8_t*)"bar", 3)); |
Konrad Grochowski | 16a23a6 | 2014-11-13 15:33:38 +0100 | [diff] [blame] | 118 | } |
David Reiss | f7baf54 | 2008-02-04 21:56:27 +0000 | [diff] [blame] | 119 | |
James E. King, III | aded00b | 2017-09-22 11:20:15 -0700 | [diff] [blame^] | 120 | BOOST_AUTO_TEST_CASE(test_default_maximum_buffer_size) |
| 121 | { |
| 122 | BOOST_CHECK_EQUAL(std::numeric_limits<uint32_t>::max(), TMemoryBuffer().getMaxBufferSize()); |
Alex Şuhan | 20116c6 | 2017-08-03 12:28:17 -0700 | [diff] [blame] | 123 | } |
James E. King, III | aded00b | 2017-09-22 11:20:15 -0700 | [diff] [blame^] | 124 | |
| 125 | BOOST_AUTO_TEST_CASE(test_default_buffer_size) |
| 126 | { |
| 127 | BOOST_CHECK_EQUAL(1024, TMemoryBuffer().getBufferSize()); |
| 128 | } |
| 129 | |
| 130 | BOOST_AUTO_TEST_CASE(test_error_set_max_buffer_size_too_small) |
| 131 | { |
| 132 | TMemoryBuffer buf; |
| 133 | BOOST_CHECK_THROW(buf.setMaxBufferSize(buf.getBufferSize() - 1), TTransportException); |
| 134 | } |
| 135 | |
| 136 | BOOST_AUTO_TEST_CASE(test_maximum_buffer_size) |
| 137 | { |
| 138 | TMemoryBuffer buf; |
| 139 | buf.setMaxBufferSize(8192); |
| 140 | std::vector<uint8_t> small_buff(1); |
| 141 | |
| 142 | for (size_t i = 0; i < 8192; ++i) |
| 143 | { |
| 144 | buf.write(&small_buff[0], 1); |
| 145 | } |
| 146 | |
| 147 | BOOST_CHECK_THROW(buf.write(&small_buff[0], 1), TTransportException); |
| 148 | } |
| 149 | |
| 150 | BOOST_AUTO_TEST_CASE(test_memory_buffer_to_get_sizeof_objects) |
| 151 | { |
| 152 | // This is a demonstration of how to use TMemoryBuffer to determine |
| 153 | // the serialized size of a thrift object in the Binary protocol. |
| 154 | // See THRIFT-3480 |
| 155 | |
| 156 | shared_ptr<TMemoryBuffer> memBuffer(new TMemoryBuffer()); |
| 157 | shared_ptr<TBinaryProtocol> binaryProtcol(new TBinaryProtocol(memBuffer)); |
| 158 | |
| 159 | thrift::test::Xtruct object; |
| 160 | object.i32_thing = 10; |
| 161 | object.i64_thing = 30; |
| 162 | object.string_thing = "who's your daddy?"; |
| 163 | |
| 164 | uint32_t size = object.write(binaryProtcol.get()); |
| 165 | BOOST_CHECK_EQUAL(47, size); |
| 166 | } |
Alex Şuhan | 20116c6 | 2017-08-03 12:28:17 -0700 | [diff] [blame] | 167 | |
Roger Meier | 0069cc4 | 2010-10-13 18:10:18 +0000 | [diff] [blame] | 168 | BOOST_AUTO_TEST_SUITE_END() |