/
home
/
obinna
/
html
/
restaurants
/
vendor
/
mongodb
/
mongodb
/
tests
/
GridFS
/
Upload File
HOME
<?php namespace MongoDB\Tests\GridFS; use MongoDB\BSON\Binary; use MongoDB\Exception\InvalidArgumentException; use MongoDB\GridFS\CollectionWrapper; use MongoDB\GridFS\Exception\CorruptFileException; use MongoDB\GridFS\ReadableStream; use MongoDB\Tests\CommandObserver; use Symfony\Bridge\PhpUnit\SetUpTearDownTrait; use function array_filter; /** * Functional tests for the internal ReadableStream class. */ class ReadableStreamFunctionalTest extends FunctionalTestCase { use SetUpTearDownTrait; /** @var CollectionWrapper */ private $collectionWrapper; private function doSetUp() { parent::setUp(); $this->collectionWrapper = new CollectionWrapper($this->manager, $this->getDatabaseName(), 'fs'); $this->filesCollection->insertMany([ ['_id' => 'length-0', 'length' => 0, 'chunkSize' => 4], ['_id' => 'length-0-with-empty-chunk', 'length' => 0, 'chunkSize' => 4], ['_id' => 'length-2', 'length' => 2, 'chunkSize' => 4], ['_id' => 'length-8', 'length' => 8, 'chunkSize' => 4], ['_id' => 'length-10', 'length' => 10, 'chunkSize' => 4], ]); $this->chunksCollection->insertMany([ ['_id' => 1, 'files_id' => 'length-0-with-empty-chunk', 'n' => 0, 'data' => new Binary('', Binary::TYPE_GENERIC)], ['_id' => 2, 'files_id' => 'length-2', 'n' => 0, 'data' => new Binary('ab', Binary::TYPE_GENERIC)], ['_id' => 3, 'files_id' => 'length-8', 'n' => 0, 'data' => new Binary('abcd', Binary::TYPE_GENERIC)], ['_id' => 4, 'files_id' => 'length-8', 'n' => 1, 'data' => new Binary('efgh', Binary::TYPE_GENERIC)], ['_id' => 5, 'files_id' => 'length-10', 'n' => 0, 'data' => new Binary('abcd', Binary::TYPE_GENERIC)], ['_id' => 6, 'files_id' => 'length-10', 'n' => 1, 'data' => new Binary('efgh', Binary::TYPE_GENERIC)], ['_id' => 7, 'files_id' => 'length-10', 'n' => 2, 'data' => new Binary('ij', Binary::TYPE_GENERIC)], ]); } public function testGetFile() { $fileDocument = (object) ['_id' => null, 'chunkSize' => 1, 'length' => 0]; $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->assertSame($fileDocument, $stream->getFile()); } /** * @dataProvider provideInvalidConstructorFileDocuments */ public function testConstructorFileDocumentChecks($file) { $this->expectException(CorruptFileException::class); new ReadableStream($this->collectionWrapper, $file); } public function provideInvalidConstructorFileDocuments() { $options = []; foreach ($this->getInvalidIntegerValues() as $value) { $options[][] = (object) ['_id' => 1, 'chunkSize' => $value, 'length' => 0]; } foreach ($this->getInvalidIntegerValues() as $value) { $options[][] = (object) ['_id' => 1, 'chunkSize' => 1, 'length' => $value]; } $options[][] = (object) ['_id' => 1, 'chunkSize' => 0, 'length' => 0]; $options[][] = (object) ['_id' => 1, 'chunkSize' => 1, 'length' => -1]; $options[][] = (object) ['chunkSize' => 1, 'length' => 0]; return $options; } /** * @dataProvider provideFileIdAndExpectedBytes */ public function testReadBytes($fileId, $length, $expectedBytes) { $fileDocument = $this->collectionWrapper->findFileById($fileId); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->assertSame($expectedBytes, $stream->readBytes($length)); } public function provideFileIdAndExpectedBytes() { return [ ['length-0', 0, ''], ['length-0', 2, ''], ['length-0-with-empty-chunk', 0, ''], ['length-0-with-empty-chunk', 2, ''], ['length-2', 0, ''], ['length-2', 2, 'ab'], ['length-2', 4, 'ab'], ['length-8', 0, ''], ['length-8', 2, 'ab'], ['length-8', 4, 'abcd'], ['length-8', 6, 'abcdef'], ['length-8', 8, 'abcdefgh'], ['length-8', 10, 'abcdefgh'], ['length-10', 0, ''], ['length-10', 2, 'ab'], ['length-10', 4, 'abcd'], ['length-10', 6, 'abcdef'], ['length-10', 8, 'abcdefgh'], ['length-10', 10, 'abcdefghij'], ['length-10', 12, 'abcdefghij'], ]; } public function provideFilteredFileIdAndExpectedBytes() { return array_filter( $this->provideFileIdAndExpectedBytes(), function (array $args) { return $args[1] > 0; } ); } /** * @dataProvider provideFilteredFileIdAndExpectedBytes */ public function testReadBytesCalledMultipleTimes($fileId, $length, $expectedBytes) { $fileDocument = $this->collectionWrapper->findFileById($fileId); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); for ($i = 0; $i < $length; $i++) { $expectedByte = $expectedBytes[$i] ?? ''; $this->assertSame($expectedByte, $stream->readBytes(1)); } } public function testReadBytesWithMissingChunk() { $this->chunksCollection->deleteOne(['files_id' => 'length-10', 'n' => 2]); $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->expectException(CorruptFileException::class); $this->expectExceptionMessage('Chunk not found for index "2"'); $stream->readBytes(10); } public function testReadBytesWithUnexpectedChunkIndex() { $this->chunksCollection->deleteOne(['files_id' => 'length-10', 'n' => 1]); $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->expectException(CorruptFileException::class); $this->expectExceptionMessage('Expected chunk to have index "1" but found "2"'); $stream->readBytes(10); } public function testReadBytesWithUnexpectedChunkSize() { $this->chunksCollection->updateOne( ['files_id' => 'length-10', 'n' => 2], ['$set' => ['data' => new Binary('i', Binary::TYPE_GENERIC)]] ); $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->expectException(CorruptFileException::class); $this->expectExceptionMessage('Expected chunk to have size "2" but found "1"'); $stream->readBytes(10); } public function testReadBytesWithNegativeLength() { $fileDocument = $this->collectionWrapper->findFileById('length-0'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->expectException(InvalidArgumentException::class); $stream->readBytes(-1); } public function testSeekBeforeReading() { $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $stream->seek(8); $this->assertSame('ij', $stream->readBytes(2)); } public function testSeekOutOfRange() { $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); $this->expectException(InvalidArgumentException::class); $this->expectExceptionMessage('$offset must be >= 0 and <= 10; given: 11'); $stream->seek(11); } /** * @dataProvider providePreviousChunkSeekOffsetAndBytes */ public function testSeekPreviousChunk($offset, $length, $expectedBytes) { $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); // Read to initialize and advance the chunk iterator to the last chunk $this->assertSame('abcdefghij', $stream->readBytes(10)); $commands = []; (new CommandObserver())->observe( function () use ($stream, $offset, $length, $expectedBytes) { $stream->seek($offset); $this->assertSame($expectedBytes, $stream->readBytes($length)); }, function (array $event) use (&$commands) { $commands[] = $event['started']->getCommandName(); } ); $this->assertSame(['find'], $commands); } public function providePreviousChunkSeekOffsetAndBytes() { return [ [0, 4, 'abcd'], [2, 4, 'cdef'], [4, 4, 'efgh'], [6, 4, 'ghij'], ]; } /** * @dataProvider provideSameChunkSeekOffsetAndBytes */ public function testSeekSameChunk($offset, $length, $expectedBytes) { $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); // Read to initialize and advance the chunk iterator to the middle chunk $this->assertSame('abcdef', $stream->readBytes(6)); $commands = []; (new CommandObserver())->observe( function () use ($stream, $offset, $length, $expectedBytes) { $stream->seek($offset); $this->assertSame($expectedBytes, $stream->readBytes($length)); }, function (array $event) use (&$commands) { $commands[] = $event['started']->getCommandName(); } ); $this->assertSame([], $commands); } public function provideSameChunkSeekOffsetAndBytes() { return [ [4, 4, 'efgh'], [6, 4, 'ghij'], ]; } /** * @dataProvider provideSubsequentChunkSeekOffsetAndBytes */ public function testSeekSubsequentChunk($offset, $length, $expectedBytes) { $fileDocument = $this->collectionWrapper->findFileById('length-10'); $stream = new ReadableStream($this->collectionWrapper, $fileDocument); // Read to initialize the chunk iterator to the first chunk $this->assertSame('a', $stream->readBytes(1)); $commands = []; (new CommandObserver())->observe( function () use ($stream, $offset, $length, $expectedBytes) { $stream->seek($offset); $this->assertSame($expectedBytes, $stream->readBytes($length)); }, function (array $event) use (&$commands) { $commands[] = $event['started']->getCommandName(); } ); $this->assertSame([], $commands); } public function provideSubsequentChunkSeekOffsetAndBytes() { return [ [4, 4, 'efgh'], [6, 4, 'ghij'], [8, 2, 'ij'], ]; } }