2014-06-11 15:57:24 +04:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @author Jörn Friedrich Dreyer
|
|
|
|
* @copyright (c) 2014 Jörn Friedrich Dreyer <jfd@owncloud.com>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 3 of the License, or any later version.
|
|
|
|
*
|
|
|
|
* This library 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 library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OC\Files\ObjectStore;
|
|
|
|
|
2014-06-13 19:22:21 +04:00
|
|
|
use Guzzle\Http\Exception\ClientErrorResponseException;
|
2014-06-20 14:27:47 +04:00
|
|
|
use OCP\Files\ObjectStore\IObjectStore;
|
2014-06-11 15:57:24 +04:00
|
|
|
use OpenCloud\OpenStack;
|
2014-06-24 16:36:29 +04:00
|
|
|
use OpenCloud\Rackspace;
|
2014-06-11 15:57:24 +04:00
|
|
|
|
2014-06-20 14:27:47 +04:00
|
|
|
class Swift implements IObjectStore {
|
2014-06-24 16:42:52 +04:00
|
|
|
/**
|
|
|
|
* @var \OpenCloud\OpenStack
|
|
|
|
*/
|
|
|
|
private $client;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $params;
|
2014-06-11 15:57:24 +04:00
|
|
|
|
2014-06-13 19:22:21 +04:00
|
|
|
/**
|
|
|
|
* @var \OpenCloud\ObjectStore\Service
|
|
|
|
*/
|
|
|
|
private $objectStoreService;
|
2014-06-20 14:27:47 +04:00
|
|
|
|
2014-06-11 15:57:24 +04:00
|
|
|
/**
|
|
|
|
* @var \OpenCloud\ObjectStore\Resource\Container
|
|
|
|
*/
|
|
|
|
private $container;
|
|
|
|
|
|
|
|
public function __construct($params) {
|
2014-06-13 19:22:21 +04:00
|
|
|
if (!isset($params['container'])) {
|
|
|
|
$params['container'] = 'owncloud';
|
|
|
|
}
|
|
|
|
if (!isset($params['autocreate'])) {
|
|
|
|
// should only be true for tests
|
|
|
|
$params['autocreate'] = false;
|
|
|
|
}
|
2014-06-11 15:57:24 +04:00
|
|
|
|
2014-06-24 16:36:29 +04:00
|
|
|
if (isset($params['apiKey'])) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->client = new Rackspace($params['url'], $params);
|
2014-06-24 16:36:29 +04:00
|
|
|
} else {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->client = new OpenStack($params['url'], $params);
|
|
|
|
}
|
|
|
|
$this->params = $params;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function init() {
|
|
|
|
if ($this->container) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// the OpenCloud client library will default to 'cloudFiles' if $serviceName is null
|
|
|
|
$serviceName = null;
|
|
|
|
if ($this->params['serviceName']) {
|
|
|
|
$serviceName = $this->params['serviceName'];
|
2014-06-24 16:36:29 +04:00
|
|
|
}
|
2014-06-11 15:57:24 +04:00
|
|
|
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->objectStoreService = $this->client->objectStoreService($serviceName, $this->params['region']);
|
2014-06-13 19:22:21 +04:00
|
|
|
|
|
|
|
try {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->container = $this->objectStoreService->getContainer($this->params['container']);
|
2014-06-13 19:22:21 +04:00
|
|
|
} catch (ClientErrorResponseException $ex) {
|
|
|
|
// if the container does not exist and autocreate is true try to create the container on the fly
|
2014-06-26 13:30:00 +04:00
|
|
|
if (isset($this->params['autocreate']) && $this->params['autocreate'] === true) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->container = $this->objectStoreService->createContainer($this->params['container']);
|
2014-06-13 19:22:21 +04:00
|
|
|
} else {
|
|
|
|
throw $ex;
|
|
|
|
}
|
|
|
|
}
|
2014-06-11 15:57:24 +04:00
|
|
|
}
|
|
|
|
|
2014-06-20 14:27:47 +04:00
|
|
|
/**
|
|
|
|
* @return string the container name where objects are stored
|
|
|
|
*/
|
2014-06-18 17:20:26 +04:00
|
|
|
public function getStorageId() {
|
2014-06-24 16:42:52 +04:00
|
|
|
return $this->params['container'];
|
2014-06-18 17:20:26 +04:00
|
|
|
}
|
2014-06-18 00:06:56 +04:00
|
|
|
|
2014-06-12 00:15:42 +04:00
|
|
|
/**
|
2014-06-20 14:27:47 +04:00
|
|
|
* @param string $urn the unified resource name used to identify the object
|
|
|
|
* @param resource $stream stream with the data to write
|
2014-06-12 00:15:42 +04:00
|
|
|
* @throws Exception from openstack lib when something goes wrong
|
|
|
|
*/
|
2014-06-20 14:27:47 +04:00
|
|
|
public function writeObject($urn, $stream) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->init();
|
2014-06-20 14:27:47 +04:00
|
|
|
$this->container->uploadObject($urn, $stream);
|
2014-06-11 15:57:24 +04:00
|
|
|
}
|
2014-06-12 00:15:42 +04:00
|
|
|
|
|
|
|
/**
|
2014-06-20 14:27:47 +04:00
|
|
|
* @param string $urn the unified resource name used to identify the object
|
|
|
|
* @return resource stream with the read data
|
2014-06-12 00:15:42 +04:00
|
|
|
* @throws Exception from openstack lib when something goes wrong
|
|
|
|
*/
|
2014-06-20 14:27:47 +04:00
|
|
|
public function readObject($urn) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->init();
|
2014-06-11 15:57:24 +04:00
|
|
|
$object = $this->container->getObject($urn);
|
|
|
|
|
2014-06-20 14:27:47 +04:00
|
|
|
// we need to keep a reference to objectContent or
|
|
|
|
// the stream will be closed before we can do anything with it
|
|
|
|
/** @var $objectContent \Guzzle\Http\EntityBody * */
|
2014-06-11 15:57:24 +04:00
|
|
|
$objectContent = $object->getContent();
|
|
|
|
$objectContent->rewind();
|
2014-06-20 14:27:47 +04:00
|
|
|
|
|
|
|
// directly returning the object stream does not work because the GC seems to collect it, so we need a copy
|
|
|
|
$tmpStream = fopen('php://temp', 'r+');
|
|
|
|
stream_copy_to_stream($objectContent->getStream(), $tmpStream);
|
|
|
|
rewind($tmpStream);
|
|
|
|
|
|
|
|
return $tmpStream;
|
2014-06-11 15:57:24 +04:00
|
|
|
}
|
2014-06-12 00:15:42 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $urn Unified Resource Name
|
|
|
|
* @return void
|
|
|
|
* @throws Exception from openstack lib when something goes wrong
|
|
|
|
*/
|
2014-06-18 00:06:56 +04:00
|
|
|
public function deleteObject($urn) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->init();
|
2014-06-25 16:53:38 +04:00
|
|
|
// see https://github.com/rackspace/php-opencloud/issues/243#issuecomment-30032242
|
|
|
|
$this->container->dataObject()->setName($urn)->delete();
|
2014-06-11 15:57:24 +04:00
|
|
|
}
|
|
|
|
|
2014-06-13 19:22:21 +04:00
|
|
|
public function deleteContainer($recursive = false) {
|
2014-06-24 16:42:52 +04:00
|
|
|
$this->init();
|
2014-06-13 19:22:21 +04:00
|
|
|
$this->container->delete($recursive);
|
|
|
|
}
|
|
|
|
|
2014-06-11 15:57:24 +04:00
|
|
|
}
|