mirror of
https://codeberg.org/icewind/SMB.git
synced 2026-06-03 09:14:06 +02:00
use a string based buffer for Read/Write streams
This commit is contained in:
parent
866f2b19a6
commit
0d9341c527
6 changed files with 184 additions and 38 deletions
|
|
@ -7,22 +7,21 @@
|
||||||
|
|
||||||
namespace Icewind\SMB\Native;
|
namespace Icewind\SMB\Native;
|
||||||
|
|
||||||
|
use Icewind\SMB\StringBuffer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stream optimized for read only usage
|
* Stream optimized for read only usage
|
||||||
*/
|
*/
|
||||||
class NativeReadStream extends NativeStream {
|
class NativeReadStream extends NativeStream {
|
||||||
const CHUNK_SIZE = 1048576; // 1MB chunks
|
const CHUNK_SIZE = 1048576; // 1MB chunks
|
||||||
/**
|
|
||||||
* @var resource
|
|
||||||
*/
|
|
||||||
private $readBuffer = null;
|
|
||||||
|
|
||||||
private $bufferSize = 0;
|
/** @var StringBuffer */
|
||||||
|
private $readBuffer;
|
||||||
|
|
||||||
private $pos = 0;
|
private $pos = 0;
|
||||||
|
|
||||||
public function stream_open($path, $mode, $options, &$opened_path) {
|
public function stream_open($path, $mode, $options, &$opened_path) {
|
||||||
$this->readBuffer = fopen('php://memory', 'r+');
|
$this->readBuffer = new StringBuffer();
|
||||||
|
|
||||||
return parent::stream_open($path, $mode, $options, $opened_path);
|
return parent::stream_open($path, $mode, $options, $opened_path);
|
||||||
}
|
}
|
||||||
|
|
@ -54,17 +53,11 @@ class NativeReadStream extends NativeStream {
|
||||||
// php reads 8192 bytes at once
|
// php reads 8192 bytes at once
|
||||||
// however due to network latency etc, it's faster to read in larger chunks
|
// however due to network latency etc, it's faster to read in larger chunks
|
||||||
// and buffer the result
|
// and buffer the result
|
||||||
if (!parent::stream_eof() && $this->bufferSize < $count) {
|
if (!parent::stream_eof() && $this->readBuffer->remaining() < $count) {
|
||||||
$remaining = $this->readBuffer;
|
$this->readBuffer->push(parent::stream_read(self::CHUNK_SIZE));
|
||||||
$this->readBuffer = fopen('php://memory', 'r+');
|
|
||||||
$this->bufferSize = 0;
|
|
||||||
stream_copy_to_stream($remaining, $this->readBuffer);
|
|
||||||
$this->bufferSize += fwrite($this->readBuffer, parent::stream_read(self::CHUNK_SIZE));
|
|
||||||
fseek($this->readBuffer, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$result = fread($this->readBuffer, $count);
|
$result = $this->readBuffer->read($count);
|
||||||
$this->bufferSize -= $count;
|
|
||||||
|
|
||||||
$read = strlen($result);
|
$read = strlen($result);
|
||||||
$this->pos += $read;
|
$this->pos += $read;
|
||||||
|
|
@ -75,15 +68,14 @@ class NativeReadStream extends NativeStream {
|
||||||
public function stream_seek($offset, $whence = SEEK_SET) {
|
public function stream_seek($offset, $whence = SEEK_SET) {
|
||||||
$result = parent::stream_seek($offset, $whence);
|
$result = parent::stream_seek($offset, $whence);
|
||||||
if ($result) {
|
if ($result) {
|
||||||
$this->readBuffer = fopen('php://memory', 'r+');
|
$this->readBuffer->clear();
|
||||||
$this->bufferSize = 0;
|
|
||||||
$this->pos = parent::stream_tell();
|
$this->pos = parent::stream_tell();
|
||||||
}
|
}
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function stream_eof() {
|
public function stream_eof() {
|
||||||
return $this->bufferSize <= 0 && parent::stream_eof();
|
return $this->readBuffer->remaining() <= 0 && parent::stream_eof();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function stream_tell() {
|
public function stream_tell() {
|
||||||
|
|
|
||||||
|
|
@ -7,22 +7,21 @@
|
||||||
|
|
||||||
namespace Icewind\SMB\Native;
|
namespace Icewind\SMB\Native;
|
||||||
|
|
||||||
|
use Icewind\SMB\StringBuffer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stream optimized for write only usage
|
* Stream optimized for write only usage
|
||||||
*/
|
*/
|
||||||
class NativeWriteStream extends NativeStream {
|
class NativeWriteStream extends NativeStream {
|
||||||
const CHUNK_SIZE = 1048576; // 1MB chunks
|
const CHUNK_SIZE = 1048576; // 1MB chunks
|
||||||
/**
|
|
||||||
* @var resource
|
|
||||||
*/
|
|
||||||
private $writeBuffer = null;
|
|
||||||
|
|
||||||
private $bufferSize = 0;
|
/** @var StringBuffer */
|
||||||
|
private $writeBuffer;
|
||||||
|
|
||||||
private $pos = 0;
|
private $pos = 0;
|
||||||
|
|
||||||
public function stream_open($path, $mode, $options, &$opened_path) {
|
public function stream_open($path, $mode, $options, &$opened_path) {
|
||||||
$this->writeBuffer = fopen('php://memory', 'r+');
|
$this->writeBuffer = new StringBuffer();
|
||||||
|
|
||||||
return parent::stream_open($path, $mode, $options, $opened_path);
|
return parent::stream_open($path, $mode, $options, $opened_path);
|
||||||
}
|
}
|
||||||
|
|
@ -60,18 +59,14 @@ class NativeWriteStream extends NativeStream {
|
||||||
}
|
}
|
||||||
|
|
||||||
private function flushWrite() {
|
private function flushWrite() {
|
||||||
rewind($this->writeBuffer);
|
$this->state->write($this->handle, $this->writeBuffer->flush(), $this->url);
|
||||||
$this->state->write($this->handle, stream_get_contents($this->writeBuffer), $this->url);
|
|
||||||
$this->writeBuffer = fopen('php://memory', 'r+');
|
|
||||||
$this->bufferSize = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function stream_write($data) {
|
public function stream_write($data) {
|
||||||
$written = fwrite($this->writeBuffer, $data);
|
$written = $this->writeBuffer->push($data);
|
||||||
$this->bufferSize += $written;
|
|
||||||
$this->pos += $written;
|
$this->pos += $written;
|
||||||
|
|
||||||
if ($this->bufferSize >= self::CHUNK_SIZE) {
|
if ($this->writeBuffer->remaining() >= self::CHUNK_SIZE) {
|
||||||
$this->flushWrite();
|
$this->flushWrite();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
61
src/StringBuffer.php
Normal file
61
src/StringBuffer.php
Normal file
|
|
@ -0,0 +1,61 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
/**
|
||||||
|
* @copyright Copyright (c) 2021 Robin Appelman <robin@icewind.nl>
|
||||||
|
*
|
||||||
|
* @license GNU AGPL version 3 or any later version
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Icewind\SMB;
|
||||||
|
|
||||||
|
class StringBuffer {
|
||||||
|
private $buffer = "";
|
||||||
|
private $pos = 0;
|
||||||
|
|
||||||
|
public function clear() {
|
||||||
|
$this->buffer = "";
|
||||||
|
$this->pos = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function push(string $data) {
|
||||||
|
$this->buffer = $this->flush() . $data;
|
||||||
|
return strlen($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function remaining(): int {
|
||||||
|
return strlen($this->buffer) - $this->pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function read(int $count): string {
|
||||||
|
$chunk = substr($this->buffer, $this->pos, $this->pos + $count);
|
||||||
|
$this->pos += strlen($chunk);
|
||||||
|
return $chunk;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function flush(): string {
|
||||||
|
if ($this->pos === 0) {
|
||||||
|
$remaining = $this->buffer;
|
||||||
|
} else {
|
||||||
|
$remaining = substr($this->buffer, $this->pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->clear();
|
||||||
|
|
||||||
|
return $remaining;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -83,7 +83,8 @@ abstract class AbstractShareTest extends TestCase {
|
||||||
public function fileDataProvider() {
|
public function fileDataProvider() {
|
||||||
return [
|
return [
|
||||||
['Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua'],
|
['Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua'],
|
||||||
['Mixed language, 日本語 が わからか and Various _/* characters \\|” €']
|
['Mixed language, 日本語 が わからか and Various _/* characters \\|” €'],
|
||||||
|
[str_repeat('Long text with lots of characters so we get a resulting string that tests the chunked writing and reading properly', 100)]
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -418,6 +419,23 @@ abstract class AbstractShareTest extends TestCase {
|
||||||
$this->assertEquals(file_get_contents($sourceFile), $content);
|
$this->assertEquals(file_get_contents($sourceFile), $content);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider nameAndDataProvider
|
||||||
|
*/
|
||||||
|
public function testReadStreamChunked($name, $text) {
|
||||||
|
$sourceFile = $this->getTextFile($text);
|
||||||
|
$this->share->put($sourceFile, $this->root . '/' . $name);
|
||||||
|
$fh = $this->share->read($this->root . '/' . $name);
|
||||||
|
$content = "";
|
||||||
|
while (!feof($fh)) {
|
||||||
|
$content .= fread($fh, 8192);
|
||||||
|
}
|
||||||
|
fclose($fh);
|
||||||
|
$this->share->del($this->root . '/' . $name);
|
||||||
|
|
||||||
|
$this->assertEquals(file_get_contents($sourceFile), $content);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @dataProvider invalidPathProvider
|
* @dataProvider invalidPathProvider
|
||||||
*/
|
*/
|
||||||
|
|
@ -441,6 +459,24 @@ abstract class AbstractShareTest extends TestCase {
|
||||||
unlink($tmpFile1);
|
unlink($tmpFile1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider nameAndDataProvider
|
||||||
|
*/
|
||||||
|
public function testWriteStreamChunked($name, $text) {
|
||||||
|
$fh = $this->share->write($this->root . '/' . $name);
|
||||||
|
|
||||||
|
foreach (str_split($text, 8192) as $chunk) {
|
||||||
|
fwrite($fh, $chunk);
|
||||||
|
}
|
||||||
|
fclose($fh);
|
||||||
|
|
||||||
|
$tmpFile1 = tempnam('/tmp', 'smb_test_');
|
||||||
|
$this->share->get($this->root . '/' . $name, $tmpFile1);
|
||||||
|
$this->assertEquals($text, file_get_contents($tmpFile1));
|
||||||
|
$this->share->del($this->root . '/' . $name);
|
||||||
|
unlink($tmpFile1);
|
||||||
|
}
|
||||||
|
|
||||||
public function testAppendStream() {
|
public function testAppendStream() {
|
||||||
$name = 'foo.txt';
|
$name = 'foo.txt';
|
||||||
$fh = $this->share->append($this->root . '/' . $name);
|
$fh = $this->share->append($this->root . '/' . $name);
|
||||||
|
|
|
||||||
|
|
@ -100,9 +100,6 @@ class NativeStreamTest extends TestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testTruncate() {
|
public function testTruncate() {
|
||||||
if (version_compare(phpversion(), '5.4.0', '<')) {
|
|
||||||
$this->markTestSkipped('php <5.4 doesn\'t support truncate for stream wrappers');
|
|
||||||
}
|
|
||||||
$fh = $this->share->write($this->root . '/foobar');
|
$fh = $this->share->write($this->root . '/foobar');
|
||||||
fwrite($fh, 'foobar');
|
fwrite($fh, 'foobar');
|
||||||
ftruncate($fh, 3);
|
ftruncate($fh, 3);
|
||||||
|
|
@ -113,9 +110,6 @@ class NativeStreamTest extends TestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testEOF() {
|
public function testEOF() {
|
||||||
if (version_compare(phpversion(), '5.4.0', '<')) {
|
|
||||||
$this->markTestSkipped('php <5.4 doesn\'t support truncate for stream wrappers');
|
|
||||||
}
|
|
||||||
$fh = $this->share->write($this->root . '/foobar');
|
$fh = $this->share->write($this->root . '/foobar');
|
||||||
fwrite($fh, 'foobar');
|
fwrite($fh, 'foobar');
|
||||||
fclose($fh);
|
fclose($fh);
|
||||||
|
|
|
||||||
68
tests/StringBufferTest.php
Normal file
68
tests/StringBufferTest.php
Normal file
|
|
@ -0,0 +1,68 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
/**
|
||||||
|
* @copyright Copyright (c) 2021 Robin Appelman <robin@icewind.nl>
|
||||||
|
*
|
||||||
|
* @license GNU AGPL version 3 or any later version
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Icewind\SMB\Test;
|
||||||
|
|
||||||
|
use Icewind\SMB\StringBuffer;
|
||||||
|
|
||||||
|
class StringBufferTest extends TestCase {
|
||||||
|
public function testPushRead() {
|
||||||
|
$buffer = new StringBuffer();
|
||||||
|
$this->assertEquals(0, $buffer->remaining());
|
||||||
|
$buffer->push("foobar");
|
||||||
|
$this->assertEquals(6, $buffer->remaining());
|
||||||
|
$this->assertEquals("foo", $buffer->read(3));
|
||||||
|
$this->assertEquals(3, $buffer->remaining());
|
||||||
|
$this->assertEquals("bar", $buffer->read(10));
|
||||||
|
$this->assertEquals(0, $buffer->remaining());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testReadEmpty() {
|
||||||
|
$buffer = new StringBuffer();
|
||||||
|
$this->assertEquals("", $buffer->read(10));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testAppend() {
|
||||||
|
$buffer = new StringBuffer();
|
||||||
|
$this->assertEquals(0, $buffer->remaining());
|
||||||
|
$buffer->push("foo");
|
||||||
|
$this->assertEquals(3, $buffer->remaining());
|
||||||
|
$this->assertEquals("f", $buffer->read(1));
|
||||||
|
$this->assertEquals(2, $buffer->remaining());
|
||||||
|
$buffer->push("bar");
|
||||||
|
$this->assertEquals(5, $buffer->remaining());
|
||||||
|
$this->assertEquals("oobar", $buffer->read(10));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testFlush() {
|
||||||
|
$buffer = new StringBuffer();
|
||||||
|
$this->assertEquals(0, $buffer->remaining());
|
||||||
|
$buffer->push("foobar");
|
||||||
|
$this->assertEquals("f", $buffer->read(1));
|
||||||
|
$this->assertEquals(5, $buffer->remaining());
|
||||||
|
$this->assertEquals("oobar", $buffer->flush());
|
||||||
|
|
||||||
|
$buffer->push("foobar");
|
||||||
|
$this->assertEquals("foobar", $buffer->flush());
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue