Import: Handle uploads with sha1 starting with 0 properly
[mediawiki.git] / tests / phpunit / includes / filebackend / FileBackendTest.php
blob0763a2e02317890a3cfa9284bdc65204af968709
1 <?php
3 /**
4 * @group FileRepo
5 * @group FileBackend
6 * @group medium
7 */
8 class FileBackendTest extends MediaWikiTestCase {
10 /** @var FileBackend */
11 private $backend;
12 /** @var FileBackendMultiWrite */
13 private $multiBackend;
14 /** @var FSFileBackend */
15 public $singleBackend;
16 private static $backendToUse;
18 protected function setUp() {
19 global $wgFileBackends;
20 parent::setUp();
21 $tmpDir = $this->getNewTempDirectory();
22 if ( $this->getCliArg( 'use-filebackend' ) ) {
23 if ( self::$backendToUse ) {
24 $this->singleBackend = self::$backendToUse;
25 } else {
26 $name = $this->getCliArg( 'use-filebackend' );
27 $useConfig = array();
28 foreach ( $wgFileBackends as $conf ) {
29 if ( $conf['name'] == $name ) {
30 $useConfig = $conf;
31 break;
34 $useConfig['name'] = 'localtesting'; // swap name
35 $useConfig['shardViaHashLevels'] = array( // test sharding
36 'unittest-cont1' => array( 'levels' => 1, 'base' => 16, 'repeat' => 1 )
38 if ( isset( $useConfig['fileJournal'] ) ) {
39 $useConfig['fileJournal'] = FileJournal::factory( $useConfig['fileJournal'], $name );
41 $useConfig['lockManager'] = LockManagerGroup::singleton()->get( $useConfig['lockManager'] );
42 $class = $useConfig['class'];
43 self::$backendToUse = new $class( $useConfig );
44 $this->singleBackend = self::$backendToUse;
46 } else {
47 $this->singleBackend = new FSFileBackend( array(
48 'name' => 'localtesting',
49 'lockManager' => LockManagerGroup::singleton()->get( 'fsLockManager' ),
50 'wikiId' => wfWikiID(),
51 'containerPaths' => array(
52 'unittest-cont1' => "{$tmpDir}/localtesting-cont1",
53 'unittest-cont2' => "{$tmpDir}/localtesting-cont2" )
54 ) );
56 $this->multiBackend = new FileBackendMultiWrite( array(
57 'name' => 'localtesting',
58 'lockManager' => LockManagerGroup::singleton()->get( 'fsLockManager' ),
59 'parallelize' => 'implicit',
60 'wikiId' => wfWikiId() . wfRandomString(),
61 'backends' => array(
62 array(
63 'name' => 'localmultitesting1',
64 'class' => 'FSFileBackend',
65 'containerPaths' => array(
66 'unittest-cont1' => "{$tmpDir}/localtestingmulti1-cont1",
67 'unittest-cont2' => "{$tmpDir}/localtestingmulti1-cont2" ),
68 'isMultiMaster' => false
70 array(
71 'name' => 'localmultitesting2',
72 'class' => 'FSFileBackend',
73 'containerPaths' => array(
74 'unittest-cont1' => "{$tmpDir}/localtestingmulti2-cont1",
75 'unittest-cont2' => "{$tmpDir}/localtestingmulti2-cont2" ),
76 'isMultiMaster' => true
79 ) );
82 private static function baseStorePath() {
83 return 'mwstore://localtesting';
86 private function backendClass() {
87 return get_class( $this->backend );
90 /**
91 * @dataProvider provider_testIsStoragePath
92 * @covers FileBackend::isStoragePath
94 public function testIsStoragePath( $path, $isStorePath ) {
95 $this->assertEquals( $isStorePath, FileBackend::isStoragePath( $path ),
96 "FileBackend::isStoragePath on path '$path'" );
99 public static function provider_testIsStoragePath() {
100 return array(
101 array( 'mwstore://', true ),
102 array( 'mwstore://backend', true ),
103 array( 'mwstore://backend/container', true ),
104 array( 'mwstore://backend/container/', true ),
105 array( 'mwstore://backend/container/path', true ),
106 array( 'mwstore://backend//container/', true ),
107 array( 'mwstore://backend//container//', true ),
108 array( 'mwstore://backend//container//path', true ),
109 array( 'mwstore:///', true ),
110 array( 'mwstore:/', false ),
111 array( 'mwstore:', false ),
116 * @dataProvider provider_testSplitStoragePath
117 * @covers FileBackend::splitStoragePath
119 public function testSplitStoragePath( $path, $res ) {
120 $this->assertEquals( $res, FileBackend::splitStoragePath( $path ),
121 "FileBackend::splitStoragePath on path '$path'" );
124 public static function provider_testSplitStoragePath() {
125 return array(
126 array( 'mwstore://backend/container', array( 'backend', 'container', '' ) ),
127 array( 'mwstore://backend/container/', array( 'backend', 'container', '' ) ),
128 array( 'mwstore://backend/container/path', array( 'backend', 'container', 'path' ) ),
129 array( 'mwstore://backend/container//path', array( 'backend', 'container', '/path' ) ),
130 array( 'mwstore://backend//container/path', array( null, null, null ) ),
131 array( 'mwstore://backend//container//path', array( null, null, null ) ),
132 array( 'mwstore://', array( null, null, null ) ),
133 array( 'mwstore://backend', array( null, null, null ) ),
134 array( 'mwstore:///', array( null, null, null ) ),
135 array( 'mwstore:/', array( null, null, null ) ),
136 array( 'mwstore:', array( null, null, null ) )
141 * @dataProvider provider_normalizeStoragePath
142 * @covers FileBackend::normalizeStoragePath
144 public function testNormalizeStoragePath( $path, $res ) {
145 $this->assertEquals( $res, FileBackend::normalizeStoragePath( $path ),
146 "FileBackend::normalizeStoragePath on path '$path'" );
149 public static function provider_normalizeStoragePath() {
150 return array(
151 array( 'mwstore://backend/container', 'mwstore://backend/container' ),
152 array( 'mwstore://backend/container/', 'mwstore://backend/container' ),
153 array( 'mwstore://backend/container/path', 'mwstore://backend/container/path' ),
154 array( 'mwstore://backend/container//path', 'mwstore://backend/container/path' ),
155 array( 'mwstore://backend/container///path', 'mwstore://backend/container/path' ),
156 array(
157 'mwstore://backend/container///path//to///obj',
158 'mwstore://backend/container/path/to/obj'
160 array( 'mwstore://', null ),
161 array( 'mwstore://backend', null ),
162 array( 'mwstore://backend//container/path', null ),
163 array( 'mwstore://backend//container//path', null ),
164 array( 'mwstore:///', null ),
165 array( 'mwstore:/', null ),
166 array( 'mwstore:', null ),
171 * @dataProvider provider_testParentStoragePath
172 * @covers FileBackend::parentStoragePath
174 public function testParentStoragePath( $path, $res ) {
175 $this->assertEquals( $res, FileBackend::parentStoragePath( $path ),
176 "FileBackend::parentStoragePath on path '$path'" );
179 public static function provider_testParentStoragePath() {
180 return array(
181 array( 'mwstore://backend/container/path/to/obj', 'mwstore://backend/container/path/to' ),
182 array( 'mwstore://backend/container/path/to', 'mwstore://backend/container/path' ),
183 array( 'mwstore://backend/container/path', 'mwstore://backend/container' ),
184 array( 'mwstore://backend/container', null ),
185 array( 'mwstore://backend/container/path/to/obj/', 'mwstore://backend/container/path/to' ),
186 array( 'mwstore://backend/container/path/to/', 'mwstore://backend/container/path' ),
187 array( 'mwstore://backend/container/path/', 'mwstore://backend/container' ),
188 array( 'mwstore://backend/container/', null ),
193 * @dataProvider provider_testExtensionFromPath
194 * @covers FileBackend::extensionFromPath
196 public function testExtensionFromPath( $path, $res ) {
197 $this->assertEquals( $res, FileBackend::extensionFromPath( $path ),
198 "FileBackend::extensionFromPath on path '$path'" );
201 public static function provider_testExtensionFromPath() {
202 return array(
203 array( 'mwstore://backend/container/path.txt', 'txt' ),
204 array( 'mwstore://backend/container/path.svg.png', 'png' ),
205 array( 'mwstore://backend/container/path', '' ),
206 array( 'mwstore://backend/container/path.', '' ),
211 * @dataProvider provider_testStore
213 public function testStore( $op ) {
214 $this->addTmpFiles( $op['src'] );
216 $this->backend = $this->singleBackend;
217 $this->tearDownFiles();
218 $this->doTestStore( $op );
219 $this->tearDownFiles();
221 $this->backend = $this->multiBackend;
222 $this->tearDownFiles();
223 $this->doTestStore( $op );
224 $this->tearDownFiles();
228 * @covers FileBackend::doOperation
230 private function doTestStore( $op ) {
231 $backendName = $this->backendClass();
233 $source = $op['src'];
234 $dest = $op['dst'];
235 $this->prepare( array( 'dir' => dirname( $dest ) ) );
237 file_put_contents( $source, "Unit test file" );
239 if ( isset( $op['overwrite'] ) || isset( $op['overwriteSame'] ) ) {
240 $this->backend->store( $op );
243 $status = $this->backend->doOperation( $op );
245 $this->assertGoodStatus( $status,
246 "Store from $source to $dest succeeded without warnings ($backendName)." );
247 $this->assertEquals( true, $status->isOK(),
248 "Store from $source to $dest succeeded ($backendName)." );
249 $this->assertEquals( array( 0 => true ), $status->success,
250 "Store from $source to $dest has proper 'success' field in Status ($backendName)." );
251 $this->assertEquals( true, file_exists( $source ),
252 "Source file $source still exists ($backendName)." );
253 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $dest ) ),
254 "Destination file $dest exists ($backendName)." );
256 $this->assertEquals( filesize( $source ),
257 $this->backend->getFileSize( array( 'src' => $dest ) ),
258 "Destination file $dest has correct size ($backendName)." );
260 $props1 = FSFile::getPropsFromPath( $source );
261 $props2 = $this->backend->getFileProps( array( 'src' => $dest ) );
262 $this->assertEquals( $props1, $props2,
263 "Source and destination have the same props ($backendName)." );
265 $this->assertBackendPathsConsistent( array( $dest ) );
268 public static function provider_testStore() {
269 $cases = array();
271 $tmpName = TempFSFile::factory( "unittests_", 'txt' )->getPath();
272 $toPath = self::baseStorePath() . '/unittest-cont1/e/fun/obj1.txt';
273 $op = array( 'op' => 'store', 'src' => $tmpName, 'dst' => $toPath );
274 $cases[] = array( $op );
276 $op2 = $op;
277 $op2['overwrite'] = true;
278 $cases[] = array( $op2 );
280 $op3 = $op;
281 $op3['overwriteSame'] = true;
282 $cases[] = array( $op3 );
284 return $cases;
288 * @dataProvider provider_testCopy
289 * @covers FileBackend::doOperation
291 public function testCopy( $op ) {
292 $this->backend = $this->singleBackend;
293 $this->tearDownFiles();
294 $this->doTestCopy( $op );
295 $this->tearDownFiles();
297 $this->backend = $this->multiBackend;
298 $this->tearDownFiles();
299 $this->doTestCopy( $op );
300 $this->tearDownFiles();
303 private function doTestCopy( $op ) {
304 $backendName = $this->backendClass();
306 $source = $op['src'];
307 $dest = $op['dst'];
308 $this->prepare( array( 'dir' => dirname( $source ) ) );
309 $this->prepare( array( 'dir' => dirname( $dest ) ) );
311 if ( isset( $op['ignoreMissingSource'] ) ) {
312 $status = $this->backend->doOperation( $op );
313 $this->assertGoodStatus( $status,
314 "Move from $source to $dest succeeded without warnings ($backendName)." );
315 $this->assertEquals( array( 0 => true ), $status->success,
316 "Move from $source to $dest has proper 'success' field in Status ($backendName)." );
317 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $source ) ),
318 "Source file $source does not exist ($backendName)." );
319 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $dest ) ),
320 "Destination file $dest does not exist ($backendName)." );
322 return; // done
325 $status = $this->backend->doOperation(
326 array( 'op' => 'create', 'content' => 'blahblah', 'dst' => $source ) );
327 $this->assertGoodStatus( $status,
328 "Creation of file at $source succeeded ($backendName)." );
330 if ( isset( $op['overwrite'] ) || isset( $op['overwriteSame'] ) ) {
331 $this->backend->copy( $op );
334 $status = $this->backend->doOperation( $op );
336 $this->assertGoodStatus( $status,
337 "Copy from $source to $dest succeeded without warnings ($backendName)." );
338 $this->assertEquals( true, $status->isOK(),
339 "Copy from $source to $dest succeeded ($backendName)." );
340 $this->assertEquals( array( 0 => true ), $status->success,
341 "Copy from $source to $dest has proper 'success' field in Status ($backendName)." );
342 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $source ) ),
343 "Source file $source still exists ($backendName)." );
344 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $dest ) ),
345 "Destination file $dest exists after copy ($backendName)." );
347 $this->assertEquals(
348 $this->backend->getFileSize( array( 'src' => $source ) ),
349 $this->backend->getFileSize( array( 'src' => $dest ) ),
350 "Destination file $dest has correct size ($backendName)." );
352 $props1 = $this->backend->getFileProps( array( 'src' => $source ) );
353 $props2 = $this->backend->getFileProps( array( 'src' => $dest ) );
354 $this->assertEquals( $props1, $props2,
355 "Source and destination have the same props ($backendName)." );
357 $this->assertBackendPathsConsistent( array( $source, $dest ) );
360 public static function provider_testCopy() {
361 $cases = array();
363 $source = self::baseStorePath() . '/unittest-cont1/e/file.txt';
364 $dest = self::baseStorePath() . '/unittest-cont2/a/fileMoved.txt';
366 $op = array( 'op' => 'copy', 'src' => $source, 'dst' => $dest );
367 $cases[] = array(
368 $op, // operation
369 $source, // source
370 $dest, // dest
373 $op2 = $op;
374 $op2['overwrite'] = true;
375 $cases[] = array(
376 $op2, // operation
377 $source, // source
378 $dest, // dest
381 $op2 = $op;
382 $op2['overwriteSame'] = true;
383 $cases[] = array(
384 $op2, // operation
385 $source, // source
386 $dest, // dest
389 $op2 = $op;
390 $op2['ignoreMissingSource'] = true;
391 $cases[] = array(
392 $op2, // operation
393 $source, // source
394 $dest, // dest
397 $op2 = $op;
398 $op2['ignoreMissingSource'] = true;
399 $cases[] = array(
400 $op2, // operation
401 self::baseStorePath() . '/unittest-cont-bad/e/file.txt', // source
402 $dest, // dest
405 return $cases;
409 * @dataProvider provider_testMove
410 * @covers FileBackend::doOperation
412 public function testMove( $op ) {
413 $this->backend = $this->singleBackend;
414 $this->tearDownFiles();
415 $this->doTestMove( $op );
416 $this->tearDownFiles();
418 $this->backend = $this->multiBackend;
419 $this->tearDownFiles();
420 $this->doTestMove( $op );
421 $this->tearDownFiles();
424 private function doTestMove( $op ) {
425 $backendName = $this->backendClass();
427 $source = $op['src'];
428 $dest = $op['dst'];
429 $this->prepare( array( 'dir' => dirname( $source ) ) );
430 $this->prepare( array( 'dir' => dirname( $dest ) ) );
432 if ( isset( $op['ignoreMissingSource'] ) ) {
433 $status = $this->backend->doOperation( $op );
434 $this->assertGoodStatus( $status,
435 "Move from $source to $dest succeeded without warnings ($backendName)." );
436 $this->assertEquals( array( 0 => true ), $status->success,
437 "Move from $source to $dest has proper 'success' field in Status ($backendName)." );
438 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $source ) ),
439 "Source file $source does not exist ($backendName)." );
440 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $dest ) ),
441 "Destination file $dest does not exist ($backendName)." );
443 return; // done
446 $status = $this->backend->doOperation(
447 array( 'op' => 'create', 'content' => 'blahblah', 'dst' => $source ) );
448 $this->assertGoodStatus( $status,
449 "Creation of file at $source succeeded ($backendName)." );
451 if ( isset( $op['overwrite'] ) || isset( $op['overwriteSame'] ) ) {
452 $this->backend->copy( $op );
455 $status = $this->backend->doOperation( $op );
456 $this->assertGoodStatus( $status,
457 "Move from $source to $dest succeeded without warnings ($backendName)." );
458 $this->assertEquals( true, $status->isOK(),
459 "Move from $source to $dest succeeded ($backendName)." );
460 $this->assertEquals( array( 0 => true ), $status->success,
461 "Move from $source to $dest has proper 'success' field in Status ($backendName)." );
462 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $source ) ),
463 "Source file $source does not still exists ($backendName)." );
464 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $dest ) ),
465 "Destination file $dest exists after move ($backendName)." );
467 $this->assertNotEquals(
468 $this->backend->getFileSize( array( 'src' => $source ) ),
469 $this->backend->getFileSize( array( 'src' => $dest ) ),
470 "Destination file $dest has correct size ($backendName)." );
472 $props1 = $this->backend->getFileProps( array( 'src' => $source ) );
473 $props2 = $this->backend->getFileProps( array( 'src' => $dest ) );
474 $this->assertEquals( false, $props1['fileExists'],
475 "Source file does not exist accourding to props ($backendName)." );
476 $this->assertEquals( true, $props2['fileExists'],
477 "Destination file exists accourding to props ($backendName)." );
479 $this->assertBackendPathsConsistent( array( $source, $dest ) );
482 public static function provider_testMove() {
483 $cases = array();
485 $source = self::baseStorePath() . '/unittest-cont1/e/file.txt';
486 $dest = self::baseStorePath() . '/unittest-cont2/a/fileMoved.txt';
488 $op = array( 'op' => 'move', 'src' => $source, 'dst' => $dest );
489 $cases[] = array(
490 $op, // operation
491 $source, // source
492 $dest, // dest
495 $op2 = $op;
496 $op2['overwrite'] = true;
497 $cases[] = array(
498 $op2, // operation
499 $source, // source
500 $dest, // dest
503 $op2 = $op;
504 $op2['overwriteSame'] = true;
505 $cases[] = array(
506 $op2, // operation
507 $source, // source
508 $dest, // dest
511 $op2 = $op;
512 $op2['ignoreMissingSource'] = true;
513 $cases[] = array(
514 $op2, // operation
515 $source, // source
516 $dest, // dest
519 $op2 = $op;
520 $op2['ignoreMissingSource'] = true;
521 $cases[] = array(
522 $op2, // operation
523 self::baseStorePath() . '/unittest-cont-bad/e/file.txt', // source
524 $dest, // dest
527 return $cases;
531 * @dataProvider provider_testDelete
532 * @covers FileBackend::doOperation
534 public function testDelete( $op, $withSource, $okStatus ) {
535 $this->backend = $this->singleBackend;
536 $this->tearDownFiles();
537 $this->doTestDelete( $op, $withSource, $okStatus );
538 $this->tearDownFiles();
540 $this->backend = $this->multiBackend;
541 $this->tearDownFiles();
542 $this->doTestDelete( $op, $withSource, $okStatus );
543 $this->tearDownFiles();
546 private function doTestDelete( $op, $withSource, $okStatus ) {
547 $backendName = $this->backendClass();
549 $source = $op['src'];
550 $this->prepare( array( 'dir' => dirname( $source ) ) );
552 if ( $withSource ) {
553 $status = $this->backend->doOperation(
554 array( 'op' => 'create', 'content' => 'blahblah', 'dst' => $source ) );
555 $this->assertGoodStatus( $status,
556 "Creation of file at $source succeeded ($backendName)." );
559 $status = $this->backend->doOperation( $op );
560 if ( $okStatus ) {
561 $this->assertGoodStatus( $status,
562 "Deletion of file at $source succeeded without warnings ($backendName)." );
563 $this->assertEquals( true, $status->isOK(),
564 "Deletion of file at $source succeeded ($backendName)." );
565 $this->assertEquals( array( 0 => true ), $status->success,
566 "Deletion of file at $source has proper 'success' field in Status ($backendName)." );
567 } else {
568 $this->assertEquals( false, $status->isOK(),
569 "Deletion of file at $source failed ($backendName)." );
572 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $source ) ),
573 "Source file $source does not exist after move ($backendName)." );
575 $this->assertFalse(
576 $this->backend->getFileSize( array( 'src' => $source ) ),
577 "Source file $source has correct size (false) ($backendName)." );
579 $props1 = $this->backend->getFileProps( array( 'src' => $source ) );
580 $this->assertFalse( $props1['fileExists'],
581 "Source file $source does not exist according to props ($backendName)." );
583 $this->assertBackendPathsConsistent( array( $source ) );
586 public static function provider_testDelete() {
587 $cases = array();
589 $source = self::baseStorePath() . '/unittest-cont1/e/myfacefile.txt';
591 $op = array( 'op' => 'delete', 'src' => $source );
592 $cases[] = array(
593 $op, // operation
594 true, // with source
595 true // succeeds
598 $cases[] = array(
599 $op, // operation
600 false, // without source
601 false // fails
604 $op['ignoreMissingSource'] = true;
605 $cases[] = array(
606 $op, // operation
607 false, // without source
608 true // succeeds
611 $op['ignoreMissingSource'] = true;
612 $op['src'] = self::baseStorePath() . '/unittest-cont-bad/e/file.txt';
613 $cases[] = array(
614 $op, // operation
615 false, // without source
616 true // succeeds
619 return $cases;
623 * @dataProvider provider_testDescribe
624 * @covers FileBackend::doOperation
626 public function testDescribe( $op, $withSource, $okStatus ) {
627 $this->backend = $this->singleBackend;
628 $this->tearDownFiles();
629 $this->doTestDescribe( $op, $withSource, $okStatus );
630 $this->tearDownFiles();
632 $this->backend = $this->multiBackend;
633 $this->tearDownFiles();
634 $this->doTestDescribe( $op, $withSource, $okStatus );
635 $this->tearDownFiles();
638 private function doTestDescribe( $op, $withSource, $okStatus ) {
639 $backendName = $this->backendClass();
641 $source = $op['src'];
642 $this->prepare( array( 'dir' => dirname( $source ) ) );
644 if ( $withSource ) {
645 $status = $this->backend->doOperation(
646 array( 'op' => 'create', 'content' => 'blahblah', 'dst' => $source,
647 'headers' => array( 'Content-Disposition' => 'xxx' ) ) );
648 $this->assertGoodStatus( $status,
649 "Creation of file at $source succeeded ($backendName)." );
650 if ( $this->backend->hasFeatures( FileBackend::ATTR_HEADERS ) ) {
651 $attr = $this->backend->getFileXAttributes( array( 'src' => $source ) );
652 $this->assertHasHeaders( array( 'Content-Disposition' => 'xxx' ), $attr );
655 $status = $this->backend->describe( array( 'src' => $source,
656 'headers' => array( 'Content-Disposition' => '' ) ) ); // remove
657 $this->assertGoodStatus( $status,
658 "Removal of header for $source succeeded ($backendName)." );
660 if ( $this->backend->hasFeatures( FileBackend::ATTR_HEADERS ) ) {
661 $attr = $this->backend->getFileXAttributes( array( 'src' => $source ) );
662 $this->assertFalse( isset( $attr['headers']['content-disposition'] ),
663 "File 'Content-Disposition' header removed." );
667 $status = $this->backend->doOperation( $op );
668 if ( $okStatus ) {
669 $this->assertGoodStatus( $status,
670 "Describe of file at $source succeeded without warnings ($backendName)." );
671 $this->assertEquals( true, $status->isOK(),
672 "Describe of file at $source succeeded ($backendName)." );
673 $this->assertEquals( array( 0 => true ), $status->success,
674 "Describe of file at $source has proper 'success' field in Status ($backendName)." );
675 if ( $this->backend->hasFeatures( FileBackend::ATTR_HEADERS ) ) {
676 $attr = $this->backend->getFileXAttributes( array( 'src' => $source ) );
677 $this->assertHasHeaders( $op['headers'], $attr );
679 } else {
680 $this->assertEquals( false, $status->isOK(),
681 "Describe of file at $source failed ($backendName)." );
684 $this->assertBackendPathsConsistent( array( $source ) );
687 private function assertHasHeaders( array $headers, array $attr ) {
688 foreach ( $headers as $n => $v ) {
689 if ( $n !== '' ) {
690 $this->assertTrue( isset( $attr['headers'][strtolower( $n )] ),
691 "File has '$n' header." );
692 $this->assertEquals( $v, $attr['headers'][strtolower( $n )],
693 "File has '$n' header value." );
694 } else {
695 $this->assertFalse( isset( $attr['headers'][strtolower( $n )] ),
696 "File does not have '$n' header." );
701 public static function provider_testDescribe() {
702 $cases = array();
704 $source = self::baseStorePath() . '/unittest-cont1/e/myfacefile.txt';
706 $op = array( 'op' => 'describe', 'src' => $source,
707 'headers' => array( 'Content-Disposition' => 'inline' ), );
708 $cases[] = array(
709 $op, // operation
710 true, // with source
711 true // succeeds
714 $cases[] = array(
715 $op, // operation
716 false, // without source
717 false // fails
720 return $cases;
724 * @dataProvider provider_testCreate
725 * @covers FileBackend::doOperation
727 public function testCreate( $op, $alreadyExists, $okStatus, $newSize ) {
728 $this->backend = $this->singleBackend;
729 $this->tearDownFiles();
730 $this->doTestCreate( $op, $alreadyExists, $okStatus, $newSize );
731 $this->tearDownFiles();
733 $this->backend = $this->multiBackend;
734 $this->tearDownFiles();
735 $this->doTestCreate( $op, $alreadyExists, $okStatus, $newSize );
736 $this->tearDownFiles();
739 private function doTestCreate( $op, $alreadyExists, $okStatus, $newSize ) {
740 $backendName = $this->backendClass();
742 $dest = $op['dst'];
743 $this->prepare( array( 'dir' => dirname( $dest ) ) );
745 $oldText = 'blah...blah...waahwaah';
746 if ( $alreadyExists ) {
747 $status = $this->backend->doOperation(
748 array( 'op' => 'create', 'content' => $oldText, 'dst' => $dest ) );
749 $this->assertGoodStatus( $status,
750 "Creation of file at $dest succeeded ($backendName)." );
753 $status = $this->backend->doOperation( $op );
754 if ( $okStatus ) {
755 $this->assertGoodStatus( $status,
756 "Creation of file at $dest succeeded without warnings ($backendName)." );
757 $this->assertEquals( true, $status->isOK(),
758 "Creation of file at $dest succeeded ($backendName)." );
759 $this->assertEquals( array( 0 => true ), $status->success,
760 "Creation of file at $dest has proper 'success' field in Status ($backendName)." );
761 } else {
762 $this->assertEquals( false, $status->isOK(),
763 "Creation of file at $dest failed ($backendName)." );
766 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $dest ) ),
767 "Destination file $dest exists after creation ($backendName)." );
769 $props1 = $this->backend->getFileProps( array( 'src' => $dest ) );
770 $this->assertEquals( true, $props1['fileExists'],
771 "Destination file $dest exists according to props ($backendName)." );
772 if ( $okStatus ) { // file content is what we saved
773 $this->assertEquals( $newSize, $props1['size'],
774 "Destination file $dest has expected size according to props ($backendName)." );
775 $this->assertEquals( $newSize,
776 $this->backend->getFileSize( array( 'src' => $dest ) ),
777 "Destination file $dest has correct size ($backendName)." );
778 } else { // file content is some other previous text
779 $this->assertEquals( strlen( $oldText ), $props1['size'],
780 "Destination file $dest has original size according to props ($backendName)." );
781 $this->assertEquals( strlen( $oldText ),
782 $this->backend->getFileSize( array( 'src' => $dest ) ),
783 "Destination file $dest has original size according to props ($backendName)." );
786 $this->assertBackendPathsConsistent( array( $dest ) );
790 * @dataProvider provider_testCreate
792 public static function provider_testCreate() {
793 $cases = array();
795 $dest = self::baseStorePath() . '/unittest-cont2/a/myspacefile.txt';
797 $op = array( 'op' => 'create', 'content' => 'test test testing', 'dst' => $dest );
798 $cases[] = array(
799 $op, // operation
800 false, // no dest already exists
801 true, // succeeds
802 strlen( $op['content'] )
805 $op2 = $op;
806 $op2['content'] = "\n";
807 $cases[] = array(
808 $op2, // operation
809 false, // no dest already exists
810 true, // succeeds
811 strlen( $op2['content'] )
814 $op2 = $op;
815 $op2['content'] = "fsf\n waf 3kt";
816 $cases[] = array(
817 $op2, // operation
818 true, // dest already exists
819 false, // fails
820 strlen( $op2['content'] )
823 $op2 = $op;
824 $op2['content'] = "egm'g gkpe gpqg eqwgwqg";
825 $op2['overwrite'] = true;
826 $cases[] = array(
827 $op2, // operation
828 true, // dest already exists
829 true, // succeeds
830 strlen( $op2['content'] )
833 $op2 = $op;
834 $op2['content'] = "39qjmg3-qg";
835 $op2['overwriteSame'] = true;
836 $cases[] = array(
837 $op2, // operation
838 true, // dest already exists
839 false, // succeeds
840 strlen( $op2['content'] )
843 return $cases;
847 * @covers FileBackend::doQuickOperations
849 public function testDoQuickOperations() {
850 $this->backend = $this->singleBackend;
851 $this->doTestDoQuickOperations();
852 $this->tearDownFiles();
854 $this->backend = $this->multiBackend;
855 $this->doTestDoQuickOperations();
856 $this->tearDownFiles();
859 private function doTestDoQuickOperations() {
860 $backendName = $this->backendClass();
862 $base = self::baseStorePath();
863 $files = array(
864 "$base/unittest-cont1/e/fileA.a",
865 "$base/unittest-cont1/e/fileB.a",
866 "$base/unittest-cont1/e/fileC.a"
868 $createOps = array();
869 $purgeOps = array();
870 foreach ( $files as $path ) {
871 $status = $this->prepare( array( 'dir' => dirname( $path ) ) );
872 $this->assertGoodStatus( $status,
873 "Preparing $path succeeded without warnings ($backendName)." );
874 $createOps[] = array( 'op' => 'create', 'dst' => $path, 'content' => mt_rand( 0, 50000 ) );
875 $copyOps[] = array( 'op' => 'copy', 'src' => $path, 'dst' => "$path-2" );
876 $moveOps[] = array( 'op' => 'move', 'src' => "$path-2", 'dst' => "$path-3" );
877 $purgeOps[] = array( 'op' => 'delete', 'src' => $path );
878 $purgeOps[] = array( 'op' => 'delete', 'src' => "$path-3" );
880 $purgeOps[] = array( 'op' => 'null' );
882 $this->assertGoodStatus(
883 $this->backend->doQuickOperations( $createOps ),
884 "Creation of source files succeeded ($backendName)." );
885 foreach ( $files as $file ) {
886 $this->assertTrue( $this->backend->fileExists( array( 'src' => $file ) ),
887 "File $file exists." );
890 $this->assertGoodStatus(
891 $this->backend->doQuickOperations( $copyOps ),
892 "Quick copy of source files succeeded ($backendName)." );
893 foreach ( $files as $file ) {
894 $this->assertTrue( $this->backend->fileExists( array( 'src' => "$file-2" ) ),
895 "File $file-2 exists." );
898 $this->assertGoodStatus(
899 $this->backend->doQuickOperations( $moveOps ),
900 "Quick move of source files succeeded ($backendName)." );
901 foreach ( $files as $file ) {
902 $this->assertTrue( $this->backend->fileExists( array( 'src' => "$file-3" ) ),
903 "File $file-3 move in." );
904 $this->assertFalse( $this->backend->fileExists( array( 'src' => "$file-2" ) ),
905 "File $file-2 moved away." );
908 $this->assertGoodStatus(
909 $this->backend->quickCopy( array( 'src' => $files[0], 'dst' => $files[0] ) ),
910 "Copy of file {$files[0]} over itself succeeded ($backendName)." );
911 $this->assertTrue( $this->backend->fileExists( array( 'src' => $files[0] ) ),
912 "File {$files[0]} still exists." );
914 $this->assertGoodStatus(
915 $this->backend->quickMove( array( 'src' => $files[0], 'dst' => $files[0] ) ),
916 "Move of file {$files[0]} over itself succeeded ($backendName)." );
917 $this->assertTrue( $this->backend->fileExists( array( 'src' => $files[0] ) ),
918 "File {$files[0]} still exists." );
920 $this->assertGoodStatus(
921 $this->backend->doQuickOperations( $purgeOps ),
922 "Quick deletion of source files succeeded ($backendName)." );
923 foreach ( $files as $file ) {
924 $this->assertFalse( $this->backend->fileExists( array( 'src' => $file ) ),
925 "File $file purged." );
926 $this->assertFalse( $this->backend->fileExists( array( 'src' => "$file-3" ) ),
927 "File $file-3 purged." );
932 * @dataProvider provider_testConcatenate
934 public function testConcatenate( $op, $srcs, $srcsContent, $alreadyExists, $okStatus ) {
935 $this->backend = $this->singleBackend;
936 $this->tearDownFiles();
937 $this->doTestConcatenate( $op, $srcs, $srcsContent, $alreadyExists, $okStatus );
938 $this->tearDownFiles();
940 $this->backend = $this->multiBackend;
941 $this->tearDownFiles();
942 $this->doTestConcatenate( $op, $srcs, $srcsContent, $alreadyExists, $okStatus );
943 $this->tearDownFiles();
946 private function doTestConcatenate( $params, $srcs, $srcsContent, $alreadyExists, $okStatus ) {
947 $backendName = $this->backendClass();
949 $expContent = '';
950 // Create sources
951 $ops = array();
952 foreach ( $srcs as $i => $source ) {
953 $this->prepare( array( 'dir' => dirname( $source ) ) );
954 $ops[] = array(
955 'op' => 'create', // operation
956 'dst' => $source, // source
957 'content' => $srcsContent[$i]
959 $expContent .= $srcsContent[$i];
961 $status = $this->backend->doOperations( $ops );
963 $this->assertGoodStatus( $status,
964 "Creation of source files succeeded ($backendName)." );
966 $dest = $params['dst'] = $this->getNewTempFile();
967 if ( $alreadyExists ) {
968 $ok = file_put_contents( $dest, 'blah...blah...waahwaah' ) !== false;
969 $this->assertEquals( true, $ok,
970 "Creation of file at $dest succeeded ($backendName)." );
971 } else {
972 $ok = file_put_contents( $dest, '' ) !== false;
973 $this->assertEquals( true, $ok,
974 "Creation of 0-byte file at $dest succeeded ($backendName)." );
977 // Combine the files into one
978 $status = $this->backend->concatenate( $params );
979 if ( $okStatus ) {
980 $this->assertGoodStatus( $status,
981 "Creation of concat file at $dest succeeded without warnings ($backendName)." );
982 $this->assertEquals( true, $status->isOK(),
983 "Creation of concat file at $dest succeeded ($backendName)." );
984 } else {
985 $this->assertEquals( false, $status->isOK(),
986 "Creation of concat file at $dest failed ($backendName)." );
989 if ( $okStatus ) {
990 $this->assertEquals( true, is_file( $dest ),
991 "Dest concat file $dest exists after creation ($backendName)." );
992 } else {
993 $this->assertEquals( true, is_file( $dest ),
994 "Dest concat file $dest exists after failed creation ($backendName)." );
997 $contents = file_get_contents( $dest );
998 $this->assertNotEquals( false, $contents, "File at $dest exists ($backendName)." );
1000 if ( $okStatus ) {
1001 $this->assertEquals( $expContent, $contents,
1002 "Concat file at $dest has correct contents ($backendName)." );
1003 } else {
1004 $this->assertNotEquals( $expContent, $contents,
1005 "Concat file at $dest has correct contents ($backendName)." );
1009 public static function provider_testConcatenate() {
1010 $cases = array();
1012 $srcs = array(
1013 self::baseStorePath() . '/unittest-cont1/e/file1.txt',
1014 self::baseStorePath() . '/unittest-cont1/e/file2.txt',
1015 self::baseStorePath() . '/unittest-cont1/e/file3.txt',
1016 self::baseStorePath() . '/unittest-cont1/e/file4.txt',
1017 self::baseStorePath() . '/unittest-cont1/e/file5.txt',
1018 self::baseStorePath() . '/unittest-cont1/e/file6.txt',
1019 self::baseStorePath() . '/unittest-cont1/e/file7.txt',
1020 self::baseStorePath() . '/unittest-cont1/e/file8.txt',
1021 self::baseStorePath() . '/unittest-cont1/e/file9.txt',
1022 self::baseStorePath() . '/unittest-cont1/e/file10.txt'
1024 $content = array(
1025 'egfage',
1026 'ageageag',
1027 'rhokohlr',
1028 'shgmslkg',
1029 'kenga',
1030 'owagmal',
1031 'kgmae',
1032 'g eak;g',
1033 'lkaem;a',
1034 'legma'
1036 $params = array( 'srcs' => $srcs );
1038 $cases[] = array(
1039 $params, // operation
1040 $srcs, // sources
1041 $content, // content for each source
1042 false, // no dest already exists
1043 true, // succeeds
1046 $cases[] = array(
1047 $params, // operation
1048 $srcs, // sources
1049 $content, // content for each source
1050 true, // dest already exists
1051 false, // succeeds
1054 return $cases;
1058 * @dataProvider provider_testGetFileStat
1059 * @covers FileBackend::getFileStat
1061 public function testGetFileStat( $path, $content, $alreadyExists ) {
1062 $this->backend = $this->singleBackend;
1063 $this->tearDownFiles();
1064 $this->doTestGetFileStat( $path, $content, $alreadyExists );
1065 $this->tearDownFiles();
1067 $this->backend = $this->multiBackend;
1068 $this->tearDownFiles();
1069 $this->doTestGetFileStat( $path, $content, $alreadyExists );
1070 $this->tearDownFiles();
1073 private function doTestGetFileStat( $path, $content, $alreadyExists ) {
1074 $backendName = $this->backendClass();
1076 if ( $alreadyExists ) {
1077 $this->prepare( array( 'dir' => dirname( $path ) ) );
1078 $status = $this->create( array( 'dst' => $path, 'content' => $content ) );
1079 $this->assertGoodStatus( $status,
1080 "Creation of file at $path succeeded ($backendName)." );
1082 $size = $this->backend->getFileSize( array( 'src' => $path ) );
1083 $time = $this->backend->getFileTimestamp( array( 'src' => $path ) );
1084 $stat = $this->backend->getFileStat( array( 'src' => $path ) );
1086 $this->assertEquals( strlen( $content ), $size,
1087 "Correct file size of '$path'" );
1088 $this->assertTrue( abs( time() - wfTimestamp( TS_UNIX, $time ) ) < 10,
1089 "Correct file timestamp of '$path'" );
1091 $size = $stat['size'];
1092 $time = $stat['mtime'];
1093 $this->assertEquals( strlen( $content ), $size,
1094 "Correct file size of '$path'" );
1095 $this->assertTrue( abs( time() - wfTimestamp( TS_UNIX, $time ) ) < 10,
1096 "Correct file timestamp of '$path'" );
1098 $this->backend->clearCache( array( $path ) );
1100 $size = $this->backend->getFileSize( array( 'src' => $path ) );
1102 $this->assertEquals( strlen( $content ), $size,
1103 "Correct file size of '$path'" );
1105 $this->backend->preloadCache( array( $path ) );
1107 $size = $this->backend->getFileSize( array( 'src' => $path ) );
1109 $this->assertEquals( strlen( $content ), $size,
1110 "Correct file size of '$path'" );
1111 } else {
1112 $size = $this->backend->getFileSize( array( 'src' => $path ) );
1113 $time = $this->backend->getFileTimestamp( array( 'src' => $path ) );
1114 $stat = $this->backend->getFileStat( array( 'src' => $path ) );
1116 $this->assertFalse( $size, "Correct file size of '$path'" );
1117 $this->assertFalse( $time, "Correct file timestamp of '$path'" );
1118 $this->assertFalse( $stat, "Correct file stat of '$path'" );
1122 public static function provider_testGetFileStat() {
1123 $cases = array();
1125 $base = self::baseStorePath();
1126 $cases[] = array( "$base/unittest-cont1/e/b/z/some_file.txt", "some file contents", true );
1127 $cases[] = array( "$base/unittest-cont1/e/b/some-other_file.txt", "", true );
1128 $cases[] = array( "$base/unittest-cont1/e/b/some-diff_file.txt", null, false );
1130 return $cases;
1134 * @dataProvider provider_testGetFileStat
1135 * @covers FileBackend::streamFile
1137 public function testStreamFile( $path, $content, $alreadyExists ) {
1138 $this->backend = $this->singleBackend;
1139 $this->tearDownFiles();
1140 $this->doTestStreamFile( $path, $content, $alreadyExists );
1141 $this->tearDownFiles();
1144 private function doTestStreamFile( $path, $content ) {
1145 $backendName = $this->backendClass();
1147 // Test doStreamFile() directly to avoid header madness
1148 $class = new ReflectionClass( $this->backend );
1149 $method = $class->getMethod( 'doStreamFile' );
1150 $method->setAccessible( true );
1152 if ( $content !== null ) {
1153 $this->prepare( array( 'dir' => dirname( $path ) ) );
1154 $status = $this->create( array( 'dst' => $path, 'content' => $content ) );
1155 $this->assertGoodStatus( $status,
1156 "Creation of file at $path succeeded ($backendName)." );
1158 ob_start();
1159 $method->invokeArgs( $this->backend, array( array( 'src' => $path ) ) );
1160 $data = ob_get_contents();
1161 ob_end_clean();
1163 $this->assertEquals( $content, $data, "Correct content streamed from '$path'" );
1164 } else { // 404 case
1165 ob_start();
1166 $method->invokeArgs( $this->backend, array( array( 'src' => $path ) ) );
1167 $data = ob_get_contents();
1168 ob_end_clean();
1170 $this->assertEquals( '', $data, "Correct content streamed from '$path' ($backendName)" );
1174 public static function provider_testStreamFile() {
1175 $cases = array();
1177 $base = self::baseStorePath();
1178 $cases[] = array( "$base/unittest-cont1/e/b/z/some_file.txt", "some file contents" );
1179 $cases[] = array( "$base/unittest-cont1/e/b/some-other_file.txt", null );
1181 return $cases;
1185 * @dataProvider provider_testGetFileContents
1186 * @covers FileBackend::getFileContents
1187 * @covers FileBackend::getFileContentsMulti
1189 public function testGetFileContents( $source, $content ) {
1190 $this->backend = $this->singleBackend;
1191 $this->tearDownFiles();
1192 $this->doTestGetFileContents( $source, $content );
1193 $this->tearDownFiles();
1195 $this->backend = $this->multiBackend;
1196 $this->tearDownFiles();
1197 $this->doTestGetFileContents( $source, $content );
1198 $this->tearDownFiles();
1201 private function doTestGetFileContents( $source, $content ) {
1202 $backendName = $this->backendClass();
1204 $srcs = (array)$source;
1205 $content = (array)$content;
1206 foreach ( $srcs as $i => $src ) {
1207 $this->prepare( array( 'dir' => dirname( $src ) ) );
1208 $status = $this->backend->doOperation(
1209 array( 'op' => 'create', 'content' => $content[$i], 'dst' => $src ) );
1210 $this->assertGoodStatus( $status,
1211 "Creation of file at $src succeeded ($backendName)." );
1214 if ( is_array( $source ) ) {
1215 $contents = $this->backend->getFileContentsMulti( array( 'srcs' => $source ) );
1216 foreach ( $contents as $path => $data ) {
1217 $this->assertNotEquals( false, $data, "Contents of $path exists ($backendName)." );
1218 $this->assertEquals(
1219 current( $content ),
1220 $data,
1221 "Contents of $path is correct ($backendName)."
1223 next( $content );
1225 $this->assertEquals(
1226 $source,
1227 array_keys( $contents ),
1228 "Contents in right order ($backendName)."
1230 $this->assertEquals(
1231 count( $source ),
1232 count( $contents ),
1233 "Contents array size correct ($backendName)."
1235 } else {
1236 $data = $this->backend->getFileContents( array( 'src' => $source ) );
1237 $this->assertNotEquals( false, $data, "Contents of $source exists ($backendName)." );
1238 $this->assertEquals( $content[0], $data, "Contents of $source is correct ($backendName)." );
1242 public static function provider_testGetFileContents() {
1243 $cases = array();
1245 $base = self::baseStorePath();
1246 $cases[] = array( "$base/unittest-cont1/e/b/z/some_file.txt", "some file contents" );
1247 $cases[] = array( "$base/unittest-cont1/e/b/some-other_file.txt", "more file contents" );
1248 $cases[] = array(
1249 array( "$base/unittest-cont1/e/a/x.txt", "$base/unittest-cont1/e/a/y.txt",
1250 "$base/unittest-cont1/e/a/z.txt" ),
1251 array( "contents xx", "contents xy", "contents xz" )
1254 return $cases;
1258 * @dataProvider provider_testGetLocalCopy
1259 * @covers FileBackend::getLocalCopy
1261 public function testGetLocalCopy( $source, $content ) {
1262 $this->backend = $this->singleBackend;
1263 $this->tearDownFiles();
1264 $this->doTestGetLocalCopy( $source, $content );
1265 $this->tearDownFiles();
1267 $this->backend = $this->multiBackend;
1268 $this->tearDownFiles();
1269 $this->doTestGetLocalCopy( $source, $content );
1270 $this->tearDownFiles();
1273 private function doTestGetLocalCopy( $source, $content ) {
1274 $backendName = $this->backendClass();
1276 $srcs = (array)$source;
1277 $content = (array)$content;
1278 foreach ( $srcs as $i => $src ) {
1279 $this->prepare( array( 'dir' => dirname( $src ) ) );
1280 $status = $this->backend->doOperation(
1281 array( 'op' => 'create', 'content' => $content[$i], 'dst' => $src ) );
1282 $this->assertGoodStatus( $status,
1283 "Creation of file at $src succeeded ($backendName)." );
1286 if ( is_array( $source ) ) {
1287 $tmpFiles = $this->backend->getLocalCopyMulti( array( 'srcs' => $source ) );
1288 foreach ( $tmpFiles as $path => $tmpFile ) {
1289 $this->assertNotNull( $tmpFile,
1290 "Creation of local copy of $path succeeded ($backendName)." );
1291 $contents = file_get_contents( $tmpFile->getPath() );
1292 $this->assertNotEquals( false, $contents, "Local copy of $path exists ($backendName)." );
1293 $this->assertEquals(
1294 current( $content ),
1295 $contents,
1296 "Local copy of $path is correct ($backendName)."
1298 next( $content );
1300 $this->assertEquals(
1301 $source,
1302 array_keys( $tmpFiles ),
1303 "Local copies in right order ($backendName)."
1305 $this->assertEquals(
1306 count( $source ),
1307 count( $tmpFiles ),
1308 "Local copies array size correct ($backendName)."
1310 } else {
1311 $tmpFile = $this->backend->getLocalCopy( array( 'src' => $source ) );
1312 $this->assertNotNull( $tmpFile,
1313 "Creation of local copy of $source succeeded ($backendName)." );
1314 $contents = file_get_contents( $tmpFile->getPath() );
1315 $this->assertNotEquals( false, $contents, "Local copy of $source exists ($backendName)." );
1316 $this->assertEquals(
1317 $content[0],
1318 $contents,
1319 "Local copy of $source is correct ($backendName)."
1323 $obj = new stdClass();
1324 $tmpFile->bind( $obj );
1327 public static function provider_testGetLocalCopy() {
1328 $cases = array();
1330 $base = self::baseStorePath();
1331 $cases[] = array( "$base/unittest-cont1/e/a/z/some_file.txt", "some file contents" );
1332 $cases[] = array( "$base/unittest-cont1/e/a/some-other_file.txt", "more file contents" );
1333 $cases[] = array( "$base/unittest-cont1/e/a/\$odd&.txt", "test file contents" );
1334 $cases[] = array(
1335 array( "$base/unittest-cont1/e/a/x.txt", "$base/unittest-cont1/e/a/y.txt",
1336 "$base/unittest-cont1/e/a/z.txt" ),
1337 array( "contents xx $", "contents xy 111", "contents xz" )
1340 return $cases;
1344 * @dataProvider provider_testGetLocalReference
1345 * @covers FileBackend::getLocalReference
1347 public function testGetLocalReference( $source, $content ) {
1348 $this->backend = $this->singleBackend;
1349 $this->tearDownFiles();
1350 $this->doTestGetLocalReference( $source, $content );
1351 $this->tearDownFiles();
1353 $this->backend = $this->multiBackend;
1354 $this->tearDownFiles();
1355 $this->doTestGetLocalReference( $source, $content );
1356 $this->tearDownFiles();
1359 private function doTestGetLocalReference( $source, $content ) {
1360 $backendName = $this->backendClass();
1362 $srcs = (array)$source;
1363 $content = (array)$content;
1364 foreach ( $srcs as $i => $src ) {
1365 $this->prepare( array( 'dir' => dirname( $src ) ) );
1366 $status = $this->backend->doOperation(
1367 array( 'op' => 'create', 'content' => $content[$i], 'dst' => $src ) );
1368 $this->assertGoodStatus( $status,
1369 "Creation of file at $src succeeded ($backendName)." );
1372 if ( is_array( $source ) ) {
1373 $tmpFiles = $this->backend->getLocalReferenceMulti( array( 'srcs' => $source ) );
1374 foreach ( $tmpFiles as $path => $tmpFile ) {
1375 $this->assertNotNull( $tmpFile,
1376 "Creation of local copy of $path succeeded ($backendName)." );
1377 $contents = file_get_contents( $tmpFile->getPath() );
1378 $this->assertNotEquals( false, $contents, "Local ref of $path exists ($backendName)." );
1379 $this->assertEquals(
1380 current( $content ),
1381 $contents,
1382 "Local ref of $path is correct ($backendName)."
1384 next( $content );
1386 $this->assertEquals(
1387 $source,
1388 array_keys( $tmpFiles ),
1389 "Local refs in right order ($backendName)."
1391 $this->assertEquals(
1392 count( $source ),
1393 count( $tmpFiles ),
1394 "Local refs array size correct ($backendName)."
1396 } else {
1397 $tmpFile = $this->backend->getLocalReference( array( 'src' => $source ) );
1398 $this->assertNotNull( $tmpFile,
1399 "Creation of local copy of $source succeeded ($backendName)." );
1400 $contents = file_get_contents( $tmpFile->getPath() );
1401 $this->assertNotEquals( false, $contents, "Local ref of $source exists ($backendName)." );
1402 $this->assertEquals( $content[0], $contents, "Local ref of $source is correct ($backendName)." );
1406 public static function provider_testGetLocalReference() {
1407 $cases = array();
1409 $base = self::baseStorePath();
1410 $cases[] = array( "$base/unittest-cont1/e/a/z/some_file.txt", "some file contents" );
1411 $cases[] = array( "$base/unittest-cont1/e/a/some-other_file.txt", "more file contents" );
1412 $cases[] = array( "$base/unittest-cont1/e/a/\$odd&.txt", "test file contents" );
1413 $cases[] = array(
1414 array( "$base/unittest-cont1/e/a/x.txt", "$base/unittest-cont1/e/a/y.txt",
1415 "$base/unittest-cont1/e/a/z.txt" ),
1416 array( "contents xx 1111", "contents xy %", "contents xz $" )
1419 return $cases;
1423 * @covers FileBackend::getLocalCopy
1424 * @covers FileBackend::getLocalReference
1426 public function testGetLocalCopyAndReference404() {
1427 $this->backend = $this->singleBackend;
1428 $this->tearDownFiles();
1429 $this->doTestGetLocalCopyAndReference404();
1430 $this->tearDownFiles();
1432 $this->backend = $this->multiBackend;
1433 $this->tearDownFiles();
1434 $this->doTestGetLocalCopyAndReference404();
1435 $this->tearDownFiles();
1438 public function doTestGetLocalCopyAndReference404() {
1439 $backendName = $this->backendClass();
1441 $base = self::baseStorePath();
1443 $tmpFile = $this->backend->getLocalCopy( array(
1444 'src' => "$base/unittest-cont1/not-there" ) );
1445 $this->assertEquals( null, $tmpFile, "Local copy of not existing file is null ($backendName)." );
1447 $tmpFile = $this->backend->getLocalReference( array(
1448 'src' => "$base/unittest-cont1/not-there" ) );
1449 $this->assertEquals( null, $tmpFile, "Local ref of not existing file is null ($backendName)." );
1453 * @dataProvider provider_testGetFileHttpUrl
1454 * @covers FileBackend::getFileHttpUrl
1456 public function testGetFileHttpUrl( $source, $content ) {
1457 $this->backend = $this->singleBackend;
1458 $this->tearDownFiles();
1459 $this->doTestGetFileHttpUrl( $source, $content );
1460 $this->tearDownFiles();
1462 $this->backend = $this->multiBackend;
1463 $this->tearDownFiles();
1464 $this->doTestGetFileHttpUrl( $source, $content );
1465 $this->tearDownFiles();
1468 private function doTestGetFileHttpUrl( $source, $content ) {
1469 $backendName = $this->backendClass();
1471 $this->prepare( array( 'dir' => dirname( $source ) ) );
1472 $status = $this->backend->doOperation(
1473 array( 'op' => 'create', 'content' => $content, 'dst' => $source ) );
1474 $this->assertGoodStatus( $status,
1475 "Creation of file at $source succeeded ($backendName)." );
1477 $url = $this->backend->getFileHttpUrl( array( 'src' => $source ) );
1479 if ( $url !== null ) { // supported
1480 $data = Http::request( "GET", $url, array(), __METHOD__ );
1481 $this->assertEquals( $content, $data,
1482 "HTTP GET of URL has right contents ($backendName)." );
1486 public static function provider_testGetFileHttpUrl() {
1487 $cases = array();
1489 $base = self::baseStorePath();
1490 $cases[] = array( "$base/unittest-cont1/e/a/z/some_file.txt", "some file contents" );
1491 $cases[] = array( "$base/unittest-cont1/e/a/some-other_file.txt", "more file contents" );
1492 $cases[] = array( "$base/unittest-cont1/e/a/\$odd&.txt", "test file contents" );
1494 return $cases;
1498 * @dataProvider provider_testPrepareAndClean
1499 * @covers FileBackend::prepare
1500 * @covers FileBackend::clean
1502 public function testPrepareAndClean( $path, $isOK ) {
1503 $this->backend = $this->singleBackend;
1504 $this->doTestPrepareAndClean( $path, $isOK );
1505 $this->tearDownFiles();
1507 $this->backend = $this->multiBackend;
1508 $this->doTestPrepareAndClean( $path, $isOK );
1509 $this->tearDownFiles();
1512 public static function provider_testPrepareAndClean() {
1513 $base = self::baseStorePath();
1515 return array(
1516 array( "$base/unittest-cont1/e/a/z/some_file1.txt", true ),
1517 array( "$base/unittest-cont2/a/z/some_file2.txt", true ),
1518 # Specific to FS backend with no basePath field set
1519 # array( "$base/unittest-cont3/a/z/some_file3.txt", false ),
1523 private function doTestPrepareAndClean( $path, $isOK ) {
1524 $backendName = $this->backendClass();
1526 $status = $this->prepare( array( 'dir' => dirname( $path ) ) );
1527 if ( $isOK ) {
1528 $this->assertGoodStatus( $status,
1529 "Preparing dir $path succeeded without warnings ($backendName)." );
1530 $this->assertEquals( true, $status->isOK(),
1531 "Preparing dir $path succeeded ($backendName)." );
1532 } else {
1533 $this->assertEquals( false, $status->isOK(),
1534 "Preparing dir $path failed ($backendName)." );
1537 $status = $this->backend->secure( array( 'dir' => dirname( $path ) ) );
1538 if ( $isOK ) {
1539 $this->assertGoodStatus( $status,
1540 "Securing dir $path succeeded without warnings ($backendName)." );
1541 $this->assertEquals( true, $status->isOK(),
1542 "Securing dir $path succeeded ($backendName)." );
1543 } else {
1544 $this->assertEquals( false, $status->isOK(),
1545 "Securing dir $path failed ($backendName)." );
1548 $status = $this->backend->publish( array( 'dir' => dirname( $path ) ) );
1549 if ( $isOK ) {
1550 $this->assertGoodStatus( $status,
1551 "Publishing dir $path succeeded without warnings ($backendName)." );
1552 $this->assertEquals( true, $status->isOK(),
1553 "Publishing dir $path succeeded ($backendName)." );
1554 } else {
1555 $this->assertEquals( false, $status->isOK(),
1556 "Publishing dir $path failed ($backendName)." );
1559 $status = $this->backend->clean( array( 'dir' => dirname( $path ) ) );
1560 if ( $isOK ) {
1561 $this->assertGoodStatus( $status,
1562 "Cleaning dir $path succeeded without warnings ($backendName)." );
1563 $this->assertEquals( true, $status->isOK(),
1564 "Cleaning dir $path succeeded ($backendName)." );
1565 } else {
1566 $this->assertEquals( false, $status->isOK(),
1567 "Cleaning dir $path failed ($backendName)." );
1571 public function testRecursiveClean() {
1572 $this->backend = $this->singleBackend;
1573 $this->doTestRecursiveClean();
1574 $this->tearDownFiles();
1576 $this->backend = $this->multiBackend;
1577 $this->doTestRecursiveClean();
1578 $this->tearDownFiles();
1582 * @covers FileBackend::clean
1584 private function doTestRecursiveClean() {
1585 $backendName = $this->backendClass();
1587 $base = self::baseStorePath();
1588 $dirs = array(
1589 "$base/unittest-cont1",
1590 "$base/unittest-cont1/e",
1591 "$base/unittest-cont1/e/a",
1592 "$base/unittest-cont1/e/a/b",
1593 "$base/unittest-cont1/e/a/b/c",
1594 "$base/unittest-cont1/e/a/b/c/d0",
1595 "$base/unittest-cont1/e/a/b/c/d1",
1596 "$base/unittest-cont1/e/a/b/c/d2",
1597 "$base/unittest-cont1/e/a/b/c/d0/1",
1598 "$base/unittest-cont1/e/a/b/c/d0/2",
1599 "$base/unittest-cont1/e/a/b/c/d1/3",
1600 "$base/unittest-cont1/e/a/b/c/d1/4",
1601 "$base/unittest-cont1/e/a/b/c/d2/5",
1602 "$base/unittest-cont1/e/a/b/c/d2/6"
1604 foreach ( $dirs as $dir ) {
1605 $status = $this->prepare( array( 'dir' => $dir ) );
1606 $this->assertGoodStatus( $status,
1607 "Preparing dir $dir succeeded without warnings ($backendName)." );
1610 if ( $this->backend instanceof FSFileBackend ) {
1611 foreach ( $dirs as $dir ) {
1612 $this->assertEquals( true, $this->backend->directoryExists( array( 'dir' => $dir ) ),
1613 "Dir $dir exists ($backendName)." );
1617 $status = $this->backend->clean(
1618 array( 'dir' => "$base/unittest-cont1", 'recursive' => 1 ) );
1619 $this->assertGoodStatus( $status,
1620 "Recursive cleaning of dir $dir succeeded without warnings ($backendName)." );
1622 foreach ( $dirs as $dir ) {
1623 $this->assertEquals( false, $this->backend->directoryExists( array( 'dir' => $dir ) ),
1624 "Dir $dir no longer exists ($backendName)." );
1629 * @covers FileBackend::doOperations
1631 public function testDoOperations() {
1632 $this->backend = $this->singleBackend;
1633 $this->tearDownFiles();
1634 $this->doTestDoOperations();
1635 $this->tearDownFiles();
1637 $this->backend = $this->multiBackend;
1638 $this->tearDownFiles();
1639 $this->doTestDoOperations();
1640 $this->tearDownFiles();
1643 private function doTestDoOperations() {
1644 $base = self::baseStorePath();
1646 $fileA = "$base/unittest-cont1/e/a/b/fileA.txt";
1647 $fileAContents = '3tqtmoeatmn4wg4qe-mg3qt3 tq';
1648 $fileB = "$base/unittest-cont1/e/a/b/fileB.txt";
1649 $fileBContents = 'g-jmq3gpqgt3qtg q3GT ';
1650 $fileC = "$base/unittest-cont1/e/a/b/fileC.txt";
1651 $fileCContents = 'eigna[ogmewt 3qt g3qg flew[ag';
1652 $fileD = "$base/unittest-cont1/e/a/b/fileD.txt";
1654 $this->prepare( array( 'dir' => dirname( $fileA ) ) );
1655 $this->create( array( 'dst' => $fileA, 'content' => $fileAContents ) );
1656 $this->prepare( array( 'dir' => dirname( $fileB ) ) );
1657 $this->create( array( 'dst' => $fileB, 'content' => $fileBContents ) );
1658 $this->prepare( array( 'dir' => dirname( $fileC ) ) );
1659 $this->create( array( 'dst' => $fileC, 'content' => $fileCContents ) );
1660 $this->prepare( array( 'dir' => dirname( $fileD ) ) );
1662 $status = $this->backend->doOperations( array(
1663 array( 'op' => 'describe', 'src' => $fileA,
1664 'headers' => array( 'X-Content-Length' => '91.3' ), 'disposition' => 'inline' ),
1665 array( 'op' => 'copy', 'src' => $fileA, 'dst' => $fileC, 'overwrite' => 1 ),
1666 // Now: A:<A>, B:<B>, C:<A>, D:<empty> (file:<orginal contents>)
1667 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileA, 'overwriteSame' => 1 ),
1668 // Now: A:<A>, B:<B>, C:<A>, D:<empty>
1669 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileD, 'overwrite' => 1 ),
1670 // Now: A:<A>, B:<B>, C:<empty>, D:<A>
1671 array( 'op' => 'move', 'src' => $fileB, 'dst' => $fileC ),
1672 // Now: A:<A>, B:<empty>, C:<B>, D:<A>
1673 array( 'op' => 'move', 'src' => $fileD, 'dst' => $fileA, 'overwriteSame' => 1 ),
1674 // Now: A:<A>, B:<empty>, C:<B>, D:<empty>
1675 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileA, 'overwrite' => 1 ),
1676 // Now: A:<B>, B:<empty>, C:<empty>, D:<empty>
1677 array( 'op' => 'copy', 'src' => $fileA, 'dst' => $fileC ),
1678 // Now: A:<B>, B:<empty>, C:<B>, D:<empty>
1679 array( 'op' => 'move', 'src' => $fileA, 'dst' => $fileC, 'overwriteSame' => 1 ),
1680 // Now: A:<empty>, B:<empty>, C:<B>, D:<empty>
1681 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileC, 'overwrite' => 1 ),
1682 // Does nothing
1683 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileC, 'overwriteSame' => 1 ),
1684 // Does nothing
1685 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileC, 'overwrite' => 1 ),
1686 // Does nothing
1687 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileC, 'overwriteSame' => 1 ),
1688 // Does nothing
1689 array( 'op' => 'null' ),
1690 // Does nothing
1691 ) );
1693 $this->assertGoodStatus( $status, "Operation batch succeeded" );
1694 $this->assertEquals( true, $status->isOK(), "Operation batch succeeded" );
1695 $this->assertEquals( 14, count( $status->success ),
1696 "Operation batch has correct success array" );
1698 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileA ) ),
1699 "File does not exist at $fileA" );
1700 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileB ) ),
1701 "File does not exist at $fileB" );
1702 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileD ) ),
1703 "File does not exist at $fileD" );
1705 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $fileC ) ),
1706 "File exists at $fileC" );
1707 $this->assertEquals( $fileBContents,
1708 $this->backend->getFileContents( array( 'src' => $fileC ) ),
1709 "Correct file contents of $fileC" );
1710 $this->assertEquals( strlen( $fileBContents ),
1711 $this->backend->getFileSize( array( 'src' => $fileC ) ),
1712 "Correct file size of $fileC" );
1713 $this->assertEquals( Wikimedia\base_convert( sha1( $fileBContents ), 16, 36, 31 ),
1714 $this->backend->getFileSha1Base36( array( 'src' => $fileC ) ),
1715 "Correct file SHA-1 of $fileC" );
1719 * @covers FileBackend::doOperations
1721 public function testDoOperationsPipeline() {
1722 $this->backend = $this->singleBackend;
1723 $this->tearDownFiles();
1724 $this->doTestDoOperationsPipeline();
1725 $this->tearDownFiles();
1727 $this->backend = $this->multiBackend;
1728 $this->tearDownFiles();
1729 $this->doTestDoOperationsPipeline();
1730 $this->tearDownFiles();
1733 // concurrency orientated
1734 private function doTestDoOperationsPipeline() {
1735 $base = self::baseStorePath();
1737 $fileAContents = '3tqtmoeatmn4wg4qe-mg3qt3 tq';
1738 $fileBContents = 'g-jmq3gpqgt3qtg q3GT ';
1739 $fileCContents = 'eigna[ogmewt 3qt g3qg flew[ag';
1741 $tmpNameA = TempFSFile::factory( "unittests_", 'txt' )->getPath();
1742 $tmpNameB = TempFSFile::factory( "unittests_", 'txt' )->getPath();
1743 $tmpNameC = TempFSFile::factory( "unittests_", 'txt' )->getPath();
1744 $this->addTmpFiles( array( $tmpNameA, $tmpNameB, $tmpNameC ) );
1745 file_put_contents( $tmpNameA, $fileAContents );
1746 file_put_contents( $tmpNameB, $fileBContents );
1747 file_put_contents( $tmpNameC, $fileCContents );
1749 $fileA = "$base/unittest-cont1/e/a/b/fileA.txt";
1750 $fileB = "$base/unittest-cont1/e/a/b/fileB.txt";
1751 $fileC = "$base/unittest-cont1/e/a/b/fileC.txt";
1752 $fileD = "$base/unittest-cont1/e/a/b/fileD.txt";
1754 $this->prepare( array( 'dir' => dirname( $fileA ) ) );
1755 $this->create( array( 'dst' => $fileA, 'content' => $fileAContents ) );
1756 $this->prepare( array( 'dir' => dirname( $fileB ) ) );
1757 $this->prepare( array( 'dir' => dirname( $fileC ) ) );
1758 $this->prepare( array( 'dir' => dirname( $fileD ) ) );
1760 $status = $this->backend->doOperations( array(
1761 array( 'op' => 'store', 'src' => $tmpNameA, 'dst' => $fileA, 'overwriteSame' => 1 ),
1762 array( 'op' => 'store', 'src' => $tmpNameB, 'dst' => $fileB, 'overwrite' => 1 ),
1763 array( 'op' => 'store', 'src' => $tmpNameC, 'dst' => $fileC, 'overwrite' => 1 ),
1764 array( 'op' => 'copy', 'src' => $fileA, 'dst' => $fileC, 'overwrite' => 1 ),
1765 // Now: A:<A>, B:<B>, C:<A>, D:<empty> (file:<orginal contents>)
1766 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileA, 'overwriteSame' => 1 ),
1767 // Now: A:<A>, B:<B>, C:<A>, D:<empty>
1768 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileD, 'overwrite' => 1 ),
1769 // Now: A:<A>, B:<B>, C:<empty>, D:<A>
1770 array( 'op' => 'move', 'src' => $fileB, 'dst' => $fileC ),
1771 // Now: A:<A>, B:<empty>, C:<B>, D:<A>
1772 array( 'op' => 'move', 'src' => $fileD, 'dst' => $fileA, 'overwriteSame' => 1 ),
1773 // Now: A:<A>, B:<empty>, C:<B>, D:<empty>
1774 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileA, 'overwrite' => 1 ),
1775 // Now: A:<B>, B:<empty>, C:<empty>, D:<empty>
1776 array( 'op' => 'copy', 'src' => $fileA, 'dst' => $fileC ),
1777 // Now: A:<B>, B:<empty>, C:<B>, D:<empty>
1778 array( 'op' => 'move', 'src' => $fileA, 'dst' => $fileC, 'overwriteSame' => 1 ),
1779 // Now: A:<empty>, B:<empty>, C:<B>, D:<empty>
1780 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileC, 'overwrite' => 1 ),
1781 // Does nothing
1782 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileC, 'overwriteSame' => 1 ),
1783 // Does nothing
1784 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileC, 'overwrite' => 1 ),
1785 // Does nothing
1786 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileC, 'overwriteSame' => 1 ),
1787 // Does nothing
1788 array( 'op' => 'null' ),
1789 // Does nothing
1790 ) );
1792 $this->assertGoodStatus( $status, "Operation batch succeeded" );
1793 $this->assertEquals( true, $status->isOK(), "Operation batch succeeded" );
1794 $this->assertEquals( 16, count( $status->success ),
1795 "Operation batch has correct success array" );
1797 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileA ) ),
1798 "File does not exist at $fileA" );
1799 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileB ) ),
1800 "File does not exist at $fileB" );
1801 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileD ) ),
1802 "File does not exist at $fileD" );
1804 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $fileC ) ),
1805 "File exists at $fileC" );
1806 $this->assertEquals( $fileBContents,
1807 $this->backend->getFileContents( array( 'src' => $fileC ) ),
1808 "Correct file contents of $fileC" );
1809 $this->assertEquals( strlen( $fileBContents ),
1810 $this->backend->getFileSize( array( 'src' => $fileC ) ),
1811 "Correct file size of $fileC" );
1812 $this->assertEquals( Wikimedia\base_convert( sha1( $fileBContents ), 16, 36, 31 ),
1813 $this->backend->getFileSha1Base36( array( 'src' => $fileC ) ),
1814 "Correct file SHA-1 of $fileC" );
1818 * @covers FileBackend::doOperations
1820 public function testDoOperationsFailing() {
1821 $this->backend = $this->singleBackend;
1822 $this->tearDownFiles();
1823 $this->doTestDoOperationsFailing();
1824 $this->tearDownFiles();
1826 $this->backend = $this->multiBackend;
1827 $this->tearDownFiles();
1828 $this->doTestDoOperationsFailing();
1829 $this->tearDownFiles();
1832 private function doTestDoOperationsFailing() {
1833 $base = self::baseStorePath();
1835 $fileA = "$base/unittest-cont2/a/b/fileA.txt";
1836 $fileAContents = '3tqtmoeatmn4wg4qe-mg3qt3 tq';
1837 $fileB = "$base/unittest-cont2/a/b/fileB.txt";
1838 $fileBContents = 'g-jmq3gpqgt3qtg q3GT ';
1839 $fileC = "$base/unittest-cont2/a/b/fileC.txt";
1840 $fileCContents = 'eigna[ogmewt 3qt g3qg flew[ag';
1841 $fileD = "$base/unittest-cont2/a/b/fileD.txt";
1843 $this->prepare( array( 'dir' => dirname( $fileA ) ) );
1844 $this->create( array( 'dst' => $fileA, 'content' => $fileAContents ) );
1845 $this->prepare( array( 'dir' => dirname( $fileB ) ) );
1846 $this->create( array( 'dst' => $fileB, 'content' => $fileBContents ) );
1847 $this->prepare( array( 'dir' => dirname( $fileC ) ) );
1848 $this->create( array( 'dst' => $fileC, 'content' => $fileCContents ) );
1850 $status = $this->backend->doOperations( array(
1851 array( 'op' => 'copy', 'src' => $fileA, 'dst' => $fileC, 'overwrite' => 1 ),
1852 // Now: A:<A>, B:<B>, C:<A>, D:<empty> (file:<orginal contents>)
1853 array( 'op' => 'copy', 'src' => $fileC, 'dst' => $fileA, 'overwriteSame' => 1 ),
1854 // Now: A:<A>, B:<B>, C:<A>, D:<empty>
1855 array( 'op' => 'copy', 'src' => $fileB, 'dst' => $fileD, 'overwrite' => 1 ),
1856 // Now: A:<A>, B:<B>, C:<A>, D:<B>
1857 array( 'op' => 'move', 'src' => $fileC, 'dst' => $fileD ),
1858 // Now: A:<A>, B:<B>, C:<A>, D:<empty> (failed)
1859 array( 'op' => 'move', 'src' => $fileB, 'dst' => $fileC, 'overwriteSame' => 1 ),
1860 // Now: A:<A>, B:<B>, C:<A>, D:<empty> (failed)
1861 array( 'op' => 'move', 'src' => $fileB, 'dst' => $fileA, 'overwrite' => 1 ),
1862 // Now: A:<B>, B:<empty>, C:<A>, D:<empty>
1863 array( 'op' => 'delete', 'src' => $fileD ),
1864 // Now: A:<B>, B:<empty>, C:<A>, D:<empty>
1865 array( 'op' => 'null' ),
1866 // Does nothing
1867 ), array( 'force' => 1 ) );
1869 $this->assertNotEquals( array(), $status->errors, "Operation had warnings" );
1870 $this->assertEquals( true, $status->isOK(), "Operation batch succeeded" );
1871 $this->assertEquals( 8, count( $status->success ),
1872 "Operation batch has correct success array" );
1874 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileB ) ),
1875 "File does not exist at $fileB" );
1876 $this->assertEquals( false, $this->backend->fileExists( array( 'src' => $fileD ) ),
1877 "File does not exist at $fileD" );
1879 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $fileA ) ),
1880 "File does not exist at $fileA" );
1881 $this->assertEquals( true, $this->backend->fileExists( array( 'src' => $fileC ) ),
1882 "File exists at $fileC" );
1883 $this->assertEquals( $fileBContents,
1884 $this->backend->getFileContents( array( 'src' => $fileA ) ),
1885 "Correct file contents of $fileA" );
1886 $this->assertEquals( strlen( $fileBContents ),
1887 $this->backend->getFileSize( array( 'src' => $fileA ) ),
1888 "Correct file size of $fileA" );
1889 $this->assertEquals( Wikimedia\base_convert( sha1( $fileBContents ), 16, 36, 31 ),
1890 $this->backend->getFileSha1Base36( array( 'src' => $fileA ) ),
1891 "Correct file SHA-1 of $fileA" );
1895 * @covers FileBackend::getFileList
1897 public function testGetFileList() {
1898 $this->backend = $this->singleBackend;
1899 $this->tearDownFiles();
1900 $this->doTestGetFileList();
1901 $this->tearDownFiles();
1903 $this->backend = $this->multiBackend;
1904 $this->tearDownFiles();
1905 $this->doTestGetFileList();
1906 $this->tearDownFiles();
1909 private function doTestGetFileList() {
1910 $backendName = $this->backendClass();
1911 $base = self::baseStorePath();
1913 // Should have no errors
1914 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont-notexists" ) );
1916 $files = array(
1917 "$base/unittest-cont1/e/test1.txt",
1918 "$base/unittest-cont1/e/test2.txt",
1919 "$base/unittest-cont1/e/test3.txt",
1920 "$base/unittest-cont1/e/subdir1/test1.txt",
1921 "$base/unittest-cont1/e/subdir1/test2.txt",
1922 "$base/unittest-cont1/e/subdir2/test3.txt",
1923 "$base/unittest-cont1/e/subdir2/test4.txt",
1924 "$base/unittest-cont1/e/subdir2/subdir/test1.txt",
1925 "$base/unittest-cont1/e/subdir2/subdir/test2.txt",
1926 "$base/unittest-cont1/e/subdir2/subdir/test3.txt",
1927 "$base/unittest-cont1/e/subdir2/subdir/test4.txt",
1928 "$base/unittest-cont1/e/subdir2/subdir/test5.txt",
1929 "$base/unittest-cont1/e/subdir2/subdir/sub/test0.txt",
1930 "$base/unittest-cont1/e/subdir2/subdir/sub/120-px-file.txt",
1933 // Add the files
1934 $ops = array();
1935 foreach ( $files as $file ) {
1936 $this->prepare( array( 'dir' => dirname( $file ) ) );
1937 $ops[] = array( 'op' => 'create', 'content' => 'xxy', 'dst' => $file );
1939 $status = $this->backend->doQuickOperations( $ops );
1940 $this->assertGoodStatus( $status,
1941 "Creation of files succeeded ($backendName)." );
1942 $this->assertEquals( true, $status->isOK(),
1943 "Creation of files succeeded with OK status ($backendName)." );
1945 // Expected listing at root
1946 $expected = array(
1947 "e/test1.txt",
1948 "e/test2.txt",
1949 "e/test3.txt",
1950 "e/subdir1/test1.txt",
1951 "e/subdir1/test2.txt",
1952 "e/subdir2/test3.txt",
1953 "e/subdir2/test4.txt",
1954 "e/subdir2/subdir/test1.txt",
1955 "e/subdir2/subdir/test2.txt",
1956 "e/subdir2/subdir/test3.txt",
1957 "e/subdir2/subdir/test4.txt",
1958 "e/subdir2/subdir/test5.txt",
1959 "e/subdir2/subdir/sub/test0.txt",
1960 "e/subdir2/subdir/sub/120-px-file.txt",
1962 sort( $expected );
1964 // Actual listing (no trailing slash) at root
1965 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont1" ) );
1966 $list = $this->listToArray( $iter );
1967 sort( $list );
1968 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
1970 // Actual listing (no trailing slash) at root with advise
1971 $iter = $this->backend->getFileList( array(
1972 'dir' => "$base/unittest-cont1",
1973 'adviseStat' => 1
1974 ) );
1975 $list = $this->listToArray( $iter );
1976 sort( $list );
1977 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
1979 // Actual listing (with trailing slash) at root
1980 $list = array();
1981 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont1/" ) );
1982 foreach ( $iter as $file ) {
1983 $list[] = $file;
1985 sort( $list );
1986 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
1988 // Expected listing at subdir
1989 $expected = array(
1990 "test1.txt",
1991 "test2.txt",
1992 "test3.txt",
1993 "test4.txt",
1994 "test5.txt",
1995 "sub/test0.txt",
1996 "sub/120-px-file.txt",
1998 sort( $expected );
2000 // Actual listing (no trailing slash) at subdir
2001 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont1/e/subdir2/subdir" ) );
2002 $list = $this->listToArray( $iter );
2003 sort( $list );
2004 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
2006 // Actual listing (no trailing slash) at subdir with advise
2007 $iter = $this->backend->getFileList( array(
2008 'dir' => "$base/unittest-cont1/e/subdir2/subdir",
2009 'adviseStat' => 1
2010 ) );
2011 $list = $this->listToArray( $iter );
2012 sort( $list );
2013 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
2015 // Actual listing (with trailing slash) at subdir
2016 $list = array();
2017 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont1/e/subdir2/subdir/" ) );
2018 foreach ( $iter as $file ) {
2019 $list[] = $file;
2021 sort( $list );
2022 $this->assertEquals( $expected, $list, "Correct file listing ($backendName)." );
2024 // Actual listing (using iterator second time)
2025 $list = $this->listToArray( $iter );
2026 sort( $list );
2027 $this->assertEquals( $expected, $list, "Correct file listing ($backendName), second iteration." );
2029 // Actual listing (top files only) at root
2030 $iter = $this->backend->getTopFileList( array( 'dir' => "$base/unittest-cont1" ) );
2031 $list = $this->listToArray( $iter );
2032 sort( $list );
2033 $this->assertEquals( array(), $list, "Correct top file listing ($backendName)." );
2035 // Expected listing (top files only) at subdir
2036 $expected = array(
2037 "test1.txt",
2038 "test2.txt",
2039 "test3.txt",
2040 "test4.txt",
2041 "test5.txt"
2043 sort( $expected );
2045 // Actual listing (top files only) at subdir
2046 $iter = $this->backend->getTopFileList(
2047 array( 'dir' => "$base/unittest-cont1/e/subdir2/subdir" )
2049 $list = $this->listToArray( $iter );
2050 sort( $list );
2051 $this->assertEquals( $expected, $list, "Correct top file listing ($backendName)." );
2053 // Actual listing (top files only) at subdir with advise
2054 $iter = $this->backend->getTopFileList( array(
2055 'dir' => "$base/unittest-cont1/e/subdir2/subdir",
2056 'adviseStat' => 1
2057 ) );
2058 $list = $this->listToArray( $iter );
2059 sort( $list );
2060 $this->assertEquals( $expected, $list, "Correct top file listing ($backendName)." );
2062 foreach ( $files as $file ) { // clean up
2063 $this->backend->doOperation( array( 'op' => 'delete', 'src' => $file ) );
2066 $iter = $this->backend->getFileList( array( 'dir' => "$base/unittest-cont1/not/exists" ) );
2067 foreach ( $iter as $iter ) {
2068 // no errors
2073 * @covers FileBackend::getTopDirectoryList
2074 * @covers FileBackend::getDirectoryList
2076 public function testGetDirectoryList() {
2077 $this->backend = $this->singleBackend;
2078 $this->tearDownFiles();
2079 $this->doTestGetDirectoryList();
2080 $this->tearDownFiles();
2082 $this->backend = $this->multiBackend;
2083 $this->tearDownFiles();
2084 $this->doTestGetDirectoryList();
2085 $this->tearDownFiles();
2088 private function doTestGetDirectoryList() {
2089 $backendName = $this->backendClass();
2091 $base = self::baseStorePath();
2092 $files = array(
2093 "$base/unittest-cont1/e/test1.txt",
2094 "$base/unittest-cont1/e/test2.txt",
2095 "$base/unittest-cont1/e/test3.txt",
2096 "$base/unittest-cont1/e/subdir1/test1.txt",
2097 "$base/unittest-cont1/e/subdir1/test2.txt",
2098 "$base/unittest-cont1/e/subdir2/test3.txt",
2099 "$base/unittest-cont1/e/subdir2/test4.txt",
2100 "$base/unittest-cont1/e/subdir2/subdir/test1.txt",
2101 "$base/unittest-cont1/e/subdir3/subdir/test2.txt",
2102 "$base/unittest-cont1/e/subdir4/subdir/test3.txt",
2103 "$base/unittest-cont1/e/subdir4/subdir/test4.txt",
2104 "$base/unittest-cont1/e/subdir4/subdir/test5.txt",
2105 "$base/unittest-cont1/e/subdir4/subdir/sub/test0.txt",
2106 "$base/unittest-cont1/e/subdir4/subdir/sub/120-px-file.txt",
2109 // Add the files
2110 $ops = array();
2111 foreach ( $files as $file ) {
2112 $this->prepare( array( 'dir' => dirname( $file ) ) );
2113 $ops[] = array( 'op' => 'create', 'content' => 'xxy', 'dst' => $file );
2115 $status = $this->backend->doQuickOperations( $ops );
2116 $this->assertGoodStatus( $status,
2117 "Creation of files succeeded ($backendName)." );
2118 $this->assertEquals( true, $status->isOK(),
2119 "Creation of files succeeded with OK status ($backendName)." );
2121 $this->assertEquals( true,
2122 $this->backend->directoryExists( array( 'dir' => "$base/unittest-cont1/e/subdir1" ) ),
2123 "Directory exists in ($backendName)." );
2124 $this->assertEquals( true,
2125 $this->backend->directoryExists( array( 'dir' => "$base/unittest-cont1/e/subdir2/subdir" ) ),
2126 "Directory exists in ($backendName)." );
2127 $this->assertEquals( false,
2128 $this->backend->directoryExists( array( 'dir' => "$base/unittest-cont1/e/subdir2/test1.txt" ) ),
2129 "Directory does not exists in ($backendName)." );
2131 // Expected listing
2132 $expected = array(
2133 "e",
2135 sort( $expected );
2137 // Actual listing (no trailing slash)
2138 $list = array();
2139 $iter = $this->backend->getTopDirectoryList( array( 'dir' => "$base/unittest-cont1" ) );
2140 foreach ( $iter as $file ) {
2141 $list[] = $file;
2143 sort( $list );
2145 $this->assertEquals( $expected, $list, "Correct top dir listing ($backendName)." );
2147 // Expected listing
2148 $expected = array(
2149 "subdir1",
2150 "subdir2",
2151 "subdir3",
2152 "subdir4",
2154 sort( $expected );
2156 // Actual listing (no trailing slash)
2157 $list = array();
2158 $iter = $this->backend->getTopDirectoryList( array( 'dir' => "$base/unittest-cont1/e" ) );
2159 foreach ( $iter as $file ) {
2160 $list[] = $file;
2162 sort( $list );
2164 $this->assertEquals( $expected, $list, "Correct top dir listing ($backendName)." );
2166 // Actual listing (with trailing slash)
2167 $list = array();
2168 $iter = $this->backend->getTopDirectoryList( array( 'dir' => "$base/unittest-cont1/e/" ) );
2169 foreach ( $iter as $file ) {
2170 $list[] = $file;
2172 sort( $list );
2174 $this->assertEquals( $expected, $list, "Correct top dir listing ($backendName)." );
2176 // Expected listing
2177 $expected = array(
2178 "subdir",
2180 sort( $expected );
2182 // Actual listing (no trailing slash)
2183 $list = array();
2184 $iter = $this->backend->getTopDirectoryList( array( 'dir' => "$base/unittest-cont1/e/subdir2" ) );
2185 foreach ( $iter as $file ) {
2186 $list[] = $file;
2188 sort( $list );
2190 $this->assertEquals( $expected, $list, "Correct top dir listing ($backendName)." );
2192 // Actual listing (with trailing slash)
2193 $list = array();
2194 $iter = $this->backend->getTopDirectoryList(
2195 array( 'dir' => "$base/unittest-cont1/e/subdir2/" )
2198 foreach ( $iter as $file ) {
2199 $list[] = $file;
2201 sort( $list );
2203 $this->assertEquals( $expected, $list, "Correct top dir listing ($backendName)." );
2205 // Actual listing (using iterator second time)
2206 $list = array();
2207 foreach ( $iter as $file ) {
2208 $list[] = $file;
2210 sort( $list );
2212 $this->assertEquals(
2213 $expected,
2214 $list,
2215 "Correct top dir listing ($backendName), second iteration."
2218 // Expected listing (recursive)
2219 $expected = array(
2220 "e",
2221 "e/subdir1",
2222 "e/subdir2",
2223 "e/subdir3",
2224 "e/subdir4",
2225 "e/subdir2/subdir",
2226 "e/subdir3/subdir",
2227 "e/subdir4/subdir",
2228 "e/subdir4/subdir/sub",
2230 sort( $expected );
2232 // Actual listing (recursive)
2233 $list = array();
2234 $iter = $this->backend->getDirectoryList( array( 'dir' => "$base/unittest-cont1/" ) );
2235 foreach ( $iter as $file ) {
2236 $list[] = $file;
2238 sort( $list );
2240 $this->assertEquals( $expected, $list, "Correct dir listing ($backendName)." );
2242 // Expected listing (recursive)
2243 $expected = array(
2244 "subdir",
2245 "subdir/sub",
2247 sort( $expected );
2249 // Actual listing (recursive)
2250 $list = array();
2251 $iter = $this->backend->getDirectoryList( array( 'dir' => "$base/unittest-cont1/e/subdir4" ) );
2252 foreach ( $iter as $file ) {
2253 $list[] = $file;
2255 sort( $list );
2257 $this->assertEquals( $expected, $list, "Correct dir listing ($backendName)." );
2259 // Actual listing (recursive, second time)
2260 $list = array();
2261 foreach ( $iter as $file ) {
2262 $list[] = $file;
2264 sort( $list );
2266 $this->assertEquals( $expected, $list, "Correct dir listing ($backendName)." );
2268 $iter = $this->backend->getDirectoryList( array( 'dir' => "$base/unittest-cont1/e/subdir1" ) );
2269 $items = $this->listToArray( $iter );
2270 $this->assertEquals( array(), $items, "Directory listing is empty." );
2272 foreach ( $files as $file ) { // clean up
2273 $this->backend->doOperation( array( 'op' => 'delete', 'src' => $file ) );
2276 $iter = $this->backend->getDirectoryList( array( 'dir' => "$base/unittest-cont1/not/exists" ) );
2277 foreach ( $iter as $file ) {
2278 // no errors
2281 $items = $this->listToArray( $iter );
2282 $this->assertEquals( array(), $items, "Directory listing is empty." );
2284 $iter = $this->backend->getDirectoryList( array( 'dir' => "$base/unittest-cont1/e/not/exists" ) );
2285 $items = $this->listToArray( $iter );
2286 $this->assertEquals( array(), $items, "Directory listing is empty." );
2290 * @covers FileBackend::lockFiles
2291 * @covers FileBackend::unlockFiles
2293 public function testLockCalls() {
2294 $this->backend = $this->singleBackend;
2295 $this->doTestLockCalls();
2298 private function doTestLockCalls() {
2299 $backendName = $this->backendClass();
2301 $paths = array(
2302 "test1.txt",
2303 "test2.txt",
2304 "test3.txt",
2305 "subdir1",
2306 "subdir1", // duplicate
2307 "subdir1/test1.txt",
2308 "subdir1/test2.txt",
2309 "subdir2",
2310 "subdir2", // duplicate
2311 "subdir2/test3.txt",
2312 "subdir2/test4.txt",
2313 "subdir2/subdir",
2314 "subdir2/subdir/test1.txt",
2315 "subdir2/subdir/test2.txt",
2316 "subdir2/subdir/test3.txt",
2317 "subdir2/subdir/test4.txt",
2318 "subdir2/subdir/test5.txt",
2319 "subdir2/subdir/sub",
2320 "subdir2/subdir/sub/test0.txt",
2321 "subdir2/subdir/sub/120-px-file.txt",
2324 for ( $i = 0; $i < 25; $i++ ) {
2325 $status = $this->backend->lockFiles( $paths, LockManager::LOCK_EX );
2326 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2327 "Locking of files succeeded ($backendName) ($i)." );
2328 $this->assertEquals( true, $status->isOK(),
2329 "Locking of files succeeded with OK status ($backendName) ($i)." );
2331 $status = $this->backend->lockFiles( $paths, LockManager::LOCK_SH );
2332 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2333 "Locking of files succeeded ($backendName) ($i)." );
2334 $this->assertEquals( true, $status->isOK(),
2335 "Locking of files succeeded with OK status ($backendName) ($i)." );
2337 $status = $this->backend->unlockFiles( $paths, LockManager::LOCK_SH );
2338 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2339 "Locking of files succeeded ($backendName) ($i)." );
2340 $this->assertEquals( true, $status->isOK(),
2341 "Locking of files succeeded with OK status ($backendName) ($i)." );
2343 $status = $this->backend->unlockFiles( $paths, LockManager::LOCK_EX );
2344 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2345 "Locking of files succeeded ($backendName). ($i)" );
2346 $this->assertEquals( true, $status->isOK(),
2347 "Locking of files succeeded with OK status ($backendName) ($i)." );
2349 # # Flip the acquire/release ordering around ##
2351 $status = $this->backend->lockFiles( $paths, LockManager::LOCK_SH );
2352 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2353 "Locking of files succeeded ($backendName) ($i)." );
2354 $this->assertEquals( true, $status->isOK(),
2355 "Locking of files succeeded with OK status ($backendName) ($i)." );
2357 $status = $this->backend->lockFiles( $paths, LockManager::LOCK_EX );
2358 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2359 "Locking of files succeeded ($backendName) ($i)." );
2360 $this->assertEquals( true, $status->isOK(),
2361 "Locking of files succeeded with OK status ($backendName) ($i)." );
2363 $status = $this->backend->unlockFiles( $paths, LockManager::LOCK_EX );
2364 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2365 "Locking of files succeeded ($backendName). ($i)" );
2366 $this->assertEquals( true, $status->isOK(),
2367 "Locking of files succeeded with OK status ($backendName) ($i)." );
2369 $status = $this->backend->unlockFiles( $paths, LockManager::LOCK_SH );
2370 $this->assertEquals( print_r( array(), true ), print_r( $status->errors, true ),
2371 "Locking of files succeeded ($backendName) ($i)." );
2372 $this->assertEquals( true, $status->isOK(),
2373 "Locking of files succeeded with OK status ($backendName) ($i)." );
2376 $status = Status::newGood();
2377 $sl = $this->backend->getScopedFileLocks( $paths, LockManager::LOCK_EX, $status );
2378 $this->assertInstanceOf( 'ScopedLock', $sl,
2379 "Scoped locking of files succeeded ($backendName)." );
2380 $this->assertEquals( array(), $status->errors,
2381 "Scoped locking of files succeeded ($backendName)." );
2382 $this->assertEquals( true, $status->isOK(),
2383 "Scoped locking of files succeeded with OK status ($backendName)." );
2385 ScopedLock::release( $sl );
2386 $this->assertEquals( null, $sl,
2387 "Scoped unlocking of files succeeded ($backendName)." );
2388 $this->assertEquals( array(), $status->errors,
2389 "Scoped unlocking of files succeeded ($backendName)." );
2390 $this->assertEquals( true, $status->isOK(),
2391 "Scoped unlocking of files succeeded with OK status ($backendName)." );
2395 * @dataProvider provider_testGetContentType
2397 public function testGetContentType( $mimeCallback, $mimeFromString ) {
2398 global $IP;
2400 $be = TestingAccessWrapper::newFromObject( new MemoryFileBackend(
2401 array(
2402 'name' => 'testing',
2403 'class' => 'MemoryFileBackend',
2404 'wikiId' => 'meow',
2405 'mimeCallback' => $mimeCallback
2407 ) );
2409 $dst = 'mwstore://testing/container/path/to/file_no_ext';
2410 $src = "$IP/tests/phpunit/data/media/srgb.jpg";
2411 $this->assertEquals( 'image/jpeg', $be->getContentType( $dst, null, $src ) );
2412 $this->assertEquals(
2413 $mimeFromString ? 'image/jpeg' : 'unknown/unknown',
2414 $be->getContentType( $dst, file_get_contents( $src ), null ) );
2416 $src = "$IP/tests/phpunit/data/media/Png-native-test.png";
2417 $this->assertEquals( 'image/png', $be->getContentType( $dst, null, $src ) );
2418 $this->assertEquals(
2419 $mimeFromString ? 'image/png' : 'unknown/unknown',
2420 $be->getContentType( $dst, file_get_contents( $src ), null ) );
2423 public static function provider_testGetContentType() {
2424 return array(
2425 array( null, false ),
2426 array( array( FileBackendGroup::singleton(), 'guessMimeInternal' ), true )
2430 public function testReadAffinity() {
2431 $be = TestingAccessWrapper::newFromObject(
2432 new FileBackendMultiWrite( array(
2433 'name' => 'localtesting',
2434 'wikiId' => wfWikiId() . mt_rand(),
2435 'backends' => array(
2436 array( // backend 0
2437 'name' => 'multitesting0',
2438 'class' => 'MemoryFileBackend',
2439 'isMultiMaster' => false,
2440 'readAffinity' => true
2442 array( // backend 1
2443 'name' => 'multitesting1',
2444 'class' => 'MemoryFileBackend',
2445 'isMultiMaster' => true
2451 $this->assertEquals(
2453 $be->getReadIndexFromParams( array( 'latest' => 1 ) ),
2454 'Reads with "latest" flag use backend 1'
2456 $this->assertEquals(
2458 $be->getReadIndexFromParams( array( 'latest' => 0 ) ),
2459 'Reads without "latest" flag use backend 0'
2462 $p = 'container/test-cont/file.txt';
2463 $be->backends[0]->quickCreate( array(
2464 'dst' => "mwstore://multitesting0/$p", 'content' => 'cattitude' ) );
2465 $be->backends[1]->quickCreate( array(
2466 'dst' => "mwstore://multitesting1/$p", 'content' => 'princess of power' ) );
2468 $this->assertEquals(
2469 'cattitude',
2470 $be->getFileContents( array( 'src' => "mwstore://localtesting/$p" ) ),
2471 "Non-latest read came from backend 0"
2473 $this->assertEquals(
2474 'princess of power',
2475 $be->getFileContents( array( 'src' => "mwstore://localtesting/$p", 'latest' => 1 ) ),
2476 "Latest read came from backend1"
2480 public function testAsyncWrites() {
2481 $be = TestingAccessWrapper::newFromObject(
2482 new FileBackendMultiWrite( array(
2483 'name' => 'localtesting',
2484 'wikiId' => wfWikiId() . mt_rand(),
2485 'backends' => array(
2486 array( // backend 0
2487 'name' => 'multitesting0',
2488 'class' => 'MemoryFileBackend',
2489 'isMultiMaster' => false
2491 array( // backend 1
2492 'name' => 'multitesting1',
2493 'class' => 'MemoryFileBackend',
2494 'isMultiMaster' => true
2497 'replication' => 'async'
2501 $this->setMwGlobals( 'wgCommandLineMode', false );
2503 $p = 'container/test-cont/file.txt';
2504 $be->quickCreate( array(
2505 'dst' => "mwstore://localtesting/$p", 'content' => 'cattitude' ) );
2507 $this->assertEquals(
2508 false,
2509 $be->backends[0]->getFileContents( array( 'src' => "mwstore://multitesting0/$p" ) ),
2510 "File not yet written to backend 0"
2512 $this->assertEquals(
2513 'cattitude',
2514 $be->backends[1]->getFileContents( array( 'src' => "mwstore://multitesting1/$p" ) ),
2515 "File already written to backend 1"
2518 DeferredUpdates::doUpdates();
2520 $this->assertEquals(
2521 'cattitude',
2522 $be->backends[0]->getFileContents( array( 'src' => "mwstore://multitesting0/$p" ) ),
2523 "File now written to backend 0"
2527 public function testSanitizeOpHeaders() {
2528 $be = TestingAccessWrapper::newFromObject( new MemoryFileBackend( array(
2529 'name' => 'localtesting',
2530 'wikiId' => wfWikiID()
2531 ) ) );
2533 $name = wfRandomString( 300 );
2535 $input = array(
2536 'headers' => array(
2537 'content-Disposition' => FileBackend::makeContentDisposition( 'inline', $name ),
2538 'Content-dUration' => 25.6,
2539 'X-LONG-VALUE' => str_pad( '0', 300 ),
2540 'CONTENT-LENGTH' => 855055,
2543 $expected = array(
2544 'headers' => array(
2545 'content-disposition' => FileBackend::makeContentDisposition( 'inline', $name ),
2546 'content-duration' => 25.6,
2547 'content-length' => 855055
2551 MediaWiki\suppressWarnings();
2552 $actual = $be->sanitizeOpHeaders( $input );
2553 MediaWiki\restoreWarnings();
2555 $this->assertEquals( $expected, $actual, "Header sanitized properly" );
2558 // helper function
2559 private function listToArray( $iter ) {
2560 return is_array( $iter ) ? $iter : iterator_to_array( $iter );
2563 // test helper wrapper for backend prepare() function
2564 private function prepare( array $params ) {
2565 return $this->backend->prepare( $params );
2568 // test helper wrapper for backend prepare() function
2569 private function create( array $params ) {
2570 $params['op'] = 'create';
2572 return $this->backend->doQuickOperations( array( $params ) );
2575 function tearDownFiles() {
2576 $containers = array( 'unittest-cont1', 'unittest-cont2', 'unittest-cont-bad' );
2577 foreach ( $containers as $container ) {
2578 $this->deleteFiles( $container );
2582 private function deleteFiles( $container ) {
2583 $base = self::baseStorePath();
2584 $iter = $this->backend->getFileList( array( 'dir' => "$base/$container" ) );
2585 if ( $iter ) {
2586 foreach ( $iter as $file ) {
2587 $this->backend->quickDelete( array( 'src' => "$base/$container/$file" ) );
2589 // free the directory, to avoid Permission denied under windows on rmdir
2590 unset( $iter );
2592 $this->backend->clean( array( 'dir' => "$base/$container", 'recursive' => 1 ) );
2595 function assertBackendPathsConsistent( array $paths ) {
2596 if ( $this->backend instanceof FileBackendMultiWrite ) {
2597 $status = $this->backend->consistencyCheck( $paths );
2598 $this->assertGoodStatus( $status, "Files synced: " . implode( ',', $paths ) );
2602 function assertGoodStatus( $status, $msg ) {
2603 $this->assertEquals( print_r( array(), 1 ), print_r( $status->errors, 1 ), $msg );