• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc.  All rights reserved.
3 // http://code.google.com/p/protobuf/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 //     * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 //     * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 //     * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 
31 // Author: kenton@google.com (Kenton Varda)
32 //  Based on original Protocol Buffers design by
33 //  Sanjay Ghemawat, Jeff Dean, and others.
34 //
35 // Testing strategy:  For each type of I/O (array, string, file, etc.) we
36 // create an output stream and write some data to it, then create a
37 // corresponding input stream to read the same data back and expect it to
38 // match.  When the data is written, it is written in several small chunks
39 // of varying sizes, with a BackUp() after each chunk.  It is read back
40 // similarly, but with chunks separated at different points.  The whole
41 // process is run with a variety of block sizes for both the input and
42 // the output.
43 //
44 // TODO(kenton):  Rewrite this test to bring it up to the standards of all
45 //   the other proto2 tests.  May want to wait for gTest to implement
46 //   "parametized tests" so that one set of tests can be used on all the
47 //   implementations.
48 
49 #include "config.h"
50 
51 #ifdef _MSC_VER
52 #include <io.h>
53 #else
54 #include <unistd.h>
55 #endif
56 #include <stdlib.h>
57 #include <sys/types.h>
58 #include <sys/stat.h>
59 #include <fcntl.h>
60 #include <errno.h>
61 #include <sstream>
62 
63 #include <google/protobuf/io/zero_copy_stream_impl.h>
64 
65 #if HAVE_ZLIB
66 #include <google/protobuf/io/gzip_stream.h>
67 #endif
68 
69 #include <google/protobuf/stubs/common.h>
70 #include <google/protobuf/testing/googletest.h>
71 #include <google/protobuf/testing/file.h>
72 #include <gtest/gtest.h>
73 
74 namespace google {
75 namespace protobuf {
76 namespace io {
77 namespace {
78 
79 #ifdef _WIN32
80 #define pipe(fds) _pipe(fds, 4096, O_BINARY)
81 #endif
82 
83 #ifndef O_BINARY
84 #ifdef _O_BINARY
85 #define O_BINARY _O_BINARY
86 #else
87 #define O_BINARY 0     // If this isn't defined, the platform doesn't need it.
88 #endif
89 #endif
90 
91 class IoTest : public testing::Test {
92  protected:
93   // Test helpers.
94 
95   // Helper to write an array of data to an output stream.
96   bool WriteToOutput(ZeroCopyOutputStream* output, const void* data, int size);
97   // Helper to read a fixed-length array of data from an input stream.
98   int ReadFromInput(ZeroCopyInputStream* input, void* data, int size);
99   // Write a string to the output stream.
100   void WriteString(ZeroCopyOutputStream* output, const string& str);
101   // Read a number of bytes equal to the size of the given string and checks
102   // that it matches the string.
103   void ReadString(ZeroCopyInputStream* input, const string& str);
104   // Writes some text to the output stream in a particular order.  Returns
105   // the number of bytes written, incase the caller needs that to set up an
106   // input stream.
107   int WriteStuff(ZeroCopyOutputStream* output);
108   // Reads text from an input stream and expects it to match what
109   // WriteStuff() writes.
110   void ReadStuff(ZeroCopyInputStream* input);
111 
112   // Similar to WriteStuff, but performs more sophisticated testing.
113   int WriteStuffLarge(ZeroCopyOutputStream* output);
114   // Reads and tests a stream that should have been written to
115   // via WriteStuffLarge().
116   void ReadStuffLarge(ZeroCopyInputStream* input);
117 
118 #if HAVE_ZLIB
119   string Compress(const string& data, const GzipOutputStream::Options& options);
120   string Uncompress(const string& data);
121 #endif
122 
123   static const int kBlockSizes[];
124   static const int kBlockSizeCount;
125 };
126 
127 const int IoTest::kBlockSizes[] = {-1, 1, 2, 5, 7, 10, 23, 64};
128 const int IoTest::kBlockSizeCount = GOOGLE_ARRAYSIZE(IoTest::kBlockSizes);
129 
WriteToOutput(ZeroCopyOutputStream * output,const void * data,int size)130 bool IoTest::WriteToOutput(ZeroCopyOutputStream* output,
131                            const void* data, int size) {
132   const uint8* in = reinterpret_cast<const uint8*>(data);
133   int in_size = size;
134 
135   void* out;
136   int out_size;
137 
138   while (true) {
139     if (!output->Next(&out, &out_size)) {
140       return false;
141     }
142     EXPECT_GT(out_size, 0);
143 
144     if (in_size <= out_size) {
145       memcpy(out, in, in_size);
146       output->BackUp(out_size - in_size);
147       return true;
148     }
149 
150     memcpy(out, in, out_size);
151     in += out_size;
152     in_size -= out_size;
153   }
154 }
155 
156 #define MAX_REPEATED_ZEROS 100
157 
ReadFromInput(ZeroCopyInputStream * input,void * data,int size)158 int IoTest::ReadFromInput(ZeroCopyInputStream* input, void* data, int size) {
159   uint8* out = reinterpret_cast<uint8*>(data);
160   int out_size = size;
161 
162   const void* in;
163   int in_size = 0;
164 
165   int repeated_zeros = 0;
166 
167   while (true) {
168     if (!input->Next(&in, &in_size)) {
169       return size - out_size;
170     }
171     EXPECT_GT(in_size, -1);
172     if (in_size == 0) {
173       repeated_zeros++;
174     } else {
175       repeated_zeros = 0;
176     }
177     EXPECT_LT(repeated_zeros, MAX_REPEATED_ZEROS);
178 
179     if (out_size <= in_size) {
180       memcpy(out, in, out_size);
181       if (in_size > out_size) {
182         input->BackUp(in_size - out_size);
183       }
184       return size;  // Copied all of it.
185     }
186 
187     memcpy(out, in, in_size);
188     out += in_size;
189     out_size -= in_size;
190   }
191 }
192 
WriteString(ZeroCopyOutputStream * output,const string & str)193 void IoTest::WriteString(ZeroCopyOutputStream* output, const string& str) {
194   EXPECT_TRUE(WriteToOutput(output, str.c_str(), str.size()));
195 }
196 
ReadString(ZeroCopyInputStream * input,const string & str)197 void IoTest::ReadString(ZeroCopyInputStream* input, const string& str) {
198   scoped_array<char> buffer(new char[str.size() + 1]);
199   buffer[str.size()] = '\0';
200   EXPECT_EQ(ReadFromInput(input, buffer.get(), str.size()), str.size());
201   EXPECT_STREQ(str.c_str(), buffer.get());
202 }
203 
WriteStuff(ZeroCopyOutputStream * output)204 int IoTest::WriteStuff(ZeroCopyOutputStream* output) {
205   WriteString(output, "Hello world!\n");
206   WriteString(output, "Some te");
207   WriteString(output, "xt.  Blah blah.");
208   WriteString(output, "abcdefg");
209   WriteString(output, "01234567890123456789");
210   WriteString(output, "foobar");
211 
212   EXPECT_EQ(output->ByteCount(), 68);
213 
214   int result = output->ByteCount();
215   return result;
216 }
217 
218 // Reads text from an input stream and expects it to match what WriteStuff()
219 // writes.
ReadStuff(ZeroCopyInputStream * input)220 void IoTest::ReadStuff(ZeroCopyInputStream* input) {
221   ReadString(input, "Hello world!\n");
222   ReadString(input, "Some text.  ");
223   ReadString(input, "Blah ");
224   ReadString(input, "blah.");
225   ReadString(input, "abcdefg");
226   EXPECT_TRUE(input->Skip(20));
227   ReadString(input, "foo");
228   ReadString(input, "bar");
229 
230   EXPECT_EQ(input->ByteCount(), 68);
231 
232   uint8 byte;
233   EXPECT_EQ(ReadFromInput(input, &byte, 1), 0);
234 }
235 
WriteStuffLarge(ZeroCopyOutputStream * output)236 int IoTest::WriteStuffLarge(ZeroCopyOutputStream* output) {
237   WriteString(output, "Hello world!\n");
238   WriteString(output, "Some te");
239   WriteString(output, "xt.  Blah blah.");
240   WriteString(output, string(100000, 'x'));  // A very long string
241   WriteString(output, string(100000, 'y'));  // A very long string
242   WriteString(output, "01234567890123456789");
243 
244   EXPECT_EQ(output->ByteCount(), 200055);
245 
246   int result = output->ByteCount();
247   return result;
248 }
249 
250 // Reads text from an input stream and expects it to match what WriteStuff()
251 // writes.
ReadStuffLarge(ZeroCopyInputStream * input)252 void IoTest::ReadStuffLarge(ZeroCopyInputStream* input) {
253   ReadString(input, "Hello world!\nSome text.  ");
254   EXPECT_TRUE(input->Skip(5));
255   ReadString(input, "blah.");
256   EXPECT_TRUE(input->Skip(100000 - 10));
257   ReadString(input, string(10, 'x') + string(100000 - 20000, 'y'));
258   EXPECT_TRUE(input->Skip(20000 - 10));
259   ReadString(input, "yyyyyyyyyy01234567890123456789");
260 
261   EXPECT_EQ(input->ByteCount(), 200055);
262 
263   uint8 byte;
264   EXPECT_EQ(ReadFromInput(input, &byte, 1), 0);
265 }
266 
267 // ===================================================================
268 
TEST_F(IoTest,ArrayIo)269 TEST_F(IoTest, ArrayIo) {
270   const int kBufferSize = 256;
271   uint8 buffer[kBufferSize];
272 
273   for (int i = 0; i < kBlockSizeCount; i++) {
274     for (int j = 0; j < kBlockSizeCount; j++) {
275       int size;
276       {
277         ArrayOutputStream output(buffer, kBufferSize, kBlockSizes[i]);
278         size = WriteStuff(&output);
279       }
280       {
281         ArrayInputStream input(buffer, size, kBlockSizes[j]);
282         ReadStuff(&input);
283       }
284     }
285   }
286 }
287 
288 #if HAVE_ZLIB
TEST_F(IoTest,GzipIo)289 TEST_F(IoTest, GzipIo) {
290   const int kBufferSize = 2*1024;
291   uint8* buffer = new uint8[kBufferSize];
292   for (int i = 0; i < kBlockSizeCount; i++) {
293     for (int j = 0; j < kBlockSizeCount; j++) {
294       for (int z = 0; z < kBlockSizeCount; z++) {
295         int gzip_buffer_size = kBlockSizes[z];
296         int size;
297         {
298           ArrayOutputStream output(buffer, kBufferSize, kBlockSizes[i]);
299           GzipOutputStream gzout(
300               &output, GzipOutputStream::GZIP, gzip_buffer_size);
301           WriteStuff(&gzout);
302           gzout.Close();
303           size = output.ByteCount();
304         }
305         {
306           ArrayInputStream input(buffer, size, kBlockSizes[j]);
307           GzipInputStream gzin(
308               &input, GzipInputStream::GZIP, gzip_buffer_size);
309           ReadStuff(&gzin);
310         }
311       }
312     }
313   }
314   delete [] buffer;
315 }
316 
TEST_F(IoTest,ZlibIo)317 TEST_F(IoTest, ZlibIo) {
318   const int kBufferSize = 2*1024;
319   uint8* buffer = new uint8[kBufferSize];
320   for (int i = 0; i < kBlockSizeCount; i++) {
321     for (int j = 0; j < kBlockSizeCount; j++) {
322       for (int z = 0; z < kBlockSizeCount; z++) {
323         int gzip_buffer_size = kBlockSizes[z];
324         int size;
325         {
326           ArrayOutputStream output(buffer, kBufferSize, kBlockSizes[i]);
327           GzipOutputStream gzout(
328               &output, GzipOutputStream::ZLIB, gzip_buffer_size);
329           WriteStuff(&gzout);
330           gzout.Close();
331           size = output.ByteCount();
332         }
333         {
334           ArrayInputStream input(buffer, size, kBlockSizes[j]);
335           GzipInputStream gzin(
336               &input, GzipInputStream::ZLIB, gzip_buffer_size);
337           ReadStuff(&gzin);
338         }
339       }
340     }
341   }
342   delete [] buffer;
343 }
344 
TEST_F(IoTest,ZlibIoInputAutodetect)345 TEST_F(IoTest, ZlibIoInputAutodetect) {
346   const int kBufferSize = 2*1024;
347   uint8* buffer = new uint8[kBufferSize];
348   int size;
349   {
350     ArrayOutputStream output(buffer, kBufferSize);
351     GzipOutputStream gzout(&output, GzipOutputStream::ZLIB);
352     WriteStuff(&gzout);
353     gzout.Close();
354     size = output.ByteCount();
355   }
356   {
357     ArrayInputStream input(buffer, size);
358     GzipInputStream gzin(&input, GzipInputStream::AUTO);
359     ReadStuff(&gzin);
360   }
361   {
362     ArrayOutputStream output(buffer, kBufferSize);
363     GzipOutputStream gzout(&output, GzipOutputStream::GZIP);
364     WriteStuff(&gzout);
365     gzout.Close();
366     size = output.ByteCount();
367   }
368   {
369     ArrayInputStream input(buffer, size);
370     GzipInputStream gzin(&input, GzipInputStream::AUTO);
371     ReadStuff(&gzin);
372   }
373   delete [] buffer;
374 }
375 
Compress(const string & data,const GzipOutputStream::Options & options)376 string IoTest::Compress(const string& data,
377                         const GzipOutputStream::Options& options) {
378   string result;
379   {
380     StringOutputStream output(&result);
381     GzipOutputStream gzout(&output, options);
382     WriteToOutput(&gzout, data.data(), data.size());
383   }
384   return result;
385 }
386 
Uncompress(const string & data)387 string IoTest::Uncompress(const string& data) {
388   string result;
389   {
390     ArrayInputStream input(data.data(), data.size());
391     GzipInputStream gzin(&input);
392     const void* buffer;
393     int size;
394     while (gzin.Next(&buffer, &size)) {
395       result.append(reinterpret_cast<const char*>(buffer), size);
396     }
397   }
398   return result;
399 }
400 
TEST_F(IoTest,CompressionOptions)401 TEST_F(IoTest, CompressionOptions) {
402   // Some ad-hoc testing of compression options.
403 
404   string golden;
405   File::ReadFileToStringOrDie(
406     TestSourceDir() + "/google/protobuf/testdata/golden_message",
407     &golden);
408 
409   GzipOutputStream::Options options;
410   string gzip_compressed = Compress(golden, options);
411 
412   options.compression_level = 0;
413   string not_compressed = Compress(golden, options);
414 
415   // Try zlib compression for fun.
416   options = GzipOutputStream::Options();
417   options.format = GzipOutputStream::ZLIB;
418   string zlib_compressed = Compress(golden, options);
419 
420   // Uncompressed should be bigger than the original since it should have some
421   // sort of header.
422   EXPECT_GT(not_compressed.size(), golden.size());
423 
424   // Higher compression levels should result in smaller sizes.
425   EXPECT_LT(zlib_compressed.size(), not_compressed.size());
426 
427   // ZLIB format should differ from GZIP format.
428   EXPECT_TRUE(zlib_compressed != gzip_compressed);
429 
430   // Everything should decompress correctly.
431   EXPECT_TRUE(Uncompress(not_compressed) == golden);
432   EXPECT_TRUE(Uncompress(gzip_compressed) == golden);
433   EXPECT_TRUE(Uncompress(zlib_compressed) == golden);
434 }
435 #endif
436 
437 // There is no string input, only string output.  Also, it doesn't support
438 // explicit block sizes.  So, we'll only run one test and we'll use
439 // ArrayInput to read back the results.
TEST_F(IoTest,StringIo)440 TEST_F(IoTest, StringIo) {
441   string str;
442   {
443     StringOutputStream output(&str);
444     WriteStuff(&output);
445   }
446   {
447     ArrayInputStream input(str.data(), str.size());
448     ReadStuff(&input);
449   }
450 }
451 
452 
453 // To test files, we create a temporary file, write, read, truncate, repeat.
TEST_F(IoTest,FileIo)454 TEST_F(IoTest, FileIo) {
455   string filename = TestTempDir() + "/zero_copy_stream_test_file";
456 
457   for (int i = 0; i < kBlockSizeCount; i++) {
458     for (int j = 0; j < kBlockSizeCount; j++) {
459       // Make a temporary file.
460       int file =
461         open(filename.c_str(), O_RDWR | O_CREAT | O_TRUNC | O_BINARY, 0777);
462       ASSERT_GE(file, 0);
463 
464       {
465         FileOutputStream output(file, kBlockSizes[i]);
466         WriteStuff(&output);
467         EXPECT_EQ(0, output.GetErrno());
468       }
469 
470       // Rewind.
471       ASSERT_NE(lseek(file, 0, SEEK_SET), (off_t)-1);
472 
473       {
474         FileInputStream input(file, kBlockSizes[j]);
475         ReadStuff(&input);
476         EXPECT_EQ(0, input.GetErrno());
477       }
478 
479       close(file);
480     }
481   }
482 }
483 
484 #if HAVE_ZLIB
TEST_F(IoTest,GzipFileIo)485 TEST_F(IoTest, GzipFileIo) {
486   string filename = TestTempDir() + "/zero_copy_stream_test_file";
487 
488   for (int i = 0; i < kBlockSizeCount; i++) {
489     for (int j = 0; j < kBlockSizeCount; j++) {
490       // Make a temporary file.
491       int file =
492         open(filename.c_str(), O_RDWR | O_CREAT | O_TRUNC | O_BINARY, 0777);
493       ASSERT_GE(file, 0);
494       {
495         FileOutputStream output(file, kBlockSizes[i]);
496         GzipOutputStream gzout(&output);
497         WriteStuffLarge(&gzout);
498         gzout.Close();
499         output.Flush();
500         EXPECT_EQ(0, output.GetErrno());
501       }
502 
503       // Rewind.
504       ASSERT_NE(lseek(file, 0, SEEK_SET), (off_t)-1);
505 
506       {
507         FileInputStream input(file, kBlockSizes[j]);
508         GzipInputStream gzin(&input);
509         ReadStuffLarge(&gzin);
510         EXPECT_EQ(0, input.GetErrno());
511       }
512 
513       close(file);
514     }
515   }
516 }
517 #endif
518 
519 // MSVC raises various debugging exceptions if we try to use a file
520 // descriptor of -1, defeating our tests below.  This class will disable
521 // these debug assertions while in scope.
522 class MsvcDebugDisabler {
523  public:
524 #if defined(_MSC_VER) && _MSC_VER >= 1400
MsvcDebugDisabler()525   MsvcDebugDisabler() {
526     old_handler_ = _set_invalid_parameter_handler(MyHandler);
527     old_mode_ = _CrtSetReportMode(_CRT_ASSERT, 0);
528   }
~MsvcDebugDisabler()529   ~MsvcDebugDisabler() {
530     old_handler_ = _set_invalid_parameter_handler(old_handler_);
531     old_mode_ = _CrtSetReportMode(_CRT_ASSERT, old_mode_);
532   }
533 
MyHandler(const wchar_t * expr,const wchar_t * func,const wchar_t * file,unsigned int line,uintptr_t pReserved)534   static void MyHandler(const wchar_t *expr,
535                         const wchar_t *func,
536                         const wchar_t *file,
537                         unsigned int line,
538                         uintptr_t pReserved) {
539     // do nothing
540   }
541 
542   _invalid_parameter_handler old_handler_;
543   int old_mode_;
544 #else
545   // Dummy constructor and destructor to ensure that GCC doesn't complain
546   // that debug_disabler is an unused variable.
547   MsvcDebugDisabler() {}
548   ~MsvcDebugDisabler() {}
549 #endif
550 };
551 
552 // Test that FileInputStreams report errors correctly.
TEST_F(IoTest,FileReadError)553 TEST_F(IoTest, FileReadError) {
554   MsvcDebugDisabler debug_disabler;
555 
556   // -1 = invalid file descriptor.
557   FileInputStream input(-1);
558 
559   const void* buffer;
560   int size;
561   EXPECT_FALSE(input.Next(&buffer, &size));
562   EXPECT_EQ(EBADF, input.GetErrno());
563 }
564 
565 // Test that FileOutputStreams report errors correctly.
TEST_F(IoTest,FileWriteError)566 TEST_F(IoTest, FileWriteError) {
567   MsvcDebugDisabler debug_disabler;
568 
569   // -1 = invalid file descriptor.
570   FileOutputStream input(-1);
571 
572   void* buffer;
573   int size;
574 
575   // The first call to Next() succeeds because it doesn't have anything to
576   // write yet.
577   EXPECT_TRUE(input.Next(&buffer, &size));
578 
579   // Second call fails.
580   EXPECT_FALSE(input.Next(&buffer, &size));
581 
582   EXPECT_EQ(EBADF, input.GetErrno());
583 }
584 
585 // Pipes are not seekable, so File{Input,Output}Stream ends up doing some
586 // different things to handle them.  We'll test by writing to a pipe and
587 // reading back from it.
TEST_F(IoTest,PipeIo)588 TEST_F(IoTest, PipeIo) {
589   int files[2];
590 
591   for (int i = 0; i < kBlockSizeCount; i++) {
592     for (int j = 0; j < kBlockSizeCount; j++) {
593       // Need to create a new pipe each time because ReadStuff() expects
594       // to see EOF at the end.
595       ASSERT_EQ(pipe(files), 0);
596 
597       {
598         FileOutputStream output(files[1], kBlockSizes[i]);
599         WriteStuff(&output);
600         EXPECT_EQ(0, output.GetErrno());
601       }
602       close(files[1]);  // Send EOF.
603 
604       {
605         FileInputStream input(files[0], kBlockSizes[j]);
606         ReadStuff(&input);
607         EXPECT_EQ(0, input.GetErrno());
608       }
609       close(files[0]);
610     }
611   }
612 }
613 
614 // Test using C++ iostreams.
TEST_F(IoTest,IostreamIo)615 TEST_F(IoTest, IostreamIo) {
616   for (int i = 0; i < kBlockSizeCount; i++) {
617     for (int j = 0; j < kBlockSizeCount; j++) {
618       {
619         stringstream stream;
620 
621         {
622           OstreamOutputStream output(&stream, kBlockSizes[i]);
623           WriteStuff(&output);
624           EXPECT_FALSE(stream.fail());
625         }
626 
627         {
628           IstreamInputStream input(&stream, kBlockSizes[j]);
629           ReadStuff(&input);
630           EXPECT_TRUE(stream.eof());
631         }
632       }
633 
634       {
635         stringstream stream;
636 
637         {
638           OstreamOutputStream output(&stream, kBlockSizes[i]);
639           WriteStuffLarge(&output);
640           EXPECT_FALSE(stream.fail());
641         }
642 
643         {
644           IstreamInputStream input(&stream, kBlockSizes[j]);
645           ReadStuffLarge(&input);
646           EXPECT_TRUE(stream.eof());
647         }
648       }
649     }
650   }
651 }
652 
653 // To test ConcatenatingInputStream, we create several ArrayInputStreams
654 // covering a buffer and then concatenate them.
TEST_F(IoTest,ConcatenatingInputStream)655 TEST_F(IoTest, ConcatenatingInputStream) {
656   const int kBufferSize = 256;
657   uint8 buffer[kBufferSize];
658 
659   // Fill the buffer.
660   ArrayOutputStream output(buffer, kBufferSize);
661   WriteStuff(&output);
662 
663   // Now split it up into multiple streams of varying sizes.
664   ASSERT_EQ(68, output.ByteCount());  // Test depends on this.
665   ArrayInputStream input1(buffer     , 12);
666   ArrayInputStream input2(buffer + 12,  7);
667   ArrayInputStream input3(buffer + 19,  6);
668   ArrayInputStream input4(buffer + 25, 15);
669   ArrayInputStream input5(buffer + 40,  0);
670   // Note:  We want to make sure we have a stream boundary somewhere between
671   // bytes 42 and 62, which is the range that it Skip()ed by ReadStuff().  This
672   // tests that a bug that existed in the original code for Skip() is fixed.
673   ArrayInputStream input6(buffer + 40, 10);
674   ArrayInputStream input7(buffer + 50, 18);  // Total = 68 bytes.
675 
676   ZeroCopyInputStream* streams[] =
677     {&input1, &input2, &input3, &input4, &input5, &input6, &input7};
678 
679   // Create the concatenating stream and read.
680   ConcatenatingInputStream input(streams, GOOGLE_ARRAYSIZE(streams));
681   ReadStuff(&input);
682 }
683 
684 // To test LimitingInputStream, we write our golden text to a buffer, then
685 // create an ArrayInputStream that contains the whole buffer (not just the
686 // bytes written), then use a LimitingInputStream to limit it just to the
687 // bytes written.
TEST_F(IoTest,LimitingInputStream)688 TEST_F(IoTest, LimitingInputStream) {
689   const int kBufferSize = 256;
690   uint8 buffer[kBufferSize];
691 
692   // Fill the buffer.
693   ArrayOutputStream output(buffer, kBufferSize);
694   WriteStuff(&output);
695 
696   // Set up input.
697   ArrayInputStream array_input(buffer, kBufferSize);
698   LimitingInputStream input(&array_input, output.ByteCount());
699 
700   ReadStuff(&input);
701 }
702 
703 // Check that a zero-size array doesn't confuse the code.
TEST(ZeroSizeArray,Input)704 TEST(ZeroSizeArray, Input) {
705   ArrayInputStream input(NULL, 0);
706   const void* data;
707   int size;
708   EXPECT_FALSE(input.Next(&data, &size));
709 }
710 
TEST(ZeroSizeArray,Output)711 TEST(ZeroSizeArray, Output) {
712   ArrayOutputStream output(NULL, 0);
713   void* data;
714   int size;
715   EXPECT_FALSE(output.Next(&data, &size));
716 }
717 
718 }  // namespace
719 }  // namespace io
720 }  // namespace protobuf
721 }  // namespace google
722