2012-06-09 23:02:26 +04:00
|
|
|
<?php
|
|
|
|
/**
|
2016-07-22 10:45:36 +03:00
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
|
|
|
*
|
|
|
|
* @author Joas Schilling <coding@schilljs.com>
|
2015-03-26 13:44:34 +03:00
|
|
|
* @author Jörn Friedrich Dreyer <jfd@butonic.de>
|
|
|
|
* @author Michael Gapczynski <GapczynskiM@gmail.com>
|
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
|
|
|
* @author Philipp Kapfer <philipp.kapfer@gmx.at>
|
2016-07-22 10:45:36 +03:00
|
|
|
* @author Robin Appelman <robin@icewind.nl>
|
2016-01-12 17:02:16 +03:00
|
|
|
* @author Robin McCorkell <robin@mccorkell.me.uk>
|
2015-03-26 13:44:34 +03:00
|
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
|
|
|
* @author Vincent Petry <pvince81@owncloud.com>
|
|
|
|
*
|
|
|
|
* @license AGPL-3.0
|
|
|
|
*
|
|
|
|
* This code is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
|
|
* as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* 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, version 3,
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*
|
|
|
|
*/
|
2015-02-26 13:37:37 +03:00
|
|
|
|
2016-04-14 01:18:07 +03:00
|
|
|
namespace OCA\Files_External\Lib\Storage;
|
2012-09-07 20:30:48 +04:00
|
|
|
|
2016-03-23 17:38:20 +03:00
|
|
|
use GuzzleHttp\Exception\RequestException;
|
2015-03-10 18:30:13 +03:00
|
|
|
use Icewind\Streams\IteratorDirectory;
|
2016-03-23 17:38:20 +03:00
|
|
|
use Icewind\Streams\RetryWrapper;
|
2016-06-14 18:38:32 +03:00
|
|
|
use OCP\Files\StorageNotAvailableException;
|
2015-03-10 18:30:13 +03:00
|
|
|
|
2016-05-24 13:15:53 +03:00
|
|
|
require_once __DIR__ . '/../../../3rdparty/Dropbox/autoload.php';
|
2012-10-12 01:17:59 +04:00
|
|
|
|
2012-09-07 20:30:48 +04:00
|
|
|
class Dropbox extends \OC\Files\Storage\Common {
|
2012-06-09 23:02:26 +04:00
|
|
|
|
|
|
|
private $dropbox;
|
2012-10-06 15:44:53 +04:00
|
|
|
private $root;
|
2012-10-12 01:06:57 +04:00
|
|
|
private $id;
|
2012-06-09 23:02:26 +04:00
|
|
|
private $metaData = array();
|
2016-03-15 19:22:16 +03:00
|
|
|
private $oauth;
|
2012-06-09 23:02:26 +04:00
|
|
|
|
|
|
|
private static $tempFiles = array();
|
|
|
|
|
|
|
|
public function __construct($params) {
|
2012-11-30 19:27:11 +04:00
|
|
|
if (isset($params['configured']) && $params['configured'] == 'true'
|
|
|
|
&& isset($params['app_key'])
|
|
|
|
&& isset($params['app_secret'])
|
|
|
|
&& isset($params['token'])
|
|
|
|
&& isset($params['token_secret'])
|
|
|
|
) {
|
2013-02-27 02:52:06 +04:00
|
|
|
$this->root = isset($params['root']) ? $params['root'] : '';
|
|
|
|
$this->id = 'dropbox::'.$params['app_key'] . $params['token']. '/' . $this->root;
|
2016-03-15 19:22:16 +03:00
|
|
|
$this->oauth = new \Dropbox_OAuth_Curl($params['app_key'], $params['app_secret']);
|
|
|
|
$this->oauth->setToken($params['token'], $params['token_secret']);
|
2014-10-21 18:18:44 +04:00
|
|
|
// note: Dropbox_API connection is lazy
|
2016-03-15 19:22:16 +03:00
|
|
|
$this->dropbox = new \Dropbox_API($this->oauth, 'auto');
|
2012-08-14 01:07:56 +04:00
|
|
|
} else {
|
2016-04-14 01:18:07 +03:00
|
|
|
throw new \Exception('Creating Dropbox storage failed');
|
2012-08-14 01:07:56 +04:00
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
|
2014-02-06 19:30:58 +04:00
|
|
|
/**
|
|
|
|
* @param string $path
|
|
|
|
*/
|
2013-11-26 13:19:45 +04:00
|
|
|
private function deleteMetaData($path) {
|
2015-10-19 21:49:54 +03:00
|
|
|
$path = ltrim($this->root.$path, '/');
|
2013-11-26 13:19:45 +04:00
|
|
|
if (isset($this->metaData[$path])) {
|
|
|
|
unset($this->metaData[$path]);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-10-19 21:49:54 +03:00
|
|
|
private function setMetaData($path, $metaData) {
|
|
|
|
$this->metaData[ltrim($path, '/')] = $metaData;
|
|
|
|
}
|
|
|
|
|
2013-11-26 13:19:45 +04:00
|
|
|
/**
|
2014-05-19 19:50:53 +04:00
|
|
|
* Returns the path's metadata
|
2014-02-06 19:30:58 +04:00
|
|
|
* @param string $path path for which to return the metadata
|
2014-05-13 15:29:25 +04:00
|
|
|
* @param bool $list if true, also return the directory's contents
|
|
|
|
* @return mixed directory contents if $list is true, file metadata if $list is
|
2013-11-26 13:19:45 +04:00
|
|
|
* false, null if the file doesn't exist or "false" if the operation failed
|
|
|
|
*/
|
2015-04-20 15:56:51 +03:00
|
|
|
private function getDropBoxMetaData($path, $list = false) {
|
2015-10-19 21:49:54 +03:00
|
|
|
$path = ltrim($this->root.$path, '/');
|
2012-11-30 19:27:11 +04:00
|
|
|
if ( ! $list && isset($this->metaData[$path])) {
|
2012-06-09 23:02:26 +04:00
|
|
|
return $this->metaData[$path];
|
|
|
|
} else {
|
|
|
|
if ($list) {
|
2012-06-27 04:37:36 +04:00
|
|
|
try {
|
|
|
|
$response = $this->dropbox->getMetaData($path);
|
2016-06-14 18:38:32 +03:00
|
|
|
} catch (\Dropbox_Exception_Forbidden $e) {
|
|
|
|
throw new StorageNotAvailableException('Dropbox API rate limit exceeded', StorageNotAvailableException::STATUS_ERROR, $e);
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 04:37:36 +04:00
|
|
|
return false;
|
|
|
|
}
|
2013-11-26 13:19:45 +04:00
|
|
|
$contents = array();
|
2012-06-09 23:02:26 +04:00
|
|
|
if ($response && isset($response['contents'])) {
|
|
|
|
// Cache folder's contents
|
2013-11-26 13:19:45 +04:00
|
|
|
foreach ($response['contents'] as $file) {
|
|
|
|
if (!isset($file['is_deleted']) || !$file['is_deleted']) {
|
2015-10-19 21:49:54 +03:00
|
|
|
$this->setMetaData($path.'/'.basename($file['path']), $file);
|
2013-11-26 13:19:45 +04:00
|
|
|
$contents[] = $file;
|
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
unset($response['contents']);
|
2013-11-26 13:19:45 +04:00
|
|
|
}
|
|
|
|
if (!isset($response['is_deleted']) || !$response['is_deleted']) {
|
2015-10-19 21:49:54 +03:00
|
|
|
$this->setMetaData($path, $response);
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
// Return contents of folder only
|
|
|
|
return $contents;
|
|
|
|
} else {
|
|
|
|
try {
|
2015-03-17 12:56:24 +03:00
|
|
|
$requestPath = $path;
|
|
|
|
if ($path === '.') {
|
|
|
|
$requestPath = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
$response = $this->dropbox->getMetaData($requestPath, 'false');
|
2013-11-26 13:19:45 +04:00
|
|
|
if (!isset($response['is_deleted']) || !$response['is_deleted']) {
|
2015-10-19 21:49:54 +03:00
|
|
|
$this->setMetaData($path, $response);
|
2013-11-26 13:19:45 +04:00
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
return null;
|
2016-06-14 18:38:32 +03:00
|
|
|
} catch (\Dropbox_Exception_Forbidden $e) {
|
|
|
|
throw new StorageNotAvailableException('Dropbox API rate limit exceeded', StorageNotAvailableException::STATUS_ERROR, $e);
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
2013-11-26 13:19:45 +04:00
|
|
|
if ($exception instanceof \Dropbox_Exception_NotFound) {
|
|
|
|
// don't log, might be a file_exist check
|
|
|
|
return false;
|
|
|
|
}
|
2012-10-08 15:50:59 +04:00
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-09 23:02:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-12 01:06:57 +04:00
|
|
|
public function getId(){
|
|
|
|
return $this->id;
|
|
|
|
}
|
|
|
|
|
2012-06-09 23:02:26 +04:00
|
|
|
public function mkdir($path) {
|
2012-10-06 15:44:53 +04:00
|
|
|
$path = $this->root.$path;
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
|
|
|
$this->dropbox->createFolder($path);
|
|
|
|
return true;
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function rmdir($path) {
|
2012-06-27 23:23:26 +04:00
|
|
|
return $this->unlink($path);
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function opendir($path) {
|
2015-04-20 15:56:51 +03:00
|
|
|
$contents = $this->getDropBoxMetaData($path, true);
|
2013-11-26 13:19:45 +04:00
|
|
|
if ($contents !== false) {
|
2012-06-09 23:02:26 +04:00
|
|
|
$files = array();
|
|
|
|
foreach ($contents as $file) {
|
|
|
|
$files[] = basename($file['path']);
|
|
|
|
}
|
2015-03-10 18:30:13 +03:00
|
|
|
return IteratorDirectory::wrap($files);
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function stat($path) {
|
2015-04-20 15:56:51 +03:00
|
|
|
$metaData = $this->getDropBoxMetaData($path);
|
2012-11-30 19:27:11 +04:00
|
|
|
if ($metaData) {
|
2012-06-09 23:02:26 +04:00
|
|
|
$stat['size'] = $metaData['bytes'];
|
|
|
|
$stat['atime'] = time();
|
2012-06-27 04:37:36 +04:00
|
|
|
$stat['mtime'] = (isset($metaData['modified'])) ? strtotime($metaData['modified']) : time();
|
2012-06-09 23:02:26 +04:00
|
|
|
return $stat;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function filetype($path) {
|
|
|
|
if ($path == '' || $path == '/') {
|
|
|
|
return 'dir';
|
2012-11-30 19:27:11 +04:00
|
|
|
} else {
|
2015-04-20 15:56:51 +03:00
|
|
|
$metaData = $this->getDropBoxMetaData($path);
|
2012-11-30 19:27:11 +04:00
|
|
|
if ($metaData) {
|
|
|
|
if ($metaData['is_dir'] == 'true') {
|
|
|
|
return 'dir';
|
|
|
|
} else {
|
|
|
|
return 'file';
|
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function file_exists($path) {
|
|
|
|
if ($path == '' || $path == '/') {
|
|
|
|
return true;
|
|
|
|
}
|
2015-04-20 15:56:51 +03:00
|
|
|
if ($this->getDropBoxMetaData($path)) {
|
2012-06-09 23:02:26 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function unlink($path) {
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
2013-11-26 13:19:45 +04:00
|
|
|
$this->dropbox->delete($this->root.$path);
|
|
|
|
$this->deleteMetaData($path);
|
2012-06-27 23:23:26 +04:00
|
|
|
return true;
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function rename($path1, $path2) {
|
|
|
|
try {
|
2013-11-26 13:19:45 +04:00
|
|
|
// overwrite if target file exists and is not a directory
|
2015-04-20 15:56:51 +03:00
|
|
|
$destMetaData = $this->getDropBoxMetaData($path2);
|
2013-11-26 13:19:45 +04:00
|
|
|
if (isset($destMetaData) && $destMetaData !== false && !$destMetaData['is_dir']) {
|
|
|
|
$this->unlink($path2);
|
|
|
|
}
|
|
|
|
$this->dropbox->move($this->root.$path1, $this->root.$path2);
|
|
|
|
$this->deleteMetaData($path1);
|
2012-06-27 23:23:26 +04:00
|
|
|
return true;
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function copy($path1, $path2) {
|
2012-10-06 15:44:53 +04:00
|
|
|
$path1 = $this->root.$path1;
|
|
|
|
$path2 = $this->root.$path2;
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
|
|
|
$this->dropbox->copy($path1, $path2);
|
|
|
|
return true;
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function fopen($path, $mode) {
|
2012-10-06 15:44:53 +04:00
|
|
|
$path = $this->root.$path;
|
2012-06-09 23:02:26 +04:00
|
|
|
switch ($mode) {
|
|
|
|
case 'r':
|
|
|
|
case 'rb':
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
2016-03-15 19:22:16 +03:00
|
|
|
// slashes need to stay
|
|
|
|
$encodedPath = str_replace('%2F', '/', rawurlencode(trim($path, '/')));
|
|
|
|
$downloadUrl = 'https://api-content.dropbox.com/1/files/auto/' . $encodedPath;
|
|
|
|
$headers = $this->oauth->getOAuthHeader($downloadUrl, [], 'GET');
|
|
|
|
|
|
|
|
$client = \OC::$server->getHTTPClientService()->newClient();
|
|
|
|
try {
|
2016-03-23 17:38:20 +03:00
|
|
|
$response = $client->get($downloadUrl, [
|
2016-03-15 19:22:16 +03:00
|
|
|
'headers' => $headers,
|
2016-03-23 17:38:20 +03:00
|
|
|
'stream' => true,
|
2016-03-15 19:22:16 +03:00
|
|
|
]);
|
|
|
|
} catch (RequestException $e) {
|
|
|
|
if (!is_null($e->getResponse())) {
|
|
|
|
if ($e->getResponse()->getStatusCode() === 404) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
throw $e;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw $e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-23 17:38:20 +03:00
|
|
|
$handle = $response->getBody();
|
|
|
|
return RetryWrapper::wrap($handle);
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
|
|
|
}
|
2012-06-09 23:02:26 +04:00
|
|
|
case 'w':
|
|
|
|
case 'wb':
|
|
|
|
case 'a':
|
|
|
|
case 'ab':
|
|
|
|
case 'r+':
|
|
|
|
case 'w+':
|
|
|
|
case 'wb+':
|
|
|
|
case 'a+':
|
|
|
|
case 'x':
|
|
|
|
case 'x+':
|
|
|
|
case 'c':
|
|
|
|
case 'c+':
|
|
|
|
if (strrpos($path, '.') !== false) {
|
|
|
|
$ext = substr($path, strrpos($path, '.'));
|
|
|
|
} else {
|
|
|
|
$ext = '';
|
|
|
|
}
|
2015-08-19 00:49:29 +03:00
|
|
|
$tmpFile = \OCP\Files::tmpFile($ext);
|
2013-01-28 18:34:15 +04:00
|
|
|
\OC\Files\Stream\Close::registerCallback($tmpFile, array($this, 'writeBack'));
|
2012-06-09 23:02:26 +04:00
|
|
|
if ($this->file_exists($path)) {
|
|
|
|
$source = $this->fopen($path, 'r');
|
|
|
|
file_put_contents($tmpFile, $source);
|
|
|
|
}
|
|
|
|
self::$tempFiles[$tmpFile] = $path;
|
|
|
|
return fopen('close://'.$tmpFile, $mode);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function writeBack($tmpFile) {
|
|
|
|
if (isset(self::$tempFiles[$tmpFile])) {
|
|
|
|
$handle = fopen($tmpFile, 'r');
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
2012-07-20 18:48:28 +04:00
|
|
|
$this->dropbox->putFile(self::$tempFiles[$tmpFile], $handle);
|
2012-06-09 23:02:26 +04:00
|
|
|
unlink($tmpFile);
|
2015-10-16 14:47:00 +03:00
|
|
|
$this->deleteMetaData(self::$tempFiles[$tmpFile]);
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function free_space($path) {
|
2012-06-27 23:23:26 +04:00
|
|
|
try {
|
|
|
|
$info = $this->dropbox->getAccountInfo();
|
2012-06-09 23:02:26 +04:00
|
|
|
return $info['quota_info']['quota'] - $info['quota_info']['normal'];
|
2012-10-08 15:50:59 +04:00
|
|
|
} catch (\Exception $exception) {
|
|
|
|
\OCP\Util::writeLog('files_external', $exception->getMessage(), \OCP\Util::ERROR);
|
2012-06-27 23:23:26 +04:00
|
|
|
return false;
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function touch($path, $mtime = null) {
|
2013-11-25 15:44:27 +04:00
|
|
|
if ($this->file_exists($path)) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
$this->file_put_contents($path, '');
|
|
|
|
}
|
2013-11-26 13:19:45 +04:00
|
|
|
return true;
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|
|
|
|
|
2013-08-02 17:44:56 +04:00
|
|
|
/**
|
|
|
|
* check if curl is installed
|
|
|
|
*/
|
|
|
|
public static function checkDependencies() {
|
2015-03-12 23:43:41 +03:00
|
|
|
return true;
|
2013-08-02 17:44:56 +04:00
|
|
|
}
|
|
|
|
|
2012-06-09 23:02:26 +04:00
|
|
|
}
|