Localisation updates from http://translatewiki.net.
[mediawiki.git] / includes / filebackend / SwiftFileBackend.php
blobf0b289e13bf444a4e5a90a0f789543bd9d248bec
1 <?php
2 /**
3 * OpenStack Swift based file backend.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
20 * @file
21 * @ingroup FileBackend
22 * @author Russ Nelson
23 * @author Aaron Schulz
26 /**
27 * @brief Class for an OpenStack Swift based file backend.
29 * This requires the SwiftCloudFiles MediaWiki extension, which includes
30 * the php-cloudfiles library (https://github.com/rackspace/php-cloudfiles).
31 * php-cloudfiles requires the curl, fileinfo, and mb_string PHP extensions.
33 * Status messages should avoid mentioning the Swift account name.
34 * Likewise, error suppression should be used to avoid path disclosure.
36 * @ingroup FileBackend
37 * @since 1.19
39 class SwiftFileBackend extends FileBackendStore {
40 /** @var CF_Authentication */
41 protected $auth; // Swift authentication handler
42 protected $authTTL; // integer seconds
43 protected $swiftTempUrlKey; // string; shared secret value for making temp urls
44 protected $swiftAnonUser; // string; username to handle unauthenticated requests
45 protected $swiftUseCDN; // boolean; whether CloudFiles CDN is enabled
46 protected $swiftCDNExpiry; // integer; how long to cache things in the CDN
47 protected $swiftCDNPurgable; // boolean; whether object CDN purging is enabled
49 // Rados Gateway specific options
50 protected $rgwS3AccessKey; // string; S3 access key
51 protected $rgwS3SecretKey; // string; S3 authentication key
53 /** @var CF_Connection */
54 protected $conn; // Swift connection handle
55 protected $sessionStarted = 0; // integer UNIX timestamp
57 /** @var CloudFilesException */
58 protected $connException;
59 protected $connErrorTime = 0; // UNIX timestamp
61 /** @var BagOStuff */
62 protected $srvCache;
64 /** @var ProcessCacheLRU */
65 protected $connContainerCache; // container object cache
67 /**
68 * @see FileBackendStore::__construct()
69 * Additional $config params include:
70 * - swiftAuthUrl : Swift authentication server URL
71 * - swiftUser : Swift user used by MediaWiki (account:username)
72 * - swiftKey : Swift authentication key for the above user
73 * - swiftAuthTTL : Swift authentication TTL (seconds)
74 * - swiftTempUrlKey : Swift "X-Account-Meta-Temp-URL-Key" value on the account.
75 * Do not set this until it has been set in the backend.
76 * - swiftAnonUser : Swift user used for end-user requests (account:username).
77 * If set, then views of public containers are assumed to go
78 * through this user. If not set, then public containers are
79 * accessible to unauthenticated requests via ".r:*" in the ACL.
80 * - swiftUseCDN : Whether a Cloud Files Content Delivery Network is set up
81 * - swiftCDNExpiry : How long (in seconds) to store content in the CDN.
82 * If files may likely change, this should probably not exceed
83 * a few days. For example, deletions may take this long to apply.
84 * If object purging is enabled, however, this is not an issue.
85 * - swiftCDNPurgable : Whether object purge requests are allowed by the CDN.
86 * - shardViaHashLevels : Map of container names to sharding config with:
87 * - base : base of hash characters, 16 or 36
88 * - levels : the number of hash levels (and digits)
89 * - repeat : hash subdirectories are prefixed with all the
90 * parent hash directory names (e.g. "a/ab/abc")
91 * - cacheAuthInfo : Whether to cache authentication tokens in APC, XCache, ect.
92 * If those are not available, then the main cache will be used.
93 * This is probably insecure in shared hosting environments.
94 * - rgwS3AccessKey : Ragos Gateway S3 "access key" value on the account.
95 * Do not set this until it has been set in the backend.
96 * This is used for generating expiring pre-authenticated URLs.
97 * Only use this when using rgw and to work around
98 * http://tracker.newdream.net/issues/3454.
99 * - rgwS3SecretKey : Ragos Gateway S3 "secret key" value on the account.
100 * Do not set this until it has been set in the backend.
101 * This is used for generating expiring pre-authenticated URLs.
102 * Only use this when using rgw and to work around
103 * http://tracker.newdream.net/issues/3454.
105 public function __construct( array $config ) {
106 parent::__construct( $config );
107 if ( !MWInit::classExists( 'CF_Constants' ) ) {
108 throw new MWException( 'SwiftCloudFiles extension not installed.' );
110 // Required settings
111 $this->auth = new CF_Authentication(
112 $config['swiftUser'],
113 $config['swiftKey'],
114 null, // account; unused
115 $config['swiftAuthUrl']
117 // Optional settings
118 $this->authTTL = isset( $config['swiftAuthTTL'] )
119 ? $config['swiftAuthTTL']
120 : 5 * 60; // some sane number
121 $this->swiftAnonUser = isset( $config['swiftAnonUser'] )
122 ? $config['swiftAnonUser']
123 : '';
124 $this->swiftTempUrlKey = isset( $config['swiftTempUrlKey'] )
125 ? $config['swiftTempUrlKey']
126 : '';
127 $this->shardViaHashLevels = isset( $config['shardViaHashLevels'] )
128 ? $config['shardViaHashLevels']
129 : '';
130 $this->swiftUseCDN = isset( $config['swiftUseCDN'] )
131 ? $config['swiftUseCDN']
132 : false;
133 $this->swiftCDNExpiry = isset( $config['swiftCDNExpiry'] )
134 ? $config['swiftCDNExpiry']
135 : 12 * 3600; // 12 hours is safe (tokens last 24 hours per http://docs.openstack.org)
136 $this->swiftCDNPurgable = isset( $config['swiftCDNPurgable'] )
137 ? $config['swiftCDNPurgable']
138 : true;
139 $this->rgwS3AccessKey = isset( $config['rgwS3AccessKey'] )
140 ? $config['rgwS3AccessKey']
141 : '';
142 $this->rgwS3SecretKey = isset( $config['rgwS3SecretKey'] )
143 ? $config['rgwS3SecretKey']
144 : '';
145 // Cache container information to mask latency
146 $this->memCache = wfGetMainCache();
147 // Process cache for container info
148 $this->connContainerCache = new ProcessCacheLRU( 300 );
149 // Cache auth token information to avoid RTTs
150 if ( !empty( $config['cacheAuthInfo'] ) ) {
151 if ( PHP_SAPI === 'cli' ) {
152 $this->srvCache = wfGetMainCache(); // preferrably memcached
153 } else {
154 try { // look for APC, XCache, WinCache, ect...
155 $this->srvCache = ObjectCache::newAccelerator( array() );
156 } catch ( Exception $e ) {}
159 $this->srvCache = $this->srvCache ? $this->srvCache : new EmptyBagOStuff();
163 * @see FileBackendStore::resolveContainerPath()
164 * @return null
166 protected function resolveContainerPath( $container, $relStoragePath ) {
167 if ( !mb_check_encoding( $relStoragePath, 'UTF-8' ) ) { // mb_string required by CF
168 return null; // not UTF-8, makes it hard to use CF and the swift HTTP API
169 } elseif ( strlen( urlencode( $relStoragePath ) ) > 1024 ) {
170 return null; // too long for Swift
172 return $relStoragePath;
176 * @see FileBackendStore::isPathUsableInternal()
177 * @return bool
179 public function isPathUsableInternal( $storagePath ) {
180 list( $container, $rel ) = $this->resolveStoragePathReal( $storagePath );
181 if ( $rel === null ) {
182 return false; // invalid
185 try {
186 $this->getContainer( $container );
187 return true; // container exists
188 } catch ( NoSuchContainerException $e ) {
189 } catch ( CloudFilesException $e ) { // some other exception?
190 $this->handleException( $e, null, __METHOD__, array( 'path' => $storagePath ) );
193 return false;
197 * @param $headers array
198 * @return array
200 protected function sanitizeHdrs( array $headers ) {
201 // By default, Swift has annoyingly low maximum header value limits
202 if ( isset( $headers['Content-Disposition'] ) ) {
203 $headers['Content-Disposition'] = $this->truncDisp( $headers['Content-Disposition'] );
205 return $headers;
209 * @param $disposition string Content-Disposition header value
210 * @return string Truncated Content-Disposition header value to meet Swift limits
212 protected function truncDisp( $disposition ) {
213 $res = '';
214 foreach ( explode( ';', $disposition ) as $part ) {
215 $part = trim( $part );
216 $new = ( $res === '' ) ? $part : "{$res};{$part}";
217 if ( strlen( $new ) <= 255 ) {
218 $res = $new;
219 } else {
220 break; // too long; sigh
223 return $res;
227 * @see FileBackendStore::doCreateInternal()
228 * @return Status
230 protected function doCreateInternal( array $params ) {
231 $status = Status::newGood();
233 list( $dstCont, $dstRel ) = $this->resolveStoragePathReal( $params['dst'] );
234 if ( $dstRel === null ) {
235 $status->fatal( 'backend-fail-invalidpath', $params['dst'] );
236 return $status;
239 // (a) Check the destination container and object
240 try {
241 $dContObj = $this->getContainer( $dstCont );
242 } catch ( NoSuchContainerException $e ) {
243 $status->fatal( 'backend-fail-create', $params['dst'] );
244 return $status;
245 } catch ( CloudFilesException $e ) { // some other exception?
246 $this->handleException( $e, $status, __METHOD__, $params );
247 return $status;
250 // (b) Get a SHA-1 hash of the object
251 $sha1Hash = wfBaseConvert( sha1( $params['content'] ), 16, 36, 31 );
253 // (c) Actually create the object
254 try {
255 // Create a fresh CF_Object with no fields preloaded.
256 // We don't want to preserve headers, metadata, and such.
257 $obj = new CF_Object( $dContObj, $dstRel, false, false ); // skip HEAD
258 $obj->setMetadataValues( array( 'Sha1base36' => $sha1Hash ) );
259 // Manually set the ETag (https://github.com/rackspace/php-cloudfiles/issues/59).
260 // The MD5 here will be checked within Swift against its own MD5.
261 $obj->set_etag( md5( $params['content'] ) );
262 // Use the same content type as StreamFile for security
263 $obj->content_type = StreamFile::contentTypeFromPath( $params['dst'] );
264 if ( !strlen( $obj->content_type ) ) { // special case
265 $obj->content_type = 'unknown/unknown';
267 // Set any other custom headers if requested
268 if ( isset( $params['headers'] ) ) {
269 $obj->headers += $this->sanitizeHdrs( $params['headers'] );
271 if ( !empty( $params['async'] ) ) { // deferred
272 $op = $obj->write_async( $params['content'] );
273 $status->value = new SwiftFileOpHandle( $this, $params, 'Create', $op );
274 $status->value->affectedObjects[] = $obj;
275 } else { // actually write the object in Swift
276 $obj->write( $params['content'] );
277 $this->purgeCDNCache( array( $obj ) );
279 } catch ( CDNNotEnabledException $e ) {
280 // CDN not enabled; nothing to see here
281 } catch ( BadContentTypeException $e ) {
282 $status->fatal( 'backend-fail-contenttype', $params['dst'] );
283 } catch ( CloudFilesException $e ) { // some other exception?
284 $this->handleException( $e, $status, __METHOD__, $params );
287 return $status;
291 * @see SwiftFileBackend::doExecuteOpHandlesInternal()
293 protected function _getResponseCreate( CF_Async_Op $cfOp, Status $status, array $params ) {
294 try {
295 $cfOp->getLastResponse();
296 } catch ( BadContentTypeException $e ) {
297 $status->fatal( 'backend-fail-contenttype', $params['dst'] );
302 * @see FileBackendStore::doStoreInternal()
303 * @return Status
305 protected function doStoreInternal( array $params ) {
306 $status = Status::newGood();
308 list( $dstCont, $dstRel ) = $this->resolveStoragePathReal( $params['dst'] );
309 if ( $dstRel === null ) {
310 $status->fatal( 'backend-fail-invalidpath', $params['dst'] );
311 return $status;
314 // (a) Check the destination container and object
315 try {
316 $dContObj = $this->getContainer( $dstCont );
317 } catch ( NoSuchContainerException $e ) {
318 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
319 return $status;
320 } catch ( CloudFilesException $e ) { // some other exception?
321 $this->handleException( $e, $status, __METHOD__, $params );
322 return $status;
325 // (b) Get a SHA-1 hash of the object
326 wfSuppressWarnings();
327 $sha1Hash = sha1_file( $params['src'] );
328 wfRestoreWarnings();
329 if ( $sha1Hash === false ) { // source doesn't exist?
330 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
331 return $status;
333 $sha1Hash = wfBaseConvert( $sha1Hash, 16, 36, 31 );
335 // (c) Actually store the object
336 try {
337 // Create a fresh CF_Object with no fields preloaded.
338 // We don't want to preserve headers, metadata, and such.
339 $obj = new CF_Object( $dContObj, $dstRel, false, false ); // skip HEAD
340 $obj->setMetadataValues( array( 'Sha1base36' => $sha1Hash ) );
341 // The MD5 here will be checked within Swift against its own MD5.
342 $obj->set_etag( md5_file( $params['src'] ) );
343 // Use the same content type as StreamFile for security
344 $obj->content_type = StreamFile::contentTypeFromPath( $params['dst'] );
345 if ( !strlen( $obj->content_type ) ) { // special case
346 $obj->content_type = 'unknown/unknown';
348 // Set any other custom headers if requested
349 if ( isset( $params['headers'] ) ) {
350 $obj->headers += $this->sanitizeHdrs( $params['headers'] );
352 if ( !empty( $params['async'] ) ) { // deferred
353 wfSuppressWarnings();
354 $fp = fopen( $params['src'], 'rb' );
355 wfRestoreWarnings();
356 if ( !$fp ) {
357 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
358 } else {
359 $op = $obj->write_async( $fp, filesize( $params['src'] ), true );
360 $status->value = new SwiftFileOpHandle( $this, $params, 'Store', $op );
361 $status->value->resourcesToClose[] = $fp;
362 $status->value->affectedObjects[] = $obj;
364 } else { // actually write the object in Swift
365 $obj->load_from_filename( $params['src'], true ); // calls $obj->write()
366 $this->purgeCDNCache( array( $obj ) );
368 } catch ( CDNNotEnabledException $e ) {
369 // CDN not enabled; nothing to see here
370 } catch ( BadContentTypeException $e ) {
371 $status->fatal( 'backend-fail-contenttype', $params['dst'] );
372 } catch ( IOException $e ) {
373 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
374 } catch ( CloudFilesException $e ) { // some other exception?
375 $this->handleException( $e, $status, __METHOD__, $params );
378 return $status;
382 * @see SwiftFileBackend::doExecuteOpHandlesInternal()
384 protected function _getResponseStore( CF_Async_Op $cfOp, Status $status, array $params ) {
385 try {
386 $cfOp->getLastResponse();
387 } catch ( BadContentTypeException $e ) {
388 $status->fatal( 'backend-fail-contenttype', $params['dst'] );
389 } catch ( IOException $e ) {
390 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
395 * @see FileBackendStore::doCopyInternal()
396 * @return Status
398 protected function doCopyInternal( array $params ) {
399 $status = Status::newGood();
401 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
402 if ( $srcRel === null ) {
403 $status->fatal( 'backend-fail-invalidpath', $params['src'] );
404 return $status;
407 list( $dstCont, $dstRel ) = $this->resolveStoragePathReal( $params['dst'] );
408 if ( $dstRel === null ) {
409 $status->fatal( 'backend-fail-invalidpath', $params['dst'] );
410 return $status;
413 // (a) Check the source/destination containers and destination object
414 try {
415 $sContObj = $this->getContainer( $srcCont );
416 $dContObj = $this->getContainer( $dstCont );
417 } catch ( NoSuchContainerException $e ) {
418 if ( empty( $params['ignoreMissingSource'] ) || isset( $sContObj ) ) {
419 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
421 return $status;
422 } catch ( CloudFilesException $e ) { // some other exception?
423 $this->handleException( $e, $status, __METHOD__, $params );
424 return $status;
427 // (b) Actually copy the file to the destination
428 try {
429 $dstObj = new CF_Object( $dContObj, $dstRel, false, false ); // skip HEAD
430 $hdrs = array(); // source file headers to override with new values
431 // Set any other custom headers if requested
432 if ( isset( $params['headers'] ) ) {
433 $hdrs += $this->sanitizeHdrs( $params['headers'] );
435 if ( !empty( $params['async'] ) ) { // deferred
436 $op = $sContObj->copy_object_to_async( $srcRel, $dContObj, $dstRel, null, $hdrs );
437 $status->value = new SwiftFileOpHandle( $this, $params, 'Copy', $op );
438 $status->value->affectedObjects[] = $dstObj;
439 } else { // actually write the object in Swift
440 $sContObj->copy_object_to( $srcRel, $dContObj, $dstRel, null, $hdrs );
441 $this->purgeCDNCache( array( $dstObj ) );
443 } catch ( CDNNotEnabledException $e ) {
444 // CDN not enabled; nothing to see here
445 } catch ( NoSuchObjectException $e ) { // source object does not exist
446 if ( empty( $params['ignoreMissingSource'] ) ) {
447 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
449 } catch ( CloudFilesException $e ) { // some other exception?
450 $this->handleException( $e, $status, __METHOD__, $params );
453 return $status;
457 * @see SwiftFileBackend::doExecuteOpHandlesInternal()
459 protected function _getResponseCopy( CF_Async_Op $cfOp, Status $status, array $params ) {
460 try {
461 $cfOp->getLastResponse();
462 } catch ( NoSuchObjectException $e ) { // source object does not exist
463 $status->fatal( 'backend-fail-copy', $params['src'], $params['dst'] );
468 * @see FileBackendStore::doMoveInternal()
469 * @return Status
471 protected function doMoveInternal( array $params ) {
472 $status = Status::newGood();
474 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
475 if ( $srcRel === null ) {
476 $status->fatal( 'backend-fail-invalidpath', $params['src'] );
477 return $status;
480 list( $dstCont, $dstRel ) = $this->resolveStoragePathReal( $params['dst'] );
481 if ( $dstRel === null ) {
482 $status->fatal( 'backend-fail-invalidpath', $params['dst'] );
483 return $status;
486 // (a) Check the source/destination containers and destination object
487 try {
488 $sContObj = $this->getContainer( $srcCont );
489 $dContObj = $this->getContainer( $dstCont );
490 } catch ( NoSuchContainerException $e ) {
491 if ( empty( $params['ignoreMissingSource'] ) || isset( $sContObj ) ) {
492 $status->fatal( 'backend-fail-move', $params['src'], $params['dst'] );
494 return $status;
495 } catch ( CloudFilesException $e ) { // some other exception?
496 $this->handleException( $e, $status, __METHOD__, $params );
497 return $status;
500 // (b) Actually move the file to the destination
501 try {
502 $srcObj = new CF_Object( $sContObj, $srcRel, false, false ); // skip HEAD
503 $dstObj = new CF_Object( $dContObj, $dstRel, false, false ); // skip HEAD
504 $hdrs = array(); // source file headers to override with new values
505 // Set any other custom headers if requested
506 if ( isset( $params['headers'] ) ) {
507 $hdrs += $this->sanitizeHdrs( $params['headers'] );
509 if ( !empty( $params['async'] ) ) { // deferred
510 $op = $sContObj->move_object_to_async( $srcRel, $dContObj, $dstRel, null, $hdrs );
511 $status->value = new SwiftFileOpHandle( $this, $params, 'Move', $op );
512 $status->value->affectedObjects[] = $srcObj;
513 $status->value->affectedObjects[] = $dstObj;
514 } else { // actually write the object in Swift
515 $sContObj->move_object_to( $srcRel, $dContObj, $dstRel, null, $hdrs );
516 $this->purgeCDNCache( array( $srcObj ) );
517 $this->purgeCDNCache( array( $dstObj ) );
519 } catch ( CDNNotEnabledException $e ) {
520 // CDN not enabled; nothing to see here
521 } catch ( NoSuchObjectException $e ) { // source object does not exist
522 if ( empty( $params['ignoreMissingSource'] ) ) {
523 $status->fatal( 'backend-fail-move', $params['src'], $params['dst'] );
525 } catch ( CloudFilesException $e ) { // some other exception?
526 $this->handleException( $e, $status, __METHOD__, $params );
529 return $status;
533 * @see SwiftFileBackend::doExecuteOpHandlesInternal()
535 protected function _getResponseMove( CF_Async_Op $cfOp, Status $status, array $params ) {
536 try {
537 $cfOp->getLastResponse();
538 } catch ( NoSuchObjectException $e ) { // source object does not exist
539 $status->fatal( 'backend-fail-move', $params['src'], $params['dst'] );
544 * @see FileBackendStore::doDeleteInternal()
545 * @return Status
547 protected function doDeleteInternal( array $params ) {
548 $status = Status::newGood();
550 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
551 if ( $srcRel === null ) {
552 $status->fatal( 'backend-fail-invalidpath', $params['src'] );
553 return $status;
556 try {
557 $sContObj = $this->getContainer( $srcCont );
558 $srcObj = new CF_Object( $sContObj, $srcRel, false, false ); // skip HEAD
559 if ( !empty( $params['async'] ) ) { // deferred
560 $op = $sContObj->delete_object_async( $srcRel );
561 $status->value = new SwiftFileOpHandle( $this, $params, 'Delete', $op );
562 $status->value->affectedObjects[] = $srcObj;
563 } else { // actually write the object in Swift
564 $sContObj->delete_object( $srcRel );
565 $this->purgeCDNCache( array( $srcObj ) );
567 } catch ( CDNNotEnabledException $e ) {
568 // CDN not enabled; nothing to see here
569 } catch ( NoSuchContainerException $e ) {
570 if ( empty( $params['ignoreMissingSource'] ) ) {
571 $status->fatal( 'backend-fail-delete', $params['src'] );
573 } catch ( NoSuchObjectException $e ) {
574 if ( empty( $params['ignoreMissingSource'] ) ) {
575 $status->fatal( 'backend-fail-delete', $params['src'] );
577 } catch ( CloudFilesException $e ) { // some other exception?
578 $this->handleException( $e, $status, __METHOD__, $params );
581 return $status;
585 * @see SwiftFileBackend::doExecuteOpHandlesInternal()
587 protected function _getResponseDelete( CF_Async_Op $cfOp, Status $status, array $params ) {
588 try {
589 $cfOp->getLastResponse();
590 } catch ( NoSuchContainerException $e ) {
591 $status->fatal( 'backend-fail-delete', $params['src'] );
592 } catch ( NoSuchObjectException $e ) {
593 if ( empty( $params['ignoreMissingSource'] ) ) {
594 $status->fatal( 'backend-fail-delete', $params['src'] );
600 * @see FileBackendStore::doDescribeInternal()
601 * @return Status
603 protected function doDescribeInternal( array $params ) {
604 $status = Status::newGood();
606 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
607 if ( $srcRel === null ) {
608 $status->fatal( 'backend-fail-invalidpath', $params['src'] );
609 return $status;
612 try {
613 $sContObj = $this->getContainer( $srcCont );
614 // Get the latest version of the current metadata
615 $srcObj = $sContObj->get_object( $srcRel,
616 $this->headersFromParams( array( 'latest' => true ) ) );
617 // Merge in the metadata updates...
618 if ( isset( $params['headers'] ) ) {
619 $srcObj->headers = $this->sanitizeHdrs( $params['headers'] ) + $srcObj->headers;
621 $srcObj->sync_metadata(); // save to Swift
622 $this->purgeCDNCache( array( $srcObj ) );
623 } catch ( CDNNotEnabledException $e ) {
624 // CDN not enabled; nothing to see here
625 } catch ( NoSuchContainerException $e ) {
626 $status->fatal( 'backend-fail-describe', $params['src'] );
627 } catch ( NoSuchObjectException $e ) {
628 $status->fatal( 'backend-fail-describe', $params['src'] );
629 } catch ( CloudFilesException $e ) { // some other exception?
630 $this->handleException( $e, $status, __METHOD__, $params );
633 return $status;
637 * @see FileBackendStore::doPrepareInternal()
638 * @return Status
640 protected function doPrepareInternal( $fullCont, $dir, array $params ) {
641 $status = Status::newGood();
643 // (a) Check if container already exists
644 try {
645 $this->getContainer( $fullCont );
646 // NoSuchContainerException not thrown: container must exist
647 return $status; // already exists
648 } catch ( NoSuchContainerException $e ) {
649 // NoSuchContainerException thrown: container does not exist
650 } catch ( CloudFilesException $e ) { // some other exception?
651 $this->handleException( $e, $status, __METHOD__, $params );
652 return $status;
655 // (b) Create container as needed
656 try {
657 $contObj = $this->createContainer( $fullCont );
658 if ( !empty( $params['noAccess'] ) ) {
659 // Make container private to end-users...
660 $status->merge( $this->doSecureInternal( $fullCont, $dir, $params ) );
661 } else {
662 // Make container public to end-users...
663 $status->merge( $this->doPublishInternal( $fullCont, $dir, $params ) );
665 if ( $this->swiftUseCDN ) { // Rackspace style CDN
666 $contObj->make_public( $this->swiftCDNExpiry );
668 } catch ( CDNNotEnabledException $e ) {
669 // CDN not enabled; nothing to see here
670 } catch ( CloudFilesException $e ) { // some other exception?
671 $this->handleException( $e, $status, __METHOD__, $params );
672 return $status;
675 return $status;
679 * @see FileBackendStore::doSecureInternal()
680 * @return Status
682 protected function doSecureInternal( $fullCont, $dir, array $params ) {
683 $status = Status::newGood();
684 if ( empty( $params['noAccess'] ) ) {
685 return $status; // nothing to do
688 // Restrict container from end-users...
689 try {
690 // doPrepareInternal() should have been called,
691 // so the Swift container should already exist...
692 $contObj = $this->getContainer( $fullCont ); // normally a cache hit
693 // NoSuchContainerException not thrown: container must exist
695 // Make container private to end-users...
696 $status->merge( $this->setContainerAccess(
697 $contObj,
698 array( $this->auth->username ), // read
699 array( $this->auth->username ) // write
700 ) );
701 if ( $this->swiftUseCDN && $contObj->is_public() ) { // Rackspace style CDN
702 $contObj->make_private();
704 } catch ( CDNNotEnabledException $e ) {
705 // CDN not enabled; nothing to see here
706 } catch ( CloudFilesException $e ) { // some other exception?
707 $this->handleException( $e, $status, __METHOD__, $params );
710 return $status;
714 * @see FileBackendStore::doPublishInternal()
715 * @return Status
717 protected function doPublishInternal( $fullCont, $dir, array $params ) {
718 $status = Status::newGood();
720 // Unrestrict container from end-users...
721 try {
722 // doPrepareInternal() should have been called,
723 // so the Swift container should already exist...
724 $contObj = $this->getContainer( $fullCont ); // normally a cache hit
725 // NoSuchContainerException not thrown: container must exist
727 // Make container public to end-users...
728 if ( $this->swiftAnonUser != '' ) {
729 $status->merge( $this->setContainerAccess(
730 $contObj,
731 array( $this->auth->username, $this->swiftAnonUser ), // read
732 array( $this->auth->username, $this->swiftAnonUser ) // write
733 ) );
734 } else {
735 $status->merge( $this->setContainerAccess(
736 $contObj,
737 array( $this->auth->username, '.r:*' ), // read
738 array( $this->auth->username ) // write
739 ) );
741 if ( $this->swiftUseCDN && !$contObj->is_public() ) { // Rackspace style CDN
742 $contObj->make_public();
744 } catch ( CDNNotEnabledException $e ) {
745 // CDN not enabled; nothing to see here
746 } catch ( CloudFilesException $e ) { // some other exception?
747 $this->handleException( $e, $status, __METHOD__, $params );
750 return $status;
754 * @see FileBackendStore::doCleanInternal()
755 * @return Status
757 protected function doCleanInternal( $fullCont, $dir, array $params ) {
758 $status = Status::newGood();
760 // Only containers themselves can be removed, all else is virtual
761 if ( $dir != '' ) {
762 return $status; // nothing to do
765 // (a) Check the container
766 try {
767 $contObj = $this->getContainer( $fullCont, true );
768 } catch ( NoSuchContainerException $e ) {
769 return $status; // ok, nothing to do
770 } catch ( CloudFilesException $e ) { // some other exception?
771 $this->handleException( $e, $status, __METHOD__, $params );
772 return $status;
775 // (b) Delete the container if empty
776 if ( $contObj->object_count == 0 ) {
777 try {
778 $this->deleteContainer( $fullCont );
779 } catch ( NoSuchContainerException $e ) {
780 return $status; // race?
781 } catch ( NonEmptyContainerException $e ) {
782 return $status; // race? consistency delay?
783 } catch ( CloudFilesException $e ) { // some other exception?
784 $this->handleException( $e, $status, __METHOD__, $params );
785 return $status;
789 return $status;
793 * @see FileBackendStore::doFileExists()
794 * @return array|bool|null
796 protected function doGetFileStat( array $params ) {
797 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
798 if ( $srcRel === null ) {
799 return false; // invalid storage path
802 $stat = false;
803 try {
804 $contObj = $this->getContainer( $srcCont );
805 $srcObj = $contObj->get_object( $srcRel, $this->headersFromParams( $params ) );
806 $this->addMissingMetadata( $srcObj, $params['src'] );
807 $stat = array(
808 // Convert dates like "Tue, 03 Jan 2012 22:01:04 GMT" to TS_MW
809 'mtime' => wfTimestamp( TS_MW, $srcObj->last_modified ),
810 'size' => (int)$srcObj->content_length,
811 'sha1' => $srcObj->getMetadataValue( 'Sha1base36' )
813 } catch ( NoSuchContainerException $e ) {
814 } catch ( NoSuchObjectException $e ) {
815 } catch ( CloudFilesException $e ) { // some other exception?
816 $stat = null;
817 $this->handleException( $e, null, __METHOD__, $params );
820 return $stat;
824 * Fill in any missing object metadata and save it to Swift
826 * @param $obj CF_Object
827 * @param string $path Storage path to object
828 * @return bool Success
829 * @throws Exception cloudfiles exceptions
831 protected function addMissingMetadata( CF_Object $obj, $path ) {
832 if ( $obj->getMetadataValue( 'Sha1base36' ) !== null ) {
833 return true; // nothing to do
835 wfProfileIn( __METHOD__ );
836 trigger_error( "$path was not stored with SHA-1 metadata.", E_USER_WARNING );
837 $status = Status::newGood();
838 $scopeLockS = $this->getScopedFileLocks( array( $path ), LockManager::LOCK_UW, $status );
839 if ( $status->isOK() ) {
840 $tmpFile = $this->getLocalCopy( array( 'src' => $path, 'latest' => 1 ) );
841 if ( $tmpFile ) {
842 $hash = $tmpFile->getSha1Base36();
843 if ( $hash !== false ) {
844 $obj->setMetadataValues( array( 'Sha1base36' => $hash ) );
845 $obj->sync_metadata(); // save to Swift
846 wfProfileOut( __METHOD__ );
847 return true; // success
851 trigger_error( "Unable to set SHA-1 metadata for $path", E_USER_WARNING );
852 $obj->setMetadataValues( array( 'Sha1base36' => false ) );
853 wfProfileOut( __METHOD__ );
854 return false; // failed
858 * @see FileBackendStore::doGetFileContentsMulti()
859 * @return Array
861 protected function doGetFileContentsMulti( array $params ) {
862 $contents = array();
864 $ep = array_diff_key( $params, array( 'srcs' => 1 ) ); // for error logging
865 // Blindly create tmp files and stream to them, catching any exception if the file does
866 // not exist. Doing stats here is useless and will loop infinitely in addMissingMetadata().
867 foreach ( array_chunk( $params['srcs'], $params['concurrency'] ) as $pathBatch ) {
868 $cfOps = array(); // (path => CF_Async_Op)
870 foreach ( $pathBatch as $path ) { // each path in this concurrent batch
871 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $path );
872 if ( $srcRel === null ) {
873 $contents[$path] = false;
874 continue;
876 $data = false;
877 try {
878 $sContObj = $this->getContainer( $srcCont );
879 $obj = new CF_Object( $sContObj, $srcRel, false, false ); // skip HEAD
880 // Create a new temporary memory file...
881 $handle = fopen( 'php://temp', 'wb' );
882 if ( $handle ) {
883 $headers = $this->headersFromParams( $params );
884 if ( count( $pathBatch ) > 1 ) {
885 $cfOps[$path] = $obj->stream_async( $handle, $headers );
886 $cfOps[$path]->_file_handle = $handle; // close this later
887 } else {
888 $obj->stream( $handle, $headers );
889 rewind( $handle ); // start from the beginning
890 $data = stream_get_contents( $handle );
891 fclose( $handle );
893 } else {
894 $data = false;
896 } catch ( NoSuchContainerException $e ) {
897 $data = false;
898 } catch ( NoSuchObjectException $e ) {
899 $data = false;
900 } catch ( CloudFilesException $e ) { // some other exception?
901 $data = false;
902 $this->handleException( $e, null, __METHOD__, array( 'src' => $path ) + $ep );
904 $contents[$path] = $data;
907 $batch = new CF_Async_Op_Batch( $cfOps );
908 $cfOps = $batch->execute();
909 foreach ( $cfOps as $path => $cfOp ) {
910 try {
911 $cfOp->getLastResponse();
912 rewind( $cfOp->_file_handle ); // start from the beginning
913 $contents[$path] = stream_get_contents( $cfOp->_file_handle );
914 } catch ( NoSuchContainerException $e ) {
915 $contents[$path] = false;
916 } catch ( NoSuchObjectException $e ) {
917 $contents[$path] = false;
918 } catch ( CloudFilesException $e ) { // some other exception?
919 $contents[$path] = false;
920 $this->handleException( $e, null, __METHOD__, array( 'src' => $path ) + $ep );
922 fclose( $cfOp->_file_handle ); // close open handle
926 return $contents;
930 * @see FileBackendStore::doDirectoryExists()
931 * @return bool|null
933 protected function doDirectoryExists( $fullCont, $dir, array $params ) {
934 try {
935 $container = $this->getContainer( $fullCont );
936 $prefix = ( $dir == '' ) ? null : "{$dir}/";
937 return ( count( $container->list_objects( 1, null, $prefix ) ) > 0 );
938 } catch ( NoSuchContainerException $e ) {
939 return false;
940 } catch ( CloudFilesException $e ) { // some other exception?
941 $this->handleException( $e, null, __METHOD__,
942 array( 'cont' => $fullCont, 'dir' => $dir ) );
945 return null; // error
949 * @see FileBackendStore::getDirectoryListInternal()
950 * @return SwiftFileBackendDirList
952 public function getDirectoryListInternal( $fullCont, $dir, array $params ) {
953 return new SwiftFileBackendDirList( $this, $fullCont, $dir, $params );
957 * @see FileBackendStore::getFileListInternal()
958 * @return SwiftFileBackendFileList
960 public function getFileListInternal( $fullCont, $dir, array $params ) {
961 return new SwiftFileBackendFileList( $this, $fullCont, $dir, $params );
965 * Do not call this function outside of SwiftFileBackendFileList
967 * @param string $fullCont Resolved container name
968 * @param string $dir Resolved storage directory with no trailing slash
969 * @param string|null $after Storage path of file to list items after
970 * @param $limit integer Max number of items to list
971 * @param array $params Includes flag for 'topOnly'
972 * @return Array List of relative paths of dirs directly under $dir
974 public function getDirListPageInternal( $fullCont, $dir, &$after, $limit, array $params ) {
975 $dirs = array();
976 if ( $after === INF ) {
977 return $dirs; // nothing more
979 wfProfileIn( __METHOD__ . '-' . $this->name );
981 try {
982 $container = $this->getContainer( $fullCont );
983 $prefix = ( $dir == '' ) ? null : "{$dir}/";
984 // Non-recursive: only list dirs right under $dir
985 if ( !empty( $params['topOnly'] ) ) {
986 $objects = $container->list_objects( $limit, $after, $prefix, null, '/' );
987 foreach ( $objects as $object ) { // files and dirs
988 if ( substr( $object, -1 ) === '/' ) {
989 $dirs[] = $object; // directories end in '/'
992 // Recursive: list all dirs under $dir and its subdirs
993 } else {
994 // Get directory from last item of prior page
995 $lastDir = $this->getParentDir( $after ); // must be first page
996 $objects = $container->list_objects( $limit, $after, $prefix );
997 foreach ( $objects as $object ) { // files
998 $objectDir = $this->getParentDir( $object ); // directory of object
999 if ( $objectDir !== false && $objectDir !== $dir ) {
1000 // Swift stores paths in UTF-8, using binary sorting.
1001 // See function "create_container_table" in common/db.py.
1002 // If a directory is not "greater" than the last one,
1003 // then it was already listed by the calling iterator.
1004 if ( strcmp( $objectDir, $lastDir ) > 0 ) {
1005 $pDir = $objectDir;
1006 do { // add dir and all its parent dirs
1007 $dirs[] = "{$pDir}/";
1008 $pDir = $this->getParentDir( $pDir );
1009 } while ( $pDir !== false // sanity
1010 && strcmp( $pDir, $lastDir ) > 0 // not done already
1011 && strlen( $pDir ) > strlen( $dir ) // within $dir
1014 $lastDir = $objectDir;
1018 if ( count( $objects ) < $limit ) {
1019 $after = INF; // avoid a second RTT
1020 } else {
1021 $after = end( $objects ); // update last item
1023 } catch ( NoSuchContainerException $e ) {
1024 } catch ( CloudFilesException $e ) { // some other exception?
1025 $this->handleException( $e, null, __METHOD__,
1026 array( 'cont' => $fullCont, 'dir' => $dir ) );
1029 wfProfileOut( __METHOD__ . '-' . $this->name );
1030 return $dirs;
1033 protected function getParentDir( $path ) {
1034 return ( strpos( $path, '/' ) !== false ) ? dirname( $path ) : false;
1038 * Do not call this function outside of SwiftFileBackendFileList
1040 * @param string $fullCont Resolved container name
1041 * @param string $dir Resolved storage directory with no trailing slash
1042 * @param string|null $after Storage path of file to list items after
1043 * @param $limit integer Max number of items to list
1044 * @param array $params Includes flag for 'topOnly'
1045 * @return Array List of relative paths of files under $dir
1047 public function getFileListPageInternal( $fullCont, $dir, &$after, $limit, array $params ) {
1048 $files = array();
1049 if ( $after === INF ) {
1050 return $files; // nothing more
1052 wfProfileIn( __METHOD__ . '-' . $this->name );
1054 try {
1055 $container = $this->getContainer( $fullCont );
1056 $prefix = ( $dir == '' ) ? null : "{$dir}/";
1057 // Non-recursive: only list files right under $dir
1058 if ( !empty( $params['topOnly'] ) ) { // files and dirs
1059 $objects = $container->list_objects( $limit, $after, $prefix, null, '/' );
1060 foreach ( $objects as $object ) {
1061 if ( substr( $object, -1 ) !== '/' ) {
1062 $files[] = $object; // directories end in '/'
1065 // Recursive: list all files under $dir and its subdirs
1066 } else { // files
1067 $objects = $container->list_objects( $limit, $after, $prefix );
1068 $files = $objects;
1070 if ( count( $objects ) < $limit ) {
1071 $after = INF; // avoid a second RTT
1072 } else {
1073 $after = end( $objects ); // update last item
1075 } catch ( NoSuchContainerException $e ) {
1076 } catch ( CloudFilesException $e ) { // some other exception?
1077 $this->handleException( $e, null, __METHOD__,
1078 array( 'cont' => $fullCont, 'dir' => $dir ) );
1081 wfProfileOut( __METHOD__ . '-' . $this->name );
1082 return $files;
1086 * @see FileBackendStore::doGetFileSha1base36()
1087 * @return bool
1089 protected function doGetFileSha1base36( array $params ) {
1090 $stat = $this->getFileStat( $params );
1091 if ( $stat ) {
1092 return $stat['sha1'];
1093 } else {
1094 return false;
1099 * @see FileBackendStore::doStreamFile()
1100 * @return Status
1102 protected function doStreamFile( array $params ) {
1103 $status = Status::newGood();
1105 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
1106 if ( $srcRel === null ) {
1107 $status->fatal( 'backend-fail-invalidpath', $params['src'] );
1110 try {
1111 $cont = $this->getContainer( $srcCont );
1112 } catch ( NoSuchContainerException $e ) {
1113 $status->fatal( 'backend-fail-stream', $params['src'] );
1114 return $status;
1115 } catch ( CloudFilesException $e ) { // some other exception?
1116 $this->handleException( $e, $status, __METHOD__, $params );
1117 return $status;
1120 try {
1121 $output = fopen( 'php://output', 'wb' );
1122 $obj = new CF_Object( $cont, $srcRel, false, false ); // skip HEAD
1123 $obj->stream( $output, $this->headersFromParams( $params ) );
1124 } catch ( NoSuchObjectException $e ) {
1125 $status->fatal( 'backend-fail-stream', $params['src'] );
1126 } catch ( CloudFilesException $e ) { // some other exception?
1127 $this->handleException( $e, $status, __METHOD__, $params );
1130 return $status;
1134 * @see FileBackendStore::doGetLocalCopyMulti()
1135 * @return null|TempFSFile
1137 protected function doGetLocalCopyMulti( array $params ) {
1138 $tmpFiles = array();
1140 $ep = array_diff_key( $params, array( 'srcs' => 1 ) ); // for error logging
1141 // Blindly create tmp files and stream to them, catching any exception if the file does
1142 // not exist. Doing a stat here is useless causes infinite loops in addMissingMetadata().
1143 foreach ( array_chunk( $params['srcs'], $params['concurrency'] ) as $pathBatch ) {
1144 $cfOps = array(); // (path => CF_Async_Op)
1146 foreach ( $pathBatch as $path ) { // each path in this concurrent batch
1147 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $path );
1148 if ( $srcRel === null ) {
1149 $tmpFiles[$path] = null;
1150 continue;
1152 $tmpFile = null;
1153 try {
1154 $sContObj = $this->getContainer( $srcCont );
1155 $obj = new CF_Object( $sContObj, $srcRel, false, false ); // skip HEAD
1156 // Get source file extension
1157 $ext = FileBackend::extensionFromPath( $path );
1158 // Create a new temporary file...
1159 $tmpFile = TempFSFile::factory( 'localcopy_', $ext );
1160 if ( $tmpFile ) {
1161 $handle = fopen( $tmpFile->getPath(), 'wb' );
1162 if ( $handle ) {
1163 $headers = $this->headersFromParams( $params );
1164 if ( count( $pathBatch ) > 1 ) {
1165 $cfOps[$path] = $obj->stream_async( $handle, $headers );
1166 $cfOps[$path]->_file_handle = $handle; // close this later
1167 } else {
1168 $obj->stream( $handle, $headers );
1169 fclose( $handle );
1171 } else {
1172 $tmpFile = null;
1175 } catch ( NoSuchContainerException $e ) {
1176 $tmpFile = null;
1177 } catch ( NoSuchObjectException $e ) {
1178 $tmpFile = null;
1179 } catch ( CloudFilesException $e ) { // some other exception?
1180 $tmpFile = null;
1181 $this->handleException( $e, null, __METHOD__, array( 'src' => $path ) + $ep );
1183 $tmpFiles[$path] = $tmpFile;
1186 $batch = new CF_Async_Op_Batch( $cfOps );
1187 $cfOps = $batch->execute();
1188 foreach ( $cfOps as $path => $cfOp ) {
1189 try {
1190 $cfOp->getLastResponse();
1191 } catch ( NoSuchContainerException $e ) {
1192 $tmpFiles[$path] = null;
1193 } catch ( NoSuchObjectException $e ) {
1194 $tmpFiles[$path] = null;
1195 } catch ( CloudFilesException $e ) { // some other exception?
1196 $tmpFiles[$path] = null;
1197 $this->handleException( $e, null, __METHOD__, array( 'src' => $path ) + $ep );
1199 fclose( $cfOp->_file_handle ); // close open handle
1203 return $tmpFiles;
1207 * @see FileBackendStore::getFileHttpUrl()
1208 * @return string|null
1210 public function getFileHttpUrl( array $params ) {
1211 if ( $this->swiftTempUrlKey != '' ||
1212 ( $this->rgwS3AccessKey != '' && $this->rgwS3SecretKey != '' ) )
1214 list( $srcCont, $srcRel ) = $this->resolveStoragePathReal( $params['src'] );
1215 if ( $srcRel === null ) {
1216 return null; // invalid path
1218 try {
1219 $ttl = isset( $params['ttl'] ) ? $params['ttl'] : 86400;
1220 $sContObj = $this->getContainer( $srcCont );
1221 $obj = new CF_Object( $sContObj, $srcRel, false, false ); // skip HEAD
1222 if ( $this->swiftTempUrlKey != '' ) {
1223 return $obj->get_temp_url( $this->swiftTempUrlKey, $ttl, "GET" );
1224 } else { // give S3 API URL for rgw
1225 $expires = time() + $ttl;
1226 // Path for signature starts with the bucket
1227 $spath = '/' . rawurlencode( $srcCont ) . '/' .
1228 str_replace( '%2F', '/', rawurlencode( $srcRel ) );
1229 // Calculate the hash
1230 $signature = base64_encode( hash_hmac(
1231 'sha1',
1232 "GET\n\n\n{$expires}\n{$spath}",
1233 $this->rgwS3SecretKey,
1234 true // raw
1235 ) );
1236 // See http://s3.amazonaws.com/doc/s3-developer-guide/RESTAuthentication.html.
1237 // Note: adding a newline for empty CanonicalizedAmzHeaders does not work.
1238 return wfAppendQuery(
1239 str_replace( '/swift/v1', '', // S3 API is the rgw default
1240 $sContObj->cfs_http->getStorageUrl() . $spath ),
1241 array(
1242 'Signature' => $signature,
1243 'Expires' => $expires,
1244 'AWSAccessKeyId' => $this->rgwS3AccessKey )
1247 } catch ( NoSuchContainerException $e ) {
1248 } catch ( CloudFilesException $e ) { // some other exception?
1249 $this->handleException( $e, null, __METHOD__, $params );
1252 return null;
1256 * @see FileBackendStore::directoriesAreVirtual()
1257 * @return bool
1259 protected function directoriesAreVirtual() {
1260 return true;
1264 * Get headers to send to Swift when reading a file based
1265 * on a FileBackend params array, e.g. that of getLocalCopy().
1266 * $params is currently only checked for a 'latest' flag.
1268 * @param array $params
1269 * @return Array
1271 protected function headersFromParams( array $params ) {
1272 $hdrs = array();
1273 if ( !empty( $params['latest'] ) ) {
1274 $hdrs[] = 'X-Newest: true';
1276 return $hdrs;
1280 * @see FileBackendStore::doExecuteOpHandlesInternal()
1281 * @return Array List of corresponding Status objects
1283 protected function doExecuteOpHandlesInternal( array $fileOpHandles ) {
1284 $statuses = array();
1286 $cfOps = array(); // list of CF_Async_Op objects
1287 foreach ( $fileOpHandles as $index => $fileOpHandle ) {
1288 $cfOps[$index] = $fileOpHandle->cfOp;
1290 $batch = new CF_Async_Op_Batch( $cfOps );
1292 $cfOps = $batch->execute();
1293 foreach ( $cfOps as $index => $cfOp ) {
1294 $status = Status::newGood();
1295 $function = '_getResponse' . $fileOpHandles[$index]->call;
1296 try { // catch exceptions; update status
1297 $this->$function( $cfOp, $status, $fileOpHandles[$index]->params );
1298 $this->purgeCDNCache( $fileOpHandles[$index]->affectedObjects );
1299 } catch ( CloudFilesException $e ) { // some other exception?
1300 $this->handleException( $e, $status,
1301 __CLASS__ . ":$function", $fileOpHandles[$index]->params );
1303 $statuses[$index] = $status;
1306 return $statuses;
1310 * Set read/write permissions for a Swift container.
1312 * $readGrps is a list of the possible criteria for a request to have
1313 * access to read a container. Each item is one of the following formats:
1314 * - account:user : Grants access if the request is by the given user
1315 * - ".r:<regex>" : Grants access if the request is from a referrer host that
1316 * matches the expression and the request is not for a listing.
1317 * Setting this to '*' effectively makes a container public.
1318 * -".rlistings:<regex>" : Grants access if the request is from a referrer host that
1319 * matches the expression and the request is for a listing.
1321 * $writeGrps is a list of the possible criteria for a request to have
1322 * access to write to a container. Each item is of the following format:
1323 * - account:user : Grants access if the request is by the given user
1325 * @see http://swift.openstack.org/misc.html#acls
1327 * In general, we don't allow listings to end-users. It's not useful, isn't well-defined
1328 * (lists are truncated to 10000 item with no way to page), and is just a performance risk.
1330 * @param $contObj CF_Container Swift container
1331 * @param array $readGrps List of read access routes
1332 * @param array $writeGrps List of write access routes
1333 * @return Status
1335 protected function setContainerAccess(
1336 CF_Container $contObj, array $readGrps, array $writeGrps
1338 $creds = $contObj->cfs_auth->export_credentials();
1340 $url = $creds['storage_url'] . '/' . rawurlencode( $contObj->name );
1342 // Note: 10 second timeout consistent with php-cloudfiles
1343 $req = MWHttpRequest::factory( $url, array( 'method' => 'POST', 'timeout' => 10 ) );
1344 $req->setHeader( 'X-Auth-Token', $creds['auth_token'] );
1345 $req->setHeader( 'X-Container-Read', implode( ',', $readGrps ) );
1346 $req->setHeader( 'X-Container-Write', implode( ',', $writeGrps ) );
1348 return $req->execute(); // should return 204
1352 * Purge the CDN cache of affected objects if CDN caching is enabled.
1353 * This is for Rackspace/Akamai CDNs.
1355 * @param array $objects List of CF_Object items
1356 * @return void
1358 public function purgeCDNCache( array $objects ) {
1359 if ( $this->swiftUseCDN && $this->swiftCDNPurgable ) {
1360 foreach ( $objects as $object ) {
1361 try {
1362 $object->purge_from_cdn();
1363 } catch ( CDNNotEnabledException $e ) {
1364 // CDN not enabled; nothing to see here
1365 } catch ( CloudFilesException $e ) {
1366 $this->handleException( $e, null, __METHOD__,
1367 array( 'cont' => $object->container->name, 'obj' => $object->name ) );
1374 * Get an authenticated connection handle to the Swift proxy
1376 * @throws CloudFilesException
1377 * @throws CloudFilesException|Exception
1378 * @return CF_Connection|bool False on failure
1380 protected function getConnection() {
1381 if ( $this->connException instanceof CloudFilesException ) {
1382 if ( ( time() - $this->connErrorTime ) < 60 ) {
1383 throw $this->connException; // failed last attempt; don't bother
1384 } else { // actually retry this time
1385 $this->connException = null;
1386 $this->connErrorTime = 0;
1389 // Session keys expire after a while, so we renew them periodically
1390 $reAuth = ( ( time() - $this->sessionStarted ) > $this->authTTL );
1391 // Authenticate with proxy and get a session key...
1392 if ( !$this->conn || $reAuth ) {
1393 $this->sessionStarted = 0;
1394 $this->connContainerCache->clear();
1395 $cacheKey = $this->getCredsCacheKey( $this->auth->username );
1396 $creds = $this->srvCache->get( $cacheKey ); // credentials
1397 if ( is_array( $creds ) ) { // cache hit
1398 $this->auth->load_cached_credentials(
1399 $creds['auth_token'], $creds['storage_url'], $creds['cdnm_url'] );
1400 $this->sessionStarted = time() - ceil( $this->authTTL / 2 ); // skew for worst case
1401 } else { // cache miss
1402 try {
1403 $this->auth->authenticate();
1404 $creds = $this->auth->export_credentials();
1405 $this->srvCache->add( $cacheKey, $creds, ceil( $this->authTTL / 2 ) ); // cache
1406 $this->sessionStarted = time();
1407 } catch ( CloudFilesException $e ) {
1408 $this->connException = $e; // don't keep re-trying
1409 $this->connErrorTime = time();
1410 throw $e; // throw it back
1413 if ( $this->conn ) { // re-authorizing?
1414 $this->conn->close(); // close active cURL handles in CF_Http object
1416 $this->conn = new CF_Connection( $this->auth );
1418 return $this->conn;
1422 * Close the connection to the Swift proxy
1424 * @return void
1426 protected function closeConnection() {
1427 if ( $this->conn ) {
1428 $this->conn->close(); // close active cURL handles in CF_Http object
1429 $this->conn = null;
1430 $this->sessionStarted = 0;
1431 $this->connContainerCache->clear();
1436 * Get the cache key for a container
1438 * @param $username string
1439 * @return string
1441 private function getCredsCacheKey( $username ) {
1442 return wfMemcKey( 'backend', $this->getName(), 'usercreds', $username );
1446 * Get a Swift container object, possibly from process cache.
1447 * Use $reCache if the file count or byte count is needed.
1449 * @param string $container Container name
1450 * @param bool $bypassCache Bypass all caches and load from Swift
1451 * @return CF_Container
1452 * @throws CloudFilesException
1454 protected function getContainer( $container, $bypassCache = false ) {
1455 $conn = $this->getConnection(); // Swift proxy connection
1456 if ( $bypassCache ) { // purge cache
1457 $this->connContainerCache->clear( $container );
1458 } elseif ( !$this->connContainerCache->has( $container, 'obj' ) ) {
1459 $this->primeContainerCache( array( $container ) ); // check persistent cache
1461 if ( !$this->connContainerCache->has( $container, 'obj' ) ) {
1462 $contObj = $conn->get_container( $container );
1463 // NoSuchContainerException not thrown: container must exist
1464 $this->connContainerCache->set( $container, 'obj', $contObj ); // cache it
1465 if ( !$bypassCache ) {
1466 $this->setContainerCache( $container, // update persistent cache
1467 array( 'bytes' => $contObj->bytes_used, 'count' => $contObj->object_count )
1471 return $this->connContainerCache->get( $container, 'obj' );
1475 * Create a Swift container
1477 * @param string $container Container name
1478 * @return CF_Container
1479 * @throws CloudFilesException
1481 protected function createContainer( $container ) {
1482 $conn = $this->getConnection(); // Swift proxy connection
1483 $contObj = $conn->create_container( $container );
1484 $this->connContainerCache->set( $container, 'obj', $contObj ); // cache
1485 return $contObj;
1489 * Delete a Swift container
1491 * @param string $container Container name
1492 * @return void
1493 * @throws CloudFilesException
1495 protected function deleteContainer( $container ) {
1496 $conn = $this->getConnection(); // Swift proxy connection
1497 $this->connContainerCache->clear( $container ); // purge
1498 $conn->delete_container( $container );
1502 * @see FileBackendStore::doPrimeContainerCache()
1503 * @return void
1505 protected function doPrimeContainerCache( array $containerInfo ) {
1506 try {
1507 $conn = $this->getConnection(); // Swift proxy connection
1508 foreach ( $containerInfo as $container => $info ) {
1509 $contObj = new CF_Container( $conn->cfs_auth, $conn->cfs_http,
1510 $container, $info['count'], $info['bytes'] );
1511 $this->connContainerCache->set( $container, 'obj', $contObj );
1513 } catch ( CloudFilesException $e ) { // some other exception?
1514 $this->handleException( $e, null, __METHOD__, array() );
1519 * Log an unexpected exception for this backend.
1520 * This also sets the Status object to have a fatal error.
1522 * @param $e Exception
1523 * @param $status Status|null
1524 * @param $func string
1525 * @param array $params
1526 * @return void
1528 protected function handleException( Exception $e, $status, $func, array $params ) {
1529 if ( $status instanceof Status ) {
1530 if ( $e instanceof AuthenticationException ) {
1531 $status->fatal( 'backend-fail-connect', $this->name );
1532 } else {
1533 $status->fatal( 'backend-fail-internal', $this->name );
1536 if ( $e->getMessage() ) {
1537 trigger_error( "$func: " . $e->getMessage(), E_USER_WARNING );
1539 if ( $e instanceof InvalidResponseException ) { // possibly a stale token
1540 $this->srvCache->delete( $this->getCredsCacheKey( $this->auth->username ) );
1541 $this->closeConnection(); // force a re-connect and re-auth next time
1543 wfDebugLog( 'SwiftBackend',
1544 get_class( $e ) . " in '{$func}' (given '" . FormatJson::encode( $params ) . "')" .
1545 ( $e->getMessage() ? ": {$e->getMessage()}" : "" )
1551 * @see FileBackendStoreOpHandle
1553 class SwiftFileOpHandle extends FileBackendStoreOpHandle {
1554 /** @var CF_Async_Op */
1555 public $cfOp;
1556 /** @var Array */
1557 public $affectedObjects = array();
1559 public function __construct( $backend, array $params, $call, CF_Async_Op $cfOp ) {
1560 $this->backend = $backend;
1561 $this->params = $params;
1562 $this->call = $call;
1563 $this->cfOp = $cfOp;
1568 * SwiftFileBackend helper class to page through listings.
1569 * Swift also has a listing limit of 10,000 objects for sanity.
1570 * Do not use this class from places outside SwiftFileBackend.
1572 * @ingroup FileBackend
1574 abstract class SwiftFileBackendList implements Iterator {
1575 /** @var Array */
1576 protected $bufferIter = array();
1577 protected $bufferAfter = null; // string; list items *after* this path
1578 protected $pos = 0; // integer
1579 /** @var Array */
1580 protected $params = array();
1582 /** @var SwiftFileBackend */
1583 protected $backend;
1584 protected $container; // string; container name
1585 protected $dir; // string; storage directory
1586 protected $suffixStart; // integer
1588 const PAGE_SIZE = 9000; // file listing buffer size
1591 * @param $backend SwiftFileBackend
1592 * @param string $fullCont Resolved container name
1593 * @param string $dir Resolved directory relative to container
1594 * @param array $params
1596 public function __construct( SwiftFileBackend $backend, $fullCont, $dir, array $params ) {
1597 $this->backend = $backend;
1598 $this->container = $fullCont;
1599 $this->dir = $dir;
1600 if ( substr( $this->dir, -1 ) === '/' ) {
1601 $this->dir = substr( $this->dir, 0, -1 ); // remove trailing slash
1603 if ( $this->dir == '' ) { // whole container
1604 $this->suffixStart = 0;
1605 } else { // dir within container
1606 $this->suffixStart = strlen( $this->dir ) + 1; // size of "path/to/dir/"
1608 $this->params = $params;
1612 * @see Iterator::key()
1613 * @return integer
1615 public function key() {
1616 return $this->pos;
1620 * @see Iterator::next()
1621 * @return void
1623 public function next() {
1624 // Advance to the next file in the page
1625 next( $this->bufferIter );
1626 ++$this->pos;
1627 // Check if there are no files left in this page and
1628 // advance to the next page if this page was not empty.
1629 if ( !$this->valid() && count( $this->bufferIter ) ) {
1630 $this->bufferIter = $this->pageFromList(
1631 $this->container, $this->dir, $this->bufferAfter, self::PAGE_SIZE, $this->params
1632 ); // updates $this->bufferAfter
1637 * @see Iterator::rewind()
1638 * @return void
1640 public function rewind() {
1641 $this->pos = 0;
1642 $this->bufferAfter = null;
1643 $this->bufferIter = $this->pageFromList(
1644 $this->container, $this->dir, $this->bufferAfter, self::PAGE_SIZE, $this->params
1645 ); // updates $this->bufferAfter
1649 * @see Iterator::valid()
1650 * @return bool
1652 public function valid() {
1653 if ( $this->bufferIter === null ) {
1654 return false; // some failure?
1655 } else {
1656 return ( current( $this->bufferIter ) !== false ); // no paths can have this value
1661 * Get the given list portion (page)
1663 * @param string $container Resolved container name
1664 * @param string $dir Resolved path relative to container
1665 * @param $after string|null
1666 * @param $limit integer
1667 * @param array $params
1668 * @return Traversable|Array|null Returns null on failure
1670 abstract protected function pageFromList( $container, $dir, &$after, $limit, array $params );
1674 * Iterator for listing directories
1676 class SwiftFileBackendDirList extends SwiftFileBackendList {
1678 * @see Iterator::current()
1679 * @return string|bool String (relative path) or false
1681 public function current() {
1682 return substr( current( $this->bufferIter ), $this->suffixStart, -1 );
1686 * @see SwiftFileBackendList::pageFromList()
1687 * @return Array|null
1689 protected function pageFromList( $container, $dir, &$after, $limit, array $params ) {
1690 return $this->backend->getDirListPageInternal( $container, $dir, $after, $limit, $params );
1695 * Iterator for listing regular files
1697 class SwiftFileBackendFileList extends SwiftFileBackendList {
1699 * @see Iterator::current()
1700 * @return string|bool String (relative path) or false
1702 public function current() {
1703 return substr( current( $this->bufferIter ), $this->suffixStart );
1707 * @see SwiftFileBackendList::pageFromList()
1708 * @return Array|null
1710 protected function pageFromList( $container, $dir, &$after, $limit, array $params ) {
1711 return $this->backend->getFileListPageInternal( $container, $dir, $after, $limit, $params );