Skip to content

Commit

Permalink
Merge pull request #18883 from adrb/optimize_swift_14116
Browse files Browse the repository at this point in the history
Optimize Openstack Swift files download
  • Loading branch information
rullzer authored Apr 30, 2020
2 parents 4472777 + 354c099 commit a1c1b35
Show file tree
Hide file tree
Showing 2 changed files with 48 additions and 18 deletions.
37 changes: 23 additions & 14 deletions lib/private/Files/ObjectStore/Swift.php
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,12 @@

namespace OC\Files\ObjectStore;

use GuzzleHttp\Client;
use GuzzleHttp\Exception\ClientException;
use GuzzleHttp\Exception\ConnectException;
use GuzzleHttp\Exception\RequestException;
use GuzzleHttp\Exception\BadResponseException;
use GuzzleHttp\HandlerStack;
use function GuzzleHttp\Psr7\stream_for;
use Icewind\Streams\RetryWrapper;
use OCP\Files\NotFoundException;
Expand Down Expand Up @@ -89,30 +95,33 @@ public function writeObject($urn, $stream) {
/**
* @param string $urn the unified resource name used to identify the object
* @return resource stream with the read data
* @throws \Exception from openstack lib when something goes wrong
* @throws \Exception from openstack or GuzzleHttp libs when something goes wrong
* @throws NotFoundException if file does not exist
*/
public function readObject($urn) {
try {
$object = $this->getContainer()->getObject($urn);

// we need to keep a reference to objectContent or
// the stream will be closed before we can do anything with it
$objectContent = $object->download();
} catch (BadResponseError $e) {
if ($e->getResponse()->getStatusCode() === 404) {
$publicUri = $this->getContainer()->getObject($urn)->getPublicUri();
$tokenId = $this->swiftFactory->getCachedTokenId();

$response = (new Client())->request('GET', $publicUri,
[
'stream' => true,
'headers' => [
'X-Auth-Token' => $tokenId,
'Cache-Control' => 'no-cache'
],
]
);

} catch (BadResponseException $e) {
if ($e->getResponse() && $e->getResponse()->getStatusCode() === 404) {
throw new NotFoundException("object $urn not found in object store");
} else {
throw $e;
}
}
$objectContent->rewind();

$stream = $objectContent->detach();
// save the object content in the context of the stream to prevent it being gc'd until the stream is closed
stream_context_set_option($stream, 'swift', 'content', $objectContent);

return RetryWrapper::wrap($stream);
return RetryWrapper::wrap($response->getBody()->detach());
}

/**
Expand Down
29 changes: 25 additions & 4 deletions lib/private/Files/ObjectStore/SwiftFactory.php
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,25 @@ public function __construct(ICache $cache, array $params, ILogger $logger) {
$this->logger = $logger;
}

/**
* Gets currently cached token id
*
* @return string
* @throws StorageAuthException
*/
public function getCachedTokenId() {
if ( !isset($this->params['cachedToken']) ) {
throw new StorageAuthException('Unauthenticated ObjectStore connection');
}

// Is it V2 token?
if ( isset($this->params['cachedToken']['token']) ) {
return $this->params['cachedToken']['token']['id'];
}

return $this->params['cachedToken']['id'];
}

private function getCachedToken(string $cacheKey) {
$cachedTokenString = $this->cache->get($cacheKey . '/token');
if ($cachedTokenString) {
Expand All @@ -83,20 +102,22 @@ private function getCachedToken(string $cacheKey) {
private function cacheToken(Token $token, string $serviceUrl, string $cacheKey) {
if ($token instanceof \OpenStack\Identity\v3\Models\Token) {
// for v3 the catalog is cached as part of the token, so no need to cache $serviceUrl separately
$value = json_encode($token->export());
$value = $token->export();
} else {
/** @var \OpenStack\Identity\v2\Models\Token $token */
$value = json_encode([
$value = [
'serviceUrl' => $serviceUrl,
'token' => [
'issued_at' => $token->issuedAt->format('c'),
'expires' => $token->expires->format('c'),
'id' => $token->id,
'tenant' => $token->tenant
]
]);
];
}
$this->cache->set($cacheKey . '/token', $value);

$this->params['cachedToken'] = $value;
$this->cache->set($cacheKey . '/token', json_encode($value));
}

/**
Expand Down

0 comments on commit a1c1b35

Please sign in to comment.