2012-10-03 09:24:49 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Copyright (c) 2012 Robin Appelman <icewind@owncloud.com>
|
|
|
|
* This file is licensed under the Affero General Public License version 3 or
|
|
|
|
* later.
|
|
|
|
* See the COPYING-README file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Test\Files\Cache;
|
|
|
|
|
2014-11-07 14:23:15 +00:00
|
|
|
class Scanner extends \Test\TestCase {
|
2012-10-03 09:24:49 +00:00
|
|
|
/**
|
|
|
|
* @var \OC\Files\Storage\Storage $storage
|
|
|
|
*/
|
|
|
|
private $storage;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \OC\Files\Cache\Scanner $scanner
|
|
|
|
*/
|
|
|
|
private $scanner;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \OC\Files\Cache\Cache $cache
|
|
|
|
*/
|
|
|
|
private $cache;
|
|
|
|
|
2014-11-07 14:23:15 +00:00
|
|
|
protected function setUp() {
|
|
|
|
parent::setUp();
|
|
|
|
|
2013-09-23 10:45:02 +00:00
|
|
|
$this->storage = new \OC\Files\Storage\Temporary(array());
|
|
|
|
$this->scanner = new \OC\Files\Cache\Scanner($this->storage);
|
|
|
|
$this->cache = new \OC\Files\Cache\Cache($this->storage);
|
|
|
|
}
|
|
|
|
|
2014-11-07 14:23:15 +00:00
|
|
|
protected function tearDown() {
|
2013-09-23 10:45:02 +00:00
|
|
|
if ($this->cache) {
|
|
|
|
$this->cache->clear();
|
|
|
|
}
|
2014-11-07 14:23:15 +00:00
|
|
|
|
|
|
|
parent::tearDown();
|
2013-09-23 10:45:02 +00:00
|
|
|
}
|
|
|
|
|
2012-10-03 09:24:49 +00:00
|
|
|
function testFile() {
|
|
|
|
$data = "dummy file data\n";
|
|
|
|
$this->storage->file_put_contents('foo.txt', $data);
|
|
|
|
$this->scanner->scanFile('foo.txt');
|
|
|
|
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($this->cache->inCache('foo.txt'), true);
|
2012-10-03 09:24:49 +00:00
|
|
|
$cachedData = $this->cache->get('foo.txt');
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($cachedData['size'], strlen($data));
|
|
|
|
$this->assertEquals($cachedData['mimetype'], 'text/plain');
|
|
|
|
$this->assertNotEquals($cachedData['parent'], -1); //parent folders should be scanned automatically
|
2012-10-03 09:40:09 +00:00
|
|
|
|
|
|
|
$data = file_get_contents(\OC::$SERVERROOT . '/core/img/logo.png');
|
|
|
|
$this->storage->file_put_contents('foo.png', $data);
|
|
|
|
$this->scanner->scanFile('foo.png');
|
|
|
|
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($this->cache->inCache('foo.png'), true);
|
2012-10-03 09:40:09 +00:00
|
|
|
$cachedData = $this->cache->get('foo.png');
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($cachedData['size'], strlen($data));
|
|
|
|
$this->assertEquals($cachedData['mimetype'], 'image/png');
|
2012-10-03 09:40:09 +00:00
|
|
|
}
|
|
|
|
|
2012-10-03 11:07:19 +00:00
|
|
|
private function fillTestFolders() {
|
2012-10-03 09:40:09 +00:00
|
|
|
$textData = "dummy file data\n";
|
|
|
|
$imgData = file_get_contents(\OC::$SERVERROOT . '/core/img/logo.png');
|
2012-10-03 11:07:19 +00:00
|
|
|
$this->storage->mkdir('folder');
|
2012-10-03 09:40:09 +00:00
|
|
|
$this->storage->file_put_contents('foo.txt', $textData);
|
|
|
|
$this->storage->file_put_contents('foo.png', $imgData);
|
2012-10-03 11:07:19 +00:00
|
|
|
$this->storage->file_put_contents('folder/bar.txt', $textData);
|
|
|
|
}
|
|
|
|
|
|
|
|
function testFolder() {
|
|
|
|
$this->fillTestFolders();
|
2012-10-03 09:40:09 +00:00
|
|
|
|
|
|
|
$this->scanner->scan('');
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($this->cache->inCache(''), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('foo.txt'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('foo.png'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('folder'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('folder/bar.txt'), true);
|
2012-10-03 11:07:19 +00:00
|
|
|
|
|
|
|
$cachedDataText = $this->cache->get('foo.txt');
|
|
|
|
$cachedDataText2 = $this->cache->get('foo.txt');
|
|
|
|
$cachedDataImage = $this->cache->get('foo.png');
|
|
|
|
$cachedDataFolder = $this->cache->get('');
|
|
|
|
$cachedDataFolder2 = $this->cache->get('folder');
|
|
|
|
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($cachedDataImage['parent'], $cachedDataText['parent']);
|
|
|
|
$this->assertEquals($cachedDataFolder['fileid'], $cachedDataImage['parent']);
|
|
|
|
$this->assertEquals($cachedDataFolder['size'], $cachedDataImage['size'] + $cachedDataText['size'] + $cachedDataText2['size']);
|
|
|
|
$this->assertEquals($cachedDataFolder2['size'], $cachedDataText2['size']);
|
2012-10-03 11:07:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function testShallow() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals($this->cache->inCache(''), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('foo.txt'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('foo.png'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('folder'), true);
|
|
|
|
$this->assertEquals($this->cache->inCache('folder/bar.txt'), false);
|
2012-10-03 11:07:19 +00:00
|
|
|
|
|
|
|
$cachedDataFolder = $this->cache->get('');
|
|
|
|
$cachedDataFolder2 = $this->cache->get('folder');
|
|
|
|
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertEquals(-1, $cachedDataFolder['size']);
|
|
|
|
$this->assertEquals(-1, $cachedDataFolder2['size']);
|
2012-10-03 16:19:45 +00:00
|
|
|
|
|
|
|
$this->scanner->scan('folder', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
|
|
|
|
|
|
|
$cachedDataFolder2 = $this->cache->get('folder');
|
|
|
|
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertNotEquals($cachedDataFolder2['size'], -1);
|
2012-10-27 16:05:40 +00:00
|
|
|
|
2012-11-08 17:10:54 +00:00
|
|
|
$this->cache->correctFolderSize('folder');
|
2012-10-27 16:05:40 +00:00
|
|
|
|
|
|
|
$cachedDataFolder = $this->cache->get('');
|
2013-01-26 20:41:14 +00:00
|
|
|
$this->assertNotEquals($cachedDataFolder['size'], -1);
|
2012-10-03 09:24:49 +00:00
|
|
|
}
|
|
|
|
|
2013-06-17 16:03:57 +00:00
|
|
|
function testBackgroundScan() {
|
2012-11-21 22:18:58 +00:00
|
|
|
$this->fillTestFolders();
|
|
|
|
$this->storage->mkdir('folder2');
|
|
|
|
$this->storage->file_put_contents('folder2/bar.txt', 'foobar');
|
|
|
|
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
|
|
|
$this->assertFalse($this->cache->inCache('folder/bar.txt'));
|
|
|
|
$this->assertFalse($this->cache->inCache('folder/2bar.txt'));
|
|
|
|
$cachedData = $this->cache->get('');
|
|
|
|
$this->assertEquals(-1, $cachedData['size']);
|
|
|
|
|
|
|
|
$this->scanner->backgroundScan();
|
|
|
|
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
|
|
|
|
|
|
|
$cachedData = $this->cache->get('');
|
|
|
|
$this->assertnotEquals(-1, $cachedData['size']);
|
|
|
|
|
|
|
|
$this->assertFalse($this->cache->getIncomplete());
|
|
|
|
}
|
|
|
|
|
2013-06-17 16:03:57 +00:00
|
|
|
public function testReuseExisting() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$oldData = $this->cache->get('');
|
|
|
|
$this->storage->unlink('folder/bar.txt');
|
2014-01-14 19:19:05 +00:00
|
|
|
$this->cache->put('folder', array('mtime' => $this->storage->filemtime('folder'), 'storage_mtime' => $this->storage->filemtime('folder')));
|
2013-06-17 16:03:57 +00:00
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW, \OC\Files\Cache\Scanner::REUSE_SIZE);
|
|
|
|
$newData = $this->cache->get('');
|
2014-02-27 08:51:26 +00:00
|
|
|
$this->assertInternalType('string', $oldData['etag']);
|
|
|
|
$this->assertInternalType('string', $newData['etag']);
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertNotSame($oldData['etag'], $newData['etag']);
|
2013-06-17 16:03:57 +00:00
|
|
|
$this->assertEquals($oldData['size'], $newData['size']);
|
|
|
|
|
|
|
|
$oldData = $newData;
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW, \OC\Files\Cache\Scanner::REUSE_ETAG);
|
|
|
|
$newData = $this->cache->get('');
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertSame($oldData['etag'], $newData['etag']);
|
2013-06-17 16:03:57 +00:00
|
|
|
$this->assertEquals(-1, $newData['size']);
|
2013-08-12 13:37:15 +00:00
|
|
|
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_RECURSIVE);
|
|
|
|
$oldData = $this->cache->get('');
|
|
|
|
$this->assertNotEquals(-1, $oldData['size']);
|
|
|
|
$this->scanner->scanFile('', \OC\Files\Cache\Scanner::REUSE_ETAG + \OC\Files\Cache\Scanner::REUSE_SIZE);
|
|
|
|
$newData = $this->cache->get('');
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertSame($oldData['etag'], $newData['etag']);
|
2013-08-12 13:37:15 +00:00
|
|
|
$this->assertEquals($oldData['size'], $newData['size']);
|
|
|
|
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_RECURSIVE, \OC\Files\Cache\Scanner::REUSE_ETAG + \OC\Files\Cache\Scanner::REUSE_SIZE);
|
|
|
|
$newData = $this->cache->get('');
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertSame($oldData['etag'], $newData['etag']);
|
2013-08-12 13:37:15 +00:00
|
|
|
$this->assertEquals($oldData['size'], $newData['size']);
|
|
|
|
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW, \OC\Files\Cache\Scanner::REUSE_ETAG + \OC\Files\Cache\Scanner::REUSE_SIZE);
|
|
|
|
$newData = $this->cache->get('');
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertSame($oldData['etag'], $newData['etag']);
|
2013-08-12 13:37:15 +00:00
|
|
|
$this->assertEquals($oldData['size'], $newData['size']);
|
2013-06-17 16:03:57 +00:00
|
|
|
}
|
|
|
|
|
2013-06-19 22:42:34 +00:00
|
|
|
public function testRemovedFile() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$this->assertTrue($this->cache->inCache('foo.txt'));
|
|
|
|
$this->storage->unlink('foo.txt');
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
|
|
|
$this->assertFalse($this->cache->inCache('foo.txt'));
|
|
|
|
}
|
|
|
|
|
2013-06-20 08:47:37 +00:00
|
|
|
public function testRemovedFolder() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
2013-07-01 16:11:33 +00:00
|
|
|
$this->storage->rmdir('/folder');
|
2013-06-20 08:47:37 +00:00
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
|
|
|
$this->assertFalse($this->cache->inCache('folder'));
|
|
|
|
$this->assertFalse($this->cache->inCache('folder/bar.txt'));
|
|
|
|
}
|
|
|
|
|
2014-06-10 13:26:18 +00:00
|
|
|
public function testScanRemovedFile() {
|
2013-08-06 13:59:06 +00:00
|
|
|
$this->fillTestFolders();
|
|
|
|
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
|
|
|
$this->storage->unlink('folder/bar.txt');
|
|
|
|
$this->scanner->scanFile('folder/bar.txt');
|
|
|
|
$this->assertFalse($this->cache->inCache('folder/bar.txt'));
|
|
|
|
}
|
|
|
|
|
2013-09-16 10:09:15 +00:00
|
|
|
public function testETagRecreation() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
|
2013-09-16 21:32:17 +00:00
|
|
|
$this->scanner->scan('folder/bar.txt');
|
2013-09-16 10:09:15 +00:00
|
|
|
|
|
|
|
// manipulate etag to simulate an empty etag
|
|
|
|
$this->scanner->scan('', \OC\Files\Cache\Scanner::SCAN_SHALLOW, \OC\Files\Cache\Scanner::REUSE_ETAG);
|
2013-09-16 21:32:17 +00:00
|
|
|
$data0 = $this->cache->get('folder/bar.txt');
|
2014-02-27 08:51:26 +00:00
|
|
|
$this->assertInternalType('string', $data0['etag']);
|
2013-09-16 21:32:17 +00:00
|
|
|
$data1 = $this->cache->get('folder');
|
2014-02-27 08:51:26 +00:00
|
|
|
$this->assertInternalType('string', $data1['etag']);
|
2013-09-16 21:32:17 +00:00
|
|
|
$data2 = $this->cache->get('');
|
2014-02-27 08:51:26 +00:00
|
|
|
$this->assertInternalType('string', $data2['etag']);
|
2013-09-16 21:32:17 +00:00
|
|
|
$data0['etag'] = '';
|
|
|
|
$this->cache->put('folder/bar.txt', $data0);
|
2013-09-16 10:09:15 +00:00
|
|
|
|
|
|
|
// rescan
|
2013-09-16 21:32:17 +00:00
|
|
|
$this->scanner->scan('folder/bar.txt', \OC\Files\Cache\Scanner::SCAN_SHALLOW, \OC\Files\Cache\Scanner::REUSE_ETAG);
|
|
|
|
|
|
|
|
// verify cache content
|
|
|
|
$newData0 = $this->cache->get('folder/bar.txt');
|
2014-02-27 08:51:26 +00:00
|
|
|
$this->assertInternalType('string', $newData0['etag']);
|
2014-02-27 08:39:34 +00:00
|
|
|
$this->assertNotEmpty($newData0['etag']);
|
2013-09-16 10:09:15 +00:00
|
|
|
}
|
2014-06-10 13:26:18 +00:00
|
|
|
|
|
|
|
public function testRepairParent() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
|
|
|
$oldFolderId = $this->cache->getId('folder');
|
|
|
|
|
|
|
|
// delete the folder without removing the childs
|
|
|
|
$sql = 'DELETE FROM `*PREFIX*filecache` WHERE `fileid` = ?';
|
|
|
|
\OC_DB::executeAudited($sql, array($oldFolderId));
|
|
|
|
|
|
|
|
$cachedData = $this->cache->get('folder/bar.txt');
|
|
|
|
$this->assertEquals($oldFolderId, $cachedData['parent']);
|
|
|
|
$this->assertFalse($this->cache->inCache('folder'));
|
|
|
|
|
|
|
|
$this->scanner->scan('');
|
|
|
|
|
|
|
|
$this->assertTrue($this->cache->inCache('folder'));
|
|
|
|
$newFolderId = $this->cache->getId('folder');
|
|
|
|
$this->assertNotEquals($oldFolderId, $newFolderId);
|
|
|
|
|
|
|
|
$cachedData = $this->cache->get('folder/bar.txt');
|
|
|
|
$this->assertEquals($newFolderId, $cachedData['parent']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testRepairParentShallow() {
|
|
|
|
$this->fillTestFolders();
|
|
|
|
$this->scanner->scan('');
|
|
|
|
$this->assertTrue($this->cache->inCache('folder/bar.txt'));
|
|
|
|
$oldFolderId = $this->cache->getId('folder');
|
|
|
|
|
|
|
|
// delete the folder without removing the childs
|
|
|
|
$sql = 'DELETE FROM `*PREFIX*filecache` WHERE `fileid` = ?';
|
|
|
|
\OC_DB::executeAudited($sql, array($oldFolderId));
|
|
|
|
|
|
|
|
$cachedData = $this->cache->get('folder/bar.txt');
|
|
|
|
$this->assertEquals($oldFolderId, $cachedData['parent']);
|
|
|
|
$this->assertFalse($this->cache->inCache('folder'));
|
|
|
|
|
|
|
|
$this->scanner->scan('folder', \OC\Files\Cache\Scanner::SCAN_SHALLOW);
|
|
|
|
|
|
|
|
$this->assertTrue($this->cache->inCache('folder'));
|
|
|
|
$newFolderId = $this->cache->getId('folder');
|
|
|
|
$this->assertNotEquals($oldFolderId, $newFolderId);
|
|
|
|
|
|
|
|
$cachedData = $this->cache->get('folder/bar.txt');
|
|
|
|
$this->assertEquals($newFolderId, $cachedData['parent']);
|
|
|
|
}
|
2015-10-12 11:59:16 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider dataTestIsPartialFile
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
* @param bool $expected
|
|
|
|
*/
|
|
|
|
public function testIsPartialFile($path, $expected) {
|
|
|
|
$this->assertSame($expected,
|
|
|
|
$this->scanner->isPartialFile($path)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function dataTestIsPartialFile() {
|
|
|
|
return [
|
|
|
|
['foo.txt.part', true],
|
|
|
|
['/sub/folder/foo.txt.part', true],
|
|
|
|
['/sub/folder.part/foo.txt', true],
|
|
|
|
['foo.txt', false],
|
|
|
|
['/sub/folder/foo.txt', false],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2012-10-03 09:24:49 +00:00
|
|
|
}
|