2016-12-30 03:59:59 +01:00
|
|
|
<?php
|
2015-08-13 01:02:41 +02:00
|
|
|
|
|
|
|
namespace Amp\File;
|
|
|
|
|
2016-08-24 06:55:06 +02:00
|
|
|
use Amp\{ Success, Failure };
|
2017-01-11 14:22:06 +01:00
|
|
|
use AsyncInterop\Promise;
|
2015-08-13 01:02:41 +02:00
|
|
|
|
|
|
|
class BlockingHandle implements Handle {
|
|
|
|
private $fh;
|
|
|
|
private $path;
|
|
|
|
private $mode;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param resource $fh An open uv filesystem descriptor
|
2016-08-30 21:05:14 +02:00
|
|
|
* @param string $path
|
|
|
|
* @param string $mode
|
2015-08-13 01:02:41 +02:00
|
|
|
*/
|
2016-08-24 06:55:06 +02:00
|
|
|
public function __construct($fh, string $path, string $mode) {
|
2015-08-13 01:02:41 +02:00
|
|
|
$this->fh = $fh;
|
|
|
|
$this->path = $path;
|
|
|
|
$this->mode = $mode;
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2016-08-30 21:05:14 +02:00
|
|
|
public function __destruct() {
|
|
|
|
if ($this->fh !== null) {
|
|
|
|
\fclose($this->fh);
|
|
|
|
}
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2015-08-13 01:02:41 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 06:17:19 +01:00
|
|
|
public function read(int $length): Promise {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2016-08-24 06:55:06 +02:00
|
|
|
$data = \fread($this->fh, $length);
|
2015-08-13 01:02:41 +02:00
|
|
|
if ($data !== false) {
|
|
|
|
return new Success($data);
|
|
|
|
} else {
|
2016-03-24 18:16:20 +01:00
|
|
|
return new Failure(new FilesystemException(
|
2015-08-13 01:02:41 +02:00
|
|
|
"Failed reading from file handle"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 06:17:19 +01:00
|
|
|
public function write(string $data): Promise {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2015-08-13 01:02:41 +02:00
|
|
|
$len = \fwrite($this->fh, $data);
|
|
|
|
if ($len !== false) {
|
2016-08-30 21:05:14 +02:00
|
|
|
return new Success($len);
|
2015-08-13 01:02:41 +02:00
|
|
|
} else {
|
2016-03-24 18:16:20 +01:00
|
|
|
return new Failure(new FilesystemException(
|
2015-08-13 01:02:41 +02:00
|
|
|
"Failed writing to file handle"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 06:17:19 +01:00
|
|
|
public function close(): Promise {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has already been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2016-08-30 21:05:14 +02:00
|
|
|
$fh = $this->fh;
|
|
|
|
$this->fh = null;
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2016-08-30 21:05:14 +02:00
|
|
|
if (\fclose($fh)) {
|
2015-08-13 01:02:41 +02:00
|
|
|
return new Success;
|
|
|
|
} else {
|
2016-03-24 18:16:20 +01:00
|
|
|
return new Failure(new FilesystemException(
|
2015-08-13 01:02:41 +02:00
|
|
|
"Failed closing file handle"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 06:17:19 +01:00
|
|
|
public function seek(int $position, int $whence = \SEEK_SET): Promise {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2015-08-13 01:02:41 +02:00
|
|
|
switch ($whence) {
|
|
|
|
case \SEEK_SET:
|
|
|
|
case \SEEK_CUR:
|
|
|
|
case \SEEK_END:
|
2016-08-30 21:05:14 +02:00
|
|
|
if (@\fseek($this->fh, $position, $whence) === -1) {
|
|
|
|
return new Failure(new FilesystemException("Could not seek in file"));
|
|
|
|
}
|
|
|
|
return new Success($this->tell());
|
2015-08-13 01:02:41 +02:00
|
|
|
default:
|
2016-08-30 21:05:14 +02:00
|
|
|
throw new \Error(
|
2015-08-13 01:02:41 +02:00
|
|
|
"Invalid whence parameter; SEEK_SET, SEEK_CUR or SEEK_END expected"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-08-24 06:55:06 +02:00
|
|
|
public function tell(): int {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2015-08-13 01:02:41 +02:00
|
|
|
return \ftell($this->fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-08-24 06:55:06 +02:00
|
|
|
public function eof(): bool {
|
2016-08-30 21:05:14 +02:00
|
|
|
if ($this->fh === null) {
|
|
|
|
throw new \Error("The file has been closed");
|
|
|
|
}
|
2017-01-11 14:22:06 +01:00
|
|
|
|
2015-08-13 01:02:41 +02:00
|
|
|
return \feof($this->fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-08-24 06:55:06 +02:00
|
|
|
public function path(): string {
|
2015-08-13 01:02:41 +02:00
|
|
|
return $this->path;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-08-24 06:55:06 +02:00
|
|
|
public function mode(): string {
|
2015-08-13 01:02:41 +02:00
|
|
|
return $this->mode;
|
|
|
|
}
|
2016-03-24 18:16:20 +01:00
|
|
|
}
|