nextcloud/tests/lib/repair/repairlegacystorage.php

343 lines
9.4 KiB
PHP
Raw Normal View History

2014-03-25 15:52:32 +04:00
<?php
/**
* Copyright (c) 2014 Vincent Petry <pvince81@owncloud.com>
* This file is licensed under the Affero General Public License version 3 or
* later.
* See the COPYING-README file.
*/
2015-03-27 01:19:27 +03:00
namespace Test\Repair;
2014-03-25 15:52:32 +04:00
use OC\Files\Cache\Cache;
use OC\Files\Cache\Storage;
use Test\TestCase;
2014-03-25 15:52:32 +04:00
/**
* Tests for the converting of legacy storages to home storages.
*
2015-11-03 03:52:41 +03:00
* @group DB
*
2014-03-25 15:52:32 +04:00
* @see \OC\Repair\RepairLegacyStorages
*/
class RepairLegacyStorages extends TestCase {
/** @var \OCP\IDBConnection */
private $connection;
/** @var \OCP\IConfig */
private $config;
2014-03-25 15:52:32 +04:00
private $user;
/** @var \OC\Repair\RepairLegacyStorages */
2014-03-25 15:52:32 +04:00
private $repair;
private $dataDir;
private $oldDataDir;
private $legacyStorageId;
private $newStorageId;
private $warnings;
protected function setUp() {
parent::setUp();
2014-03-25 15:52:32 +04:00
$this->config = \OC::$server->getConfig();
$this->connection = \OC::$server->getDatabaseConnection();
2014-03-25 15:52:32 +04:00
$this->oldDataDir = $this->config->getSystemValue('datadirectory', \OC::$SERVERROOT . '/data/');
$this->repair = new \OC\Repair\RepairLegacyStorages($this->config, $this->connection);
$this->warnings = [];
$this->repair->listen('\OC\Repair', 'warning', function ($description){
$this->warnings[] = $description;
});
2014-03-25 15:52:32 +04:00
}
protected function tearDown() {
$user = \OC::$server->getUserManager()->get($this->user);
if ($user) {
$user->delete();
}
2014-03-25 15:52:32 +04:00
$sql = 'DELETE FROM `*PREFIX*storages`';
$this->connection->executeQuery($sql);
$sql = 'DELETE FROM `*PREFIX*filecache`';
$this->connection->executeQuery($sql);
$this->config->setSystemValue('datadirectory', $this->oldDataDir);
2014-03-25 15:52:32 +04:00
$this->config->setAppValue('core', 'repairlegacystoragesdone', 'no');
parent::tearDown();
2014-03-25 15:52:32 +04:00
}
/**
* @param string $dataDir
* @param string $userId
* @throws \Exception
*/
2014-03-25 15:52:32 +04:00
function prepareSettings($dataDir, $userId) {
// hard-coded string as we want a predictable fixed length
// no data will be written there
$this->dataDir = $dataDir;
$this->config->setSystemValue('datadirectory', $this->dataDir);
2014-03-25 15:52:32 +04:00
$this->user = $userId;
$this->legacyStorageId = 'local::' . $this->dataDir . $this->user . '/';
$this->newStorageId = 'home::' . $this->user;
\OC::$server->getUserManager()->createUser($this->user, $this->user);
2014-03-25 15:52:32 +04:00
}
/**
* Create a storage entry
*
* @param string $storageId
* @return int
2014-03-25 15:52:32 +04:00
*/
private function createStorage($storageId) {
$sql = 'INSERT INTO `*PREFIX*storages` (`id`)'
. ' VALUES (?)';
$storageId = Storage::adjustStorageId($storageId);
2014-03-25 15:52:32 +04:00
$numRows = $this->connection->executeUpdate($sql, array($storageId));
$this->assertEquals(1, $numRows);
2016-01-07 12:22:30 +03:00
return \OC::$server->getDatabaseConnection()->lastInsertId('*PREFIX*storages');
2014-03-25 15:52:32 +04:00
}
/**
* Returns the storage id based on the numeric id
*
* @param int $storageId numeric id of the storage
2014-03-25 15:52:32 +04:00
* @return string storage id or null if not found
*/
private function getStorageId($storageId) {
$numericId = Storage::getNumericStorageId($storageId);
2014-03-25 15:52:32 +04:00
if (!is_null($numericId)) {
return (int)$numericId;
}
return null;
}
/**
* Create dummy data in the filecache for the given storage numeric id
*
* @param string $storageId storage id
*/
private function createData($storageId) {
$cache = new Cache($storageId);
2014-03-25 15:52:32 +04:00
$cache->put(
'dummyfile.txt',
array('size' => 5, 'mtime' => 12, 'mimetype' => 'text/plain')
);
}
/**
* Test that existing home storages are left alone when valid.
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testNoopWithExistingHomeStorage($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$newStorageNumId = $this->createStorage($this->newStorageId);
$this->repair->run();
$this->assertNull($this->getStorageId($this->legacyStorageId));
$this->assertEquals($newStorageNumId, $this->getStorageId($this->newStorageId));
}
/**
* Test that legacy storages are converted to home storages when
* the latter does not exist.
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testConvertLegacyToHomeStorage($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$legacyStorageNumId = $this->createStorage($this->legacyStorageId);
$this->repair->run();
$this->assertNull($this->getStorageId($this->legacyStorageId));
$this->assertEquals($legacyStorageNumId, $this->getStorageId($this->newStorageId));
}
/**
* Test that legacy storages are converted to home storages
* when home storage already exists but has no data.
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testConvertLegacyToExistingEmptyHomeStorage($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$legacyStorageNumId = $this->createStorage($this->legacyStorageId);
$this->createStorage($this->newStorageId);
2014-03-25 15:52:32 +04:00
$this->createData($this->legacyStorageId);
$this->repair->run();
$this->assertNull($this->getStorageId($this->legacyStorageId));
$this->assertEquals($legacyStorageNumId, $this->getStorageId($this->newStorageId));
}
/**
* Test that legacy storages are converted to home storages
* when home storage already exists and the legacy storage
* has no data.
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testConvertEmptyLegacyToHomeStorage($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$this->createStorage($this->legacyStorageId);
2014-03-25 15:52:32 +04:00
$newStorageNumId = $this->createStorage($this->newStorageId);
$this->createData($this->newStorageId);
$this->repair->run();
$this->assertNull($this->getStorageId($this->legacyStorageId));
$this->assertEquals($newStorageNumId, $this->getStorageId($this->newStorageId));
}
/**
* Test that nothing is done when both conflicting legacy
* and home storage have data.
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testConflictNoop($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$legacyStorageNumId = $this->createStorage($this->legacyStorageId);
$newStorageNumId = $this->createStorage($this->newStorageId);
$this->createData($this->legacyStorageId);
$this->createData($this->newStorageId);
$this->repair->run();
2014-03-25 15:52:32 +04:00
$this->assertEquals(2, count($this->warnings));
$this->assertEquals('Could not repair legacy storage ', substr(current($this->warnings), 0, 32));
2014-03-25 15:52:32 +04:00
// storages left alone
$this->assertEquals($legacyStorageNumId, $this->getStorageId($this->legacyStorageId));
$this->assertEquals($newStorageNumId, $this->getStorageId($this->newStorageId));
// do not set the done flag
$this->assertNotEquals('yes', $this->config->getAppValue('core', 'repairlegacystoragesdone'));
2014-03-25 15:52:32 +04:00
}
/**
* Test that the data dir local entry is left alone
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testDataDirEntryNoop($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$storageId = 'local::' . $this->dataDir;
$numId = $this->createStorage($storageId);
$this->repair->run();
$this->assertEquals($numId, $this->getStorageId($storageId));
}
/**
* Test that external local storages are left alone
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testLocalExtStorageNoop($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$storageId = 'local::/tmp/somedir/' . $this->user;
$numId = $this->createStorage($storageId);
$this->repair->run();
$this->assertEquals($numId, $this->getStorageId($storageId));
}
/**
* Test that other external storages are left alone
*
2014-03-25 15:52:32 +04:00
* @dataProvider settingsProvider
*
* @param string $dataDir
* @param string $userId
2014-03-25 15:52:32 +04:00
*/
public function testExtStorageNoop($dataDir, $userId) {
$this->prepareSettings($dataDir, $userId);
$storageId = 'smb::user@password/tmp/somedir/' . $this->user;
$numId = $this->createStorage($storageId);
$this->repair->run();
$this->assertEquals($numId, $this->getStorageId($storageId));
}
/**
* Provides data dir and user name
*/
function settingsProvider() {
return array(
// regular data dir
array(
'/tmp/oc-autotest/datadir/',
$this->getUniqueID('user_'),
2014-03-25 15:52:32 +04:00
),
// long datadir / short user
array(
'/tmp/oc-autotest/datadir01234567890123456789012345678901234567890123456789END/',
$this->getUniqueID('user_'),
2014-03-25 15:52:32 +04:00
),
// short datadir / long user
array(
'/tmp/oc-autotest/datadir/',
'u123456789012345678901234567890123456789012345678901234567890END', // 64 chars
),
);
}
/**
* Only run the repair once
*/
public function testOnlyRunOnce() {
$output = array();
$this->repair->listen('\OC\Repair', 'info', function ($description) use (&$output) {
$output[] = 'info: ' . $description;
});
$this->prepareSettings('/tmp/oc-autotest/datadir', $this->getUniqueID('user_'));
2014-03-25 15:52:32 +04:00
$this->assertNotEquals('yes', $this->config->getAppValue('core', 'repairlegacystoragesdone'));
$this->repair->run();
$this->assertEquals(1, count($output));
$this->assertEquals('yes', $this->config->getAppValue('core', 'repairlegacystoragesdone'));
$output = array();
$this->repair->run();
// no output which means it did not run
$this->assertEquals(0, count($output));
$this->assertEquals('yes', $this->config->getAppValue('core', 'repairlegacystoragesdone'));
}
}