diff options
author | Lukas Reschke <lukas@owncloud.com> | 2014-10-01 13:07:28 +0200 |
---|---|---|
committer | Lukas Reschke <lukas@owncloud.com> | 2014-10-01 13:07:28 +0200 |
commit | 001f223baaadfb7eb22f1523db0e45b44fbc7a4b (patch) | |
tree | 26800e9a13adbd21f5b7612642a8ede4cdcda247 | |
parent | 553274b28f1b5dc6734f934e2acff39a24c368b7 (diff) | |
parent | 6d60630f08ad1b7aaa93e6d7a3aef4f90c86b038 (diff) | |
download | nextcloud-server-001f223baaadfb7eb22f1523db0e45b44fbc7a4b.tar.gz nextcloud-server-001f223baaadfb7eb22f1523db0e45b44fbc7a4b.zip |
Merge pull request #10104 from owncloud/s3-overwrite
Fix S3 copy and rename overwriting target directory
-rw-r--r-- | apps/files_external/lib/amazons3.php | 67 | ||||
-rw-r--r-- | apps/files_external/tests/amazons3.php | 28 |
2 files changed, 52 insertions, 43 deletions
diff --git a/apps/files_external/lib/amazons3.php b/apps/files_external/lib/amazons3.php index c82b7391021..9daac83e066 100644 --- a/apps/files_external/lib/amazons3.php +++ b/apps/files_external/lib/amazons3.php @@ -73,6 +73,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { sleep($this->timeout); } } + private function cleanKey($path) { if ($path === '.') { return '/'; @@ -90,13 +91,13 @@ class AmazonS3 extends \OC\Files\Storage\Common { $this->bucket = $params['bucket']; $scheme = ($params['use_ssl'] === 'false') ? 'http' : 'https'; $this->test = isset($params['test']); - $this->timeout = ( ! isset($params['timeout'])) ? 15 : $params['timeout']; - $params['region'] = ( ! isset($params['region']) || $params['region'] === '' ) ? 'eu-west-1' : $params['region']; - $params['hostname'] = ( !isset($params['hostname']) || $params['hostname'] === '' ) ? 's3.amazonaws.com' : $params['hostname']; + $this->timeout = (!isset($params['timeout'])) ? 15 : $params['timeout']; + $params['region'] = (!isset($params['region']) || $params['region'] === '') ? 'eu-west-1' : $params['region']; + $params['hostname'] = (!isset($params['hostname']) || $params['hostname'] === '') ? 's3.amazonaws.com' : $params['hostname']; if (!isset($params['port']) || $params['port'] === '') { $params['port'] = ($params['use_ssl'] === 'false') ? 80 : 443; } - $base_url = $scheme.'://'.$params['hostname'].':'.$params['port'].'/'; + $base_url = $scheme . '://' . $params['hostname'] . ':' . $params['port'] . '/'; $this->connection = S3Client::factory(array( 'key' => $params['key'], @@ -119,7 +120,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { 'waiter.interval' => 1, 'waiter.max_attempts' => 15 )); - $this->testTimeout(); + $this->testTimeout(); } catch (S3Exception $e) { \OCP\Util::writeLog('files_external', $e->getMessage(), \OCP\Util::ERROR); throw new \Exception("Creation of bucket failed."); @@ -128,8 +129,8 @@ class AmazonS3 extends \OC\Files\Storage\Common { if (!$this->file_exists('.')) { $result = $this->connection->putObject(array( 'Bucket' => $this->bucket, - 'Key' => $this->cleanKey('.'), - 'Body' => '', + 'Key' => $this->cleanKey('.'), + 'Body' => '', 'ContentType' => 'httpd/unix-directory', 'ContentLength' => 0 )); @@ -145,10 +146,10 @@ class AmazonS3 extends \OC\Files\Storage\Common { } try { - $result = $this->connection->putObject(array( + $this->connection->putObject(array( 'Bucket' => $this->bucket, - 'Key' => $path . '/', - 'Body' => '', + 'Key' => $path . '/', + 'Body' => '', 'ContentType' => 'httpd/unix-directory', 'ContentLength' => 0 )); @@ -187,6 +188,10 @@ class AmazonS3 extends \OC\Files\Storage\Common { public function rmdir($path) { $path = $this->normalizePath($path); + if ($path === '.') { + return $this->clearBucket(); + } + if (!$this->file_exists($path)) { return false; } @@ -199,7 +204,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { )); try { - $result = $this->connection->deleteObjects(array( + $this->connection->deleteObjects(array( 'Bucket' => $this->bucket, 'Objects' => $objects['Contents'] )); @@ -212,6 +217,28 @@ class AmazonS3 extends \OC\Files\Storage\Common { return true; } + protected function clearBucket() { + try { + $this->connection->clearBucket($this->bucket); + // clearBucket() is not working with Ceph, so if it fails we try the slower approach + } catch (\Exception $e) { + try { + $iterator = $this->connection->getIterator('ListObjects', array( + 'Bucket' => $this->bucket + )); + + foreach ($iterator as $object) { + $this->connection->deleteObject(array( + 'Bucket' => $this->bucket, + 'Key' => $object['Key'] + )); + } + } catch (S3Exception $e) { + return false; + } + } + } + public function opendir($path) { $path = $this->normalizePath($path); @@ -271,7 +298,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { $stat['atime'] = time(); return $stat; - } catch(S3Exception $e) { + } catch (S3Exception $e) { \OCP\Util::writeLog('files_external', $e->getMessage(), \OCP\Util::ERROR); return false; } @@ -302,7 +329,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { public function unlink($path) { $path = $this->normalizePath($path); - if ( $this->is_dir($path) ) { + if ($this->is_dir($path)) { return $this->rmdir($path); } @@ -446,8 +473,10 @@ class AmazonS3 extends \OC\Files\Storage\Common { return false; } } else { - if ($this->file_exists($path2)) { - return false; + if ($this->is_dir($path2)) { + $this->rmdir($path2); + } else if ($this->file_exists($path2)) { + $this->unlink($path2); } try { @@ -463,7 +492,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { } $dh = $this->opendir($path1); - if(is_resource($dh)) { + if (is_resource($dh)) { while (($file = readdir($dh)) !== false) { if ($file === '.' || $file === '..') { continue; @@ -484,6 +513,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { $path2 = $this->normalizePath($path2); if ($this->is_file($path1)) { + if ($this->copy($path1, $path2) === false) { return false; } @@ -493,9 +523,6 @@ class AmazonS3 extends \OC\Files\Storage\Common { return false; } } else { - if ($this->file_exists($path2)) { - return false; - } if ($this->copy($path1, $path2) === false) { return false; @@ -534,7 +561,7 @@ class AmazonS3 extends \OC\Files\Storage\Common { } try { - $result= $this->connection->putObject(array( + $result = $this->connection->putObject(array( 'Bucket' => $this->bucket, 'Key' => $this->cleanKey(self::$tmpFiles[$tmpFile]), 'SourceFile' => $tmpFile, diff --git a/apps/files_external/tests/amazons3.php b/apps/files_external/tests/amazons3.php index a73b6307b01..8eaece6dad9 100644 --- a/apps/files_external/tests/amazons3.php +++ b/apps/files_external/tests/amazons3.php @@ -38,29 +38,11 @@ class AmazonS3 extends Storage { public function tearDown() { if ($this->instance) { - $connection = $this->instance->getConnection(); - - try { - // NOTE(berendt): clearBucket() is not working with Ceph - $iterator = $connection->getIterator('ListObjects', array( - 'Bucket' => $this->config['amazons3']['bucket'] - )); - - foreach ($iterator as $object) { - $connection->deleteObject(array( - 'Bucket' => $this->config['amazons3']['bucket'], - 'Key' => $object['Key'] - )); - } - } catch (S3Exception $e) { - } - - $connection->deleteBucket(array( - 'Bucket' => $this->config['amazons3']['bucket'] - )); - - //wait some seconds for completing the replication - sleep(30); + $this->instance->rmdir(''); } } + + public function testStat() { + $this->markTestSkipped('S3 doesn\'t update the parents folder mtime'); + } } |