123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236 |
- <?php
- declare(strict_types=1);
- /**
- * @copyright Copyright (c) 2020 Robin Appelman <robin@icewind.nl>
- *
- * @author Christoph Wurst <christoph@winzerhof-wurst.at>
- * @author 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 OCA\Files_External\Lib\Storage;
- use Icewind\Streams\File;
- use phpseclib\Net\SSH2;
- class SFTPReadStream implements File {
- /** @var resource */
- public $context;
- /** @var \phpseclib\Net\SFTP */
- private $sftp;
- /** @var string */
- private $handle;
- /** @var int */
- private $internalPosition = 0;
- /** @var int */
- private $readPosition = 0;
- /** @var bool */
- private $eof = false;
- private $buffer = '';
- private bool $pendingRead = false;
- private int $size = 0;
- public static function register($protocol = 'sftpread') {
- if (in_array($protocol, stream_get_wrappers(), true)) {
- return false;
- }
- return stream_wrapper_register($protocol, get_called_class());
- }
- /**
- * Load the source from the stream context and return the context options
- *
- * @param string $name
- * @throws \BadMethodCallException
- */
- protected function loadContext($name) {
- $context = stream_context_get_options($this->context);
- if (isset($context[$name])) {
- $context = $context[$name];
- } else {
- throw new \BadMethodCallException('Invalid context, "' . $name . '" options not set');
- }
- if (isset($context['session']) and $context['session'] instanceof \phpseclib\Net\SFTP) {
- $this->sftp = $context['session'];
- } else {
- throw new \BadMethodCallException('Invalid context, session not set');
- }
- if (isset($context['size'])) {
- $this->size = $context['size'];
- }
- return $context;
- }
- public function stream_open($path, $mode, $options, &$opened_path) {
- [, $path] = explode('://', $path);
- $path = '/' . ltrim($path);
- $path = str_replace('//', '/', $path);
- $this->loadContext('sftp');
- if (!($this->sftp->bitmap & SSH2::MASK_LOGIN)) {
- return false;
- }
- $remote_file = $this->sftp->_realpath($path);
- if ($remote_file === false) {
- return false;
- }
- $packet = pack('Na*N2', strlen($remote_file), $remote_file, NET_SFTP_OPEN_READ, 0);
- if (!$this->sftp->_send_sftp_packet(NET_SFTP_OPEN, $packet)) {
- return false;
- }
- $response = $this->sftp->_get_sftp_packet();
- switch ($this->sftp->packet_type) {
- case NET_SFTP_HANDLE:
- $this->handle = substr($response, 4);
- break;
- case NET_SFTP_STATUS: // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED
- $this->sftp->_logError($response);
- return false;
- default:
- user_error('Expected SSH_FXP_HANDLE or SSH_FXP_STATUS');
- return false;
- }
- $this->request_chunk(256 * 1024);
- return true;
- }
- public function stream_seek($offset, $whence = SEEK_SET) {
- switch ($whence) {
- case SEEK_SET:
- $this->seekTo($offset);
- break;
- case SEEK_CUR:
- $this->seekTo($this->readPosition + $offset);
- break;
- case SEEK_END:
- $this->seekTo($this->size + $offset);
- break;
- }
- return true;
- }
- private function seekTo(int $offset): void {
- $this->internalPosition = $offset;
- $this->readPosition = $offset;
- $this->buffer = '';
- $this->request_chunk(256 * 1024);
- }
- public function stream_tell() {
- return $this->readPosition;
- }
- public function stream_read($count) {
- if (!$this->eof && strlen($this->buffer) < $count) {
- $chunk = $this->read_chunk();
- $this->buffer .= $chunk;
- if (!$this->eof) {
- $this->request_chunk(256 * 1024);
- }
- }
- $data = substr($this->buffer, 0, $count);
- $this->buffer = substr($this->buffer, $count);
- $this->readPosition += strlen($data);
- return $data;
- }
- private function request_chunk($size) {
- if ($this->pendingRead) {
- $this->sftp->_get_sftp_packet();
- }
- $packet = pack('Na*N3', strlen($this->handle), $this->handle, $this->internalPosition / 4294967296, $this->internalPosition, $size);
- $this->pendingRead = true;
- return $this->sftp->_send_sftp_packet(NET_SFTP_READ, $packet);
- }
- private function read_chunk() {
- $this->pendingRead = false;
- $response = $this->sftp->_get_sftp_packet();
- switch ($this->sftp->packet_type) {
- case NET_SFTP_DATA:
- $temp = substr($response, 4);
- $len = strlen($temp);
- $this->internalPosition += $len;
- return $temp;
- case NET_SFTP_STATUS:
- [1 => $status] = unpack('N', substr($response, 0, 4));
- if ($status == NET_SFTP_STATUS_EOF) {
- $this->eof = true;
- }
- return '';
- default:
- return '';
- }
- }
- public function stream_write($data) {
- return false;
- }
- public function stream_set_option($option, $arg1, $arg2) {
- return false;
- }
- public function stream_truncate($size) {
- return false;
- }
- public function stream_stat() {
- return false;
- }
- public function stream_lock($operation) {
- return false;
- }
- public function stream_flush() {
- return false;
- }
- public function stream_eof() {
- return $this->eof;
- }
- public function stream_close() {
- // we still have a read request incoming that needs to be handled before we can close
- if ($this->pendingRead) {
- $this->sftp->_get_sftp_packet();
- }
- if (!$this->sftp->_close_handle($this->handle)) {
- return false;
- }
- return true;
- }
- }
|