mirror of
https://github.com/danog/file.git
synced 2024-11-26 11:54:54 +01:00
8cfe851cd2
Addresses #32.
193 lines
4.0 KiB
PHP
193 lines
4.0 KiB
PHP
<?php
|
|
|
|
namespace Amp\File;
|
|
|
|
use Amp\ByteStream\ClosedException;
|
|
use Amp\ByteStream\StreamException;
|
|
use Amp\Failure;
|
|
use Amp\Promise;
|
|
use Amp\Success;
|
|
use function Amp\call;
|
|
|
|
class BlockingHandle implements Handle
|
|
{
|
|
private $fh;
|
|
private $path;
|
|
private $mode;
|
|
|
|
/**
|
|
* @param resource $fh An open uv filesystem descriptor
|
|
* @param string $path
|
|
* @param string $mode
|
|
*/
|
|
public function __construct($fh, string $path, string $mode)
|
|
{
|
|
$this->fh = $fh;
|
|
$this->path = $path;
|
|
$this->mode = $mode;
|
|
}
|
|
|
|
public function __destruct()
|
|
{
|
|
if ($this->fh !== null) {
|
|
\fclose($this->fh);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function read(int $length = self::DEFAULT_READ_LENGTH): Promise
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
$data = \fread($this->fh, $length);
|
|
|
|
if ($data !== false) {
|
|
return new Success(\strlen($data) ? $data : null);
|
|
}
|
|
|
|
return new Failure(new StreamException(
|
|
"Failed reading from file handle"
|
|
));
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function write(string $data): Promise
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
$len = \fwrite($this->fh, $data);
|
|
|
|
if ($len !== false) {
|
|
return new Success($len);
|
|
}
|
|
|
|
return new Failure(new StreamException(
|
|
"Failed writing to file handle"
|
|
));
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function end(string $data = ""): Promise
|
|
{
|
|
return call(function () use ($data) {
|
|
$promise = $this->write($data);
|
|
|
|
// ignore any errors
|
|
yield Promise\any([$this->close()]);
|
|
|
|
return $promise;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function close(): Promise
|
|
{
|
|
if ($this->fh === null) {
|
|
return new Success;
|
|
}
|
|
|
|
$fh = $this->fh;
|
|
$this->fh = null;
|
|
|
|
if (@\fclose($fh)) {
|
|
return new Success;
|
|
}
|
|
|
|
return new Failure(new StreamException(
|
|
"Failed closing file handle"
|
|
));
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function truncate(int $size): Promise
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
if (!@\ftruncate($this->fh, $size)) {
|
|
return new Failure(new StreamException("Could not truncate file"));
|
|
}
|
|
|
|
return new Success;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function seek(int $position, int $whence = \SEEK_SET): Promise
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
switch ($whence) {
|
|
case \SEEK_SET:
|
|
case \SEEK_CUR:
|
|
case \SEEK_END:
|
|
if (@\fseek($this->fh, $position, $whence) === -1) {
|
|
return new Failure(new StreamException("Could not seek in file"));
|
|
}
|
|
return new Success($this->tell());
|
|
default:
|
|
throw new \Error(
|
|
"Invalid whence parameter; SEEK_SET, SEEK_CUR or SEEK_END expected"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function tell(): int
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
return \ftell($this->fh);
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function eof(): bool
|
|
{
|
|
if ($this->fh === null) {
|
|
throw new ClosedException("The file has been closed");
|
|
}
|
|
|
|
return \feof($this->fh);
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function path(): string
|
|
{
|
|
return $this->path;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function mode(): string
|
|
{
|
|
return $this->mode;
|
|
}
|
|
}
|