blob: 2f1aea69411ec423fdeb0a37e136bde034a69ac6 [file] [log] [blame]
David Reissea2cba82009-03-30 21:35:00 +00001/*
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
Christopher Friedtedcad982020-06-18 21:50:54 -040020#include <array>
zeshuai0076b1cb302020-05-27 12:08:01 +080021#include <boost/test/unit_test.hpp>
David Reiss2dc72c32007-08-21 23:59:34 +000022#include <climits>
Christopher Friedtedcad982020-06-18 21:50:54 -040023#include <cstdlib>
24#include <iostream>
cyy316723a2019-01-05 16:35:14 +080025#include <memory>
Christopher Friedtedcad982020-06-18 21:50:54 -040026#include <numeric>
27#include <thrift/protocol/TBinaryProtocol.h>
James E. King, III82ae9572017-08-05 12:23:54 -040028#include <thrift/transport/TBufferTransports.h>
Christopher Friedtedcad982020-06-18 21:50:54 -040029#include <vector>
30
David Reiss2dc72c32007-08-21 23:59:34 +000031#include "gen-cpp/ThriftTest_types.h"
32
Konrad Grochowski16a23a62014-11-13 15:33:38 +010033BOOST_AUTO_TEST_SUITE(TMemoryBufferTest)
David Reiss2dc72c32007-08-21 23:59:34 +000034
Jim King60774812015-05-10 08:08:18 -040035using apache::thrift::protocol::TBinaryProtocol;
36using apache::thrift::transport::TMemoryBuffer;
37using apache::thrift::transport::TTransportException;
cyy316723a2019-01-05 16:35:14 +080038using std::shared_ptr;
Jim King60774812015-05-10 08:08:18 -040039using std::cout;
40using std::endl;
41using std::string;
David Reiss2dc72c32007-08-21 23:59:34 +000042
Konrad Grochowski1f6e3802015-05-18 18:10:06 +020043BOOST_AUTO_TEST_CASE(test_read_write_grow) {
Roger Meiera6b66332015-05-15 15:21:50 +020044 // Added to test the fix for THRIFT-1248
45 TMemoryBuffer uut;
46 const int maxSize = 65536;
47 uint8_t verify[maxSize];
48 std::vector<uint8_t> buf;
49 buf.resize(maxSize);
Jim King60774812015-05-10 08:08:18 -040050
Konrad Grochowski1f6e3802015-05-18 18:10:06 +020051 for (uint32_t i = 0; i < maxSize; ++i) {
Roger Meiera6b66332015-05-15 15:21:50 +020052 buf[i] = static_cast<uint8_t>(i);
53 }
Jim King60774812015-05-10 08:08:18 -040054
Konrad Grochowski1f6e3802015-05-18 18:10:06 +020055 for (uint32_t i = 1; i < maxSize; i *= 2) {
Roger Meiera6b66332015-05-15 15:21:50 +020056 uut.write(&buf[0], i);
57 }
58
Konrad Grochowski1f6e3802015-05-18 18:10:06 +020059 for (uint32_t i = 1; i < maxSize; i *= 2) {
Roger Meiera6b66332015-05-15 15:21:50 +020060 uut.read(verify, i);
61 BOOST_CHECK_EQUAL(0, ::memcmp(verify, &buf[0], i));
62 }
Jim King60774812015-05-10 08:08:18 -040063}
64
Konrad Grochowski1f6e3802015-05-18 18:10:06 +020065BOOST_AUTO_TEST_CASE(test_roundtrip) {
Konrad Grochowski16a23a62014-11-13 15:33:38 +010066 shared_ptr<TMemoryBuffer> strBuffer(new TMemoryBuffer());
67 shared_ptr<TBinaryProtocol> binaryProtcol(new TBinaryProtocol(strBuffer));
David Reiss2dc72c32007-08-21 23:59:34 +000068
Konrad Grochowski16a23a62014-11-13 15:33:38 +010069 thrift::test::Xtruct a;
70 a.i32_thing = 10;
71 a.i64_thing = 30;
72 a.string_thing = "holla back a";
David Reiss2dc72c32007-08-21 23:59:34 +000073
Konrad Grochowski16a23a62014-11-13 15:33:38 +010074 a.write(binaryProtcol.get());
75 std::string serialized = strBuffer->getBufferAsString();
David Reiss2dc72c32007-08-21 23:59:34 +000076
Konrad Grochowski16a23a62014-11-13 15:33:38 +010077 shared_ptr<TMemoryBuffer> strBuffer2(new TMemoryBuffer());
78 shared_ptr<TBinaryProtocol> binaryProtcol2(new TBinaryProtocol(strBuffer2));
David Reiss2dc72c32007-08-21 23:59:34 +000079
Konrad Grochowski16a23a62014-11-13 15:33:38 +010080 strBuffer2->resetBuffer((uint8_t*)serialized.data(), static_cast<uint32_t>(serialized.length()));
81 thrift::test::Xtruct a2;
82 a2.read(binaryProtcol2.get());
David Reiss2dc72c32007-08-21 23:59:34 +000083
Claudius Heine5ef662b2015-06-24 10:03:50 +020084 BOOST_CHECK(a == a2);
Konrad Grochowski16a23a62014-11-13 15:33:38 +010085}
86
James E. King, III533405e2017-10-28 18:25:45 -040087BOOST_AUTO_TEST_CASE(test_readAppendToString) {
cyy64750162019-02-08 13:40:59 +080088 string str1 = "abcd1234";
89 TMemoryBuffer buf((uint8_t*)str1.data(),
90 static_cast<uint32_t>(str1.length()),
Konrad Grochowski16a23a62014-11-13 15:33:38 +010091 TMemoryBuffer::COPY);
Konrad Grochowski16a23a62014-11-13 15:33:38 +010092
93 string str3 = "wxyz", str4 = "6789";
94 buf.readAppendToString(str3, 4);
95 buf.readAppendToString(str4, INT_MAX);
96
Claudius Heine5ef662b2015-06-24 10:03:50 +020097 BOOST_CHECK(str3 == "wxyzabcd");
98 BOOST_CHECK(str4 == "67891234");
Konrad Grochowski16a23a62014-11-13 15:33:38 +010099}
100
Konrad Grochowski1f6e3802015-05-18 18:10:06 +0200101BOOST_AUTO_TEST_CASE(test_exceptions) {
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100102 char data[] = "foo\0bar";
103
104 TMemoryBuffer buf1((uint8_t*)data, 7, TMemoryBuffer::OBSERVE);
105 string str = buf1.getBufferAsString();
Claudius Heine5ef662b2015-06-24 10:03:50 +0200106 BOOST_CHECK(str.length() == 7);
107
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100108 buf1.resetBuffer();
Claudius Heine5ef662b2015-06-24 10:03:50 +0200109
110 BOOST_CHECK_THROW(buf1.write((const uint8_t*)"foo", 3), TTransportException);
David Reiss2dc72c32007-08-21 23:59:34 +0000111
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100112 TMemoryBuffer buf2((uint8_t*)data, 7, TMemoryBuffer::COPY);
Claudius Heine5ef662b2015-06-24 10:03:50 +0200113 BOOST_CHECK_NO_THROW(buf2.write((const uint8_t*)"bar", 3));
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100114}
David Reissf7baf542008-02-04 21:56:27 +0000115
James E. King, IIIaded00b2017-09-22 11:20:15 -0700116BOOST_AUTO_TEST_CASE(test_default_maximum_buffer_size)
117{
James E. King III9b75e4f2018-12-17 16:21:14 -0500118 BOOST_CHECK_EQUAL((std::numeric_limits<uint32_t>::max)(), TMemoryBuffer().getMaxBufferSize());
Alex Şuhan20116c62017-08-03 12:28:17 -0700119}
James E. King, IIIaded00b2017-09-22 11:20:15 -0700120
121BOOST_AUTO_TEST_CASE(test_default_buffer_size)
122{
123 BOOST_CHECK_EQUAL(1024, TMemoryBuffer().getBufferSize());
124}
125
126BOOST_AUTO_TEST_CASE(test_error_set_max_buffer_size_too_small)
127{
128 TMemoryBuffer buf;
129 BOOST_CHECK_THROW(buf.setMaxBufferSize(buf.getBufferSize() - 1), TTransportException);
130}
131
Christopher Friedtedcad982020-06-18 21:50:54 -0400132BOOST_AUTO_TEST_CASE(test_observe) {
133#ifdef _MSC_VER
134 #define N 73
135#else
136 constexpr size_t N = 73;
137#endif
138 constexpr size_t M = 42;
139 uint8_t one_byte = 42;
140 std::vector<uint8_t> scratch;
141 auto filler = [=]() {
142 std::array<uint8_t, N> x;
143 // Fill buf_mem with a sequence from 0 to N - 1
144 std::iota(x.begin(), x.end(), 0);
145 return x;
146 };
147 static const std::array<uint8_t, N> buf_mem = filler();
148
149 BOOST_STATIC_ASSERT(M < N);
150
151 TMemoryBuffer buf((uint8_t*)&buf_mem.front(), N, TMemoryBuffer::MemoryPolicy::OBSERVE);
152
153 // Readable
154 BOOST_CHECK_EQUAL(N, buf.available_read());
155 // No available write space
156 BOOST_CHECK_EQUAL(0, buf.available_write());
157 // Not writeable
158 BOOST_CHECK_THROW(buf.write(&one_byte, 1), TTransportException);
159
160 // Read some but not all
161 scratch.resize(M);
162 BOOST_CHECK_EQUAL(M, buf.read(&scratch[0], M));
163 // Check remaining
164 BOOST_CHECK_EQUAL(N - M, buf.available_read());
165 // No available write space
166 BOOST_CHECK_EQUAL(0, buf.available_write());
167 // Not writeable
168 BOOST_CHECK_THROW(buf.write(&one_byte, 1), TTransportException);
169 // Contents
170 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), buf_mem.begin(),
171 buf_mem.begin() + M);
172
173 // Readable (drain remaining)
174 scratch.resize(N);
175 BOOST_CHECK_EQUAL(N - M, buf.read(&scratch[M], N - M));
176 // No available write space
177 BOOST_CHECK_EQUAL(0, buf.available_write());
178 // Not writeable
179 BOOST_CHECK_THROW(buf.write(&one_byte, 1), TTransportException);
180 // Contents
181 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), buf_mem.begin(), buf_mem.end());
182
183 // Not readable
184 BOOST_CHECK_EQUAL(0, buf.read(&one_byte, 1));
185 BOOST_CHECK_EQUAL(0, buf.available_read());
186 // No available write space
187 BOOST_CHECK_EQUAL(0, buf.available_write());
188 // Not writeable
189 BOOST_CHECK_THROW(buf.write(&one_byte, 1), TTransportException);
190
191 /* OBSERVE buffer cannot be reread with the default reset */
192
193 buf.resetBuffer();
194 // Not Readable
195 BOOST_CHECK_EQUAL(0, buf.available_read());
196 // No available write space
197 BOOST_CHECK_EQUAL(0, buf.available_write());
198 // Not writeable
199 BOOST_CHECK_THROW(buf.write(&one_byte, 1), TTransportException);
200
201 /* OBSERVE buffers do not auto-resize when written to (implicit) */
202 /* OBSERVE buffers can be appended-to (implicit) */
203}
204
205BOOST_AUTO_TEST_CASE(test_copy) {
206#ifdef _MSC_VER
207 #define N 73
208#else
209 constexpr size_t N = 73;
210#endif
211 constexpr size_t M = 42;
212 uint8_t one_byte = 42;
213 std::vector<uint8_t> scratch;
214 auto filler = [&]() {
215 std::array<uint8_t, N> x;
216 // Fill buf_mem with a sequence from 0 to N - 1
217 std::iota(x.begin(), x.end(), 0);
218 return x;
219 };
220 static const std::array<uint8_t, N> buf_mem = filler();
221
222 BOOST_STATIC_ASSERT(M < N);
223
224 TMemoryBuffer buf((uint8_t*)&buf_mem.front(), N, TMemoryBuffer::MemoryPolicy::COPY);
225
226 // Readable
227 BOOST_CHECK_EQUAL(N, buf.available_read());
228 // No available write space
229 BOOST_CHECK_EQUAL(0, buf.available_write());
230
231 // Read some but not all
232 scratch.resize(M);
233 BOOST_CHECK_EQUAL(M, buf.read(&scratch[0], M));
234 // Check remaining
235 BOOST_CHECK_EQUAL(N - M, buf.available_read());
236 // No available write space
237 BOOST_CHECK_EQUAL(0, buf.available_write());
238 // Contents
239 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), buf_mem.begin(),
240 buf_mem.begin() + M);
241
242 // Readable (drain remaining)
243 scratch.resize(N);
244 BOOST_CHECK_EQUAL(N - M, buf.read(&scratch[M], N - M));
245 // No available write space
246 BOOST_CHECK_EQUAL(0, buf.available_write());
247 // Contents
248 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), buf_mem.begin(), buf_mem.end());
249
250 // Not readable
251 BOOST_CHECK_EQUAL(0, buf.read(&one_byte, 1));
252 BOOST_CHECK_EQUAL(0, buf.available_read());
253 // No available write space
254 BOOST_CHECK_EQUAL(0, buf.available_write());
255
256 /* COPY buffer cannot be reread with the default reset */
257
258 buf.resetBuffer();
259 // Not readable
260 BOOST_CHECK_EQUAL(0, buf.available_read());
261 // Has available write space
262 BOOST_CHECK_EQUAL(N, buf.available_write());
263
264 /* COPY buffers auto-resize when written to */
265
266 // Not readable
267 BOOST_CHECK_EQUAL(0, buf.read(&one_byte, 1));
268 BOOST_CHECK_EQUAL(0, buf.available_read());
269 // No available write space
270 BOOST_CHECK_GT(buf.available_write(), 0);
271 // Writeable
272 one_byte = M;
273 BOOST_CHECK_NO_THROW(buf.write(&one_byte, 1));
274 // Readable
275 one_byte = 0xff;
276 BOOST_CHECK_EQUAL(1, buf.read(&one_byte, 1));
277 BOOST_CHECK_EQUAL(one_byte, M);
278
279 /* COPY buffers can be appended-to (and auto-resize) */
280
281 buf.resetBuffer((uint8_t*)&buf_mem.front(), N, TMemoryBuffer::MemoryPolicy::COPY);
282 // Appendable
283 one_byte = N + 1;
284 BOOST_CHECK_NO_THROW(buf.write(&one_byte, 1));
285 BOOST_CHECK_EQUAL(N, buf.read(&scratch[0], N));
286 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), buf_mem.begin(),
287 buf_mem.begin() + N);
288 one_byte = 0xff;
289 BOOST_CHECK_EQUAL(1, buf.read(&one_byte, 1));
290 BOOST_CHECK_EQUAL(one_byte, N + 1);
291}
292
293BOOST_AUTO_TEST_CASE(test_take_ownership)
294{
295#ifdef _MSC_VER
296 #define N 73
297#else
298 constexpr size_t N = 73;
299#endif
300 constexpr size_t M = 42;
301 uint8_t one_byte = 42;
302 std::vector<uint8_t> scratch;
303 auto filler = [&]() {
304 /* TAKE_OWNERSHIP buffers MUST be malloc'ed */
305 uint8_t* x = static_cast<uint8_t*>(malloc(N));
306 // Fill buf_mem with a sequence from 0 to N - 1
307 std::iota(&x[0], &x[N], 0);
308 return x;
309 };
310 uint8_t* buf_mem = filler();
311
312 BOOST_STATIC_ASSERT(M < N);
313
314 TMemoryBuffer buf(buf_mem, N, TMemoryBuffer::MemoryPolicy::TAKE_OWNERSHIP);
315
316 // Readable
317 BOOST_CHECK_EQUAL(N, buf.available_read());
318 // No available write space
319 BOOST_CHECK_EQUAL(0, buf.available_write());
320
321 // Read some but not all
322 scratch.resize(M);
323 BOOST_CHECK_EQUAL(M, buf.read(&scratch[0], M));
324 // Check remaining
325 BOOST_CHECK_EQUAL(N - M, buf.available_read());
326 // No available write space
327 BOOST_CHECK_EQUAL(0, buf.available_write());
328 // Contents
329 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), &buf_mem[0], &buf_mem[M]);
330
331 // Readable (drain remaining)
332 scratch.resize(N);
333 BOOST_CHECK_EQUAL(N - M, buf.read(&scratch[M], N - M));
334 // No available write space
335 BOOST_CHECK_EQUAL(0, buf.available_write());
336 // Contents
337 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), &buf_mem[0], &buf_mem[N]);
338
339 // Not readable
340 BOOST_CHECK_EQUAL(0, buf.read(&one_byte, 1));
341 BOOST_CHECK_EQUAL(0, buf.available_read());
342 // No available write space
343 BOOST_CHECK_EQUAL(0, buf.available_write());
344
345 /* TAKE_OWNERSHIP buffers auto-resize when written to */
346
347 // Not readable
348 BOOST_CHECK_EQUAL(0, buf.read(&one_byte, 1));
349 BOOST_CHECK_EQUAL(0, buf.available_read());
350 // No available write space
351 BOOST_CHECK_EQUAL(buf.available_write(), 0);
352 // Writeable
353 one_byte = M;
354 BOOST_CHECK_NO_THROW(buf.write(&one_byte, 1));
355 // Readable
356 one_byte = 0xff;
357 BOOST_CHECK_EQUAL(1, buf.read(&one_byte, 1));
358 BOOST_CHECK_EQUAL(one_byte, M);
359
360 /* TAKE_OWNERSHIP buffers can be appended-to (and auto-resize) */
361
362 buf_mem = filler();
363 buf.resetBuffer(buf_mem, N, TMemoryBuffer::MemoryPolicy::COPY);
364 // Appendable
365 one_byte = N + 1;
366 BOOST_CHECK_NO_THROW(buf.write(&one_byte, 1));
367 BOOST_CHECK_EQUAL(N, buf.read(&scratch[0], N));
368 BOOST_CHECK_EQUAL_COLLECTIONS(scratch.begin(), scratch.end(), &buf_mem[0], &buf_mem[N]);
369 one_byte = 0xff;
370 BOOST_CHECK_EQUAL(1, buf.read(&one_byte, 1));
371 BOOST_CHECK_EQUAL(one_byte, N + 1);
372}
373
James E. King, IIIaded00b2017-09-22 11:20:15 -0700374BOOST_AUTO_TEST_CASE(test_maximum_buffer_size)
375{
376 TMemoryBuffer buf;
377 buf.setMaxBufferSize(8192);
378 std::vector<uint8_t> small_buff(1);
379
380 for (size_t i = 0; i < 8192; ++i)
381 {
382 buf.write(&small_buff[0], 1);
383 }
384
385 BOOST_CHECK_THROW(buf.write(&small_buff[0], 1), TTransportException);
386}
387
stiga-huangd665e282023-06-16 22:53:43 +0800388BOOST_AUTO_TEST_CASE(test_buffer_overflow)
389{
390 TMemoryBuffer buf;
391 std::vector<uint8_t> small_buff(1);
392 buf.write(&small_buff[0], 1);
393 BOOST_CHECK_THROW(buf.getWritePtr(std::numeric_limits<uint32_t>::max()), TTransportException);
394}
395
James E. King, IIIaded00b2017-09-22 11:20:15 -0700396BOOST_AUTO_TEST_CASE(test_memory_buffer_to_get_sizeof_objects)
397{
398 // This is a demonstration of how to use TMemoryBuffer to determine
399 // the serialized size of a thrift object in the Binary protocol.
400 // See THRIFT-3480
401
402 shared_ptr<TMemoryBuffer> memBuffer(new TMemoryBuffer());
403 shared_ptr<TBinaryProtocol> binaryProtcol(new TBinaryProtocol(memBuffer));
404
405 thrift::test::Xtruct object;
406 object.i32_thing = 10;
407 object.i64_thing = 30;
408 object.string_thing = "who's your daddy?";
409
410 uint32_t size = object.write(binaryProtcol.get());
411 BOOST_CHECK_EQUAL(47, size);
412}
Alex Şuhan20116c62017-08-03 12:28:17 -0700413
Roger Meier0069cc42010-10-13 18:10:18 +0000414BOOST_AUTO_TEST_SUITE_END()