driver = $driver; $this->watcher = $watcher; $this->fh = $fh; $this->path = $path; $this->mode = $mode; $this->size = $size; $this->loop = $driver->getHandle(); $this->position = ($mode[0] === "a") ? $size : 0; $this->queue = new \SplQueue; } public function read(int $length = self::DEFAULT_READ_LENGTH): Promise { if ($this->isActive) { throw new PendingOperationError; } $this->driver->reference($this->watcher); $deferred = new Deferred; $this->isActive = true; $onRead = function ($fh, $result, $buffer) use ($deferred) { $this->isActive = false; $this->driver->unreference($this->watcher); if ($result < 0) { $deferred->fail(new FilesystemException( \uv_strerror($result) )); } else { $length = strlen($buffer); $this->position = $this->position + $length; $deferred->resolve($length ? $buffer : null); } }; \uv_fs_read($this->loop, $this->fh, $this->position, $length, $onRead); return $deferred->promise(); } /** * {@inheritdoc} */ public function write(string $data): Promise { if ($this->isActive && $this->queue->isEmpty()) { throw new PendingOperationError; } if (!$this->writable) { throw new \Error("The file is no longer writable"); } $this->isActive = true; if ($this->queue->isEmpty()) { $promise = $this->push($data); } else { $promise = $this->queue->top(); $promise = call(function () use ($promise, $data) { yield $promise; return yield $this->push($data); }); } $this->queue->push($promise); return $promise; } /** * {@inheritdoc} */ public function end(string $data = ""): Promise { $promise = $this->write($data); $this->writable = false; $promise->onResolve([$this, "close"]); return $promise; } private function push(string $data): Promise { $length = \strlen($data); $this->driver->reference($this->watcher); $deferred = new Deferred; $onWrite = function ($fh, $result) use ($deferred, $length) { if ($this->queue->isEmpty()) { $this->driver->unreference($this->watcher); $deferred->fail(new FilesystemException('No pending write, the file may have been closed')); } $this->queue->shift(); if ($this->queue->isEmpty()) { $this->driver->unreference($this->watcher); $this->isActive = false; } if ($result < 0) { $deferred->fail(new FilesystemException( \uv_strerror($result) )); } else { StatCache::clear($this->path); $newPosition = $this->position + $length; $delta = $newPosition - $this->position; $this->position = ($this->mode[0] === "a") ? $this->position : $newPosition; $this->size += $delta; $deferred->resolve($length); } }; \uv_fs_write($this->loop, $this->fh, $data, $this->position, $onWrite); return $deferred->promise(); } /** * {@inheritdoc} */ public function seek(int $offset, int $whence = \SEEK_SET): Promise { if ($this->isActive) { throw new PendingOperationError; } $offset = (int) $offset; switch ($whence) { case \SEEK_SET: $this->position = $offset; break; case \SEEK_CUR: $this->position = $this->position + $offset; break; case \SEEK_END: $this->position = $this->size + $offset; break; default: throw new \Error( "Invalid whence parameter; SEEK_SET, SEEK_CUR or SEEK_END expected" ); } return new Success($this->position); } /** * {@inheritdoc} */ public function tell(): int { return $this->position; } /** * {@inheritdoc} */ public function eof(): bool { return !$this->queue->isEmpty() ? false : ($this->size <= $this->position); } /** * {@inheritdoc} */ public function path(): string { return $this->path; } /** * {@inheritdoc} */ public function mode(): string { return $this->mode; } /** * {@inheritdoc} */ public function close(): Promise { $this->driver->reference($this->watcher); $deferred = new Deferred; \uv_fs_close($this->loop, $this->fh, function ($fh) use ($deferred) { $this->driver->unreference($this->watcher); $deferred->resolve(); }); return $deferred->promise(); } }