Merge pull request #12883 from nextcloud/appdata-performance

try to grab the appdata folder directly without going trough the whole tree
This commit is contained in:
Joas Schilling 2018-12-17 15:11:28 +01:00 committed by GitHub
commit 6788e6e75c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 82 additions and 45 deletions

View File

@ -24,6 +24,7 @@ declare(strict_types=1);
namespace OC\Files\AppData;
use OC\Cache\CappedMemoryCache;
use OC\Files\SimpleFS\SimpleFolder;
use OCP\Files\IAppData;
use OCP\Files\IRootFolder;
@ -48,6 +49,9 @@ class AppData implements IAppData {
/** @var Folder */
private $folder;
/** @var (ISimpleFolder|NotFoundException)[]|CappedMemoryCache */
private $folders;
/**
* AppData constructor.
*
@ -62,6 +66,32 @@ class AppData implements IAppData {
$this->rootFolder = $rootFolder;
$this->config = $systemConfig;
$this->appId = $appId;
$this->folders = new CappedMemoryCache();
}
private function getAppDataFolderName() {
$instanceId = $this->config->getValue('instanceid', null);
if ($instanceId === null) {
throw new \RuntimeException('no instance id!');
}
return 'appdata_' . $instanceId;
}
private function getAppDataRootFolder(): Folder {
$name = $this->getAppDataFolderName();
try {
/** @var Folder $node */
$node = $this->rootFolder->get($name);
return $node;
} catch (NotFoundException $e) {
try {
return $this->rootFolder->newFolder($name);
} catch (NotPermittedException $e) {
throw new \RuntimeException('Could not get appdata folder');
}
}
}
/**
@ -70,56 +100,64 @@ class AppData implements IAppData {
*/
private function getAppDataFolder(): Folder {
if ($this->folder === null) {
$instanceId = $this->config->getValue('instanceid', null);
if ($instanceId === null) {
throw new \RuntimeException('no instance id!');
}
$name = 'appdata_' . $instanceId;
$name = $this->getAppDataFolderName();
try {
$appDataFolder = $this->rootFolder->get($name);
$this->folder = $this->rootFolder->get($name . '/' . $this->appId);
} catch (NotFoundException $e) {
$appDataRootFolder = $this->getAppDataRootFolder();
try {
$appDataFolder = $this->rootFolder->newFolder($name);
} catch (NotPermittedException $e) {
throw new \RuntimeException('Could not get appdata folder');
$this->folder = $appDataRootFolder->get($this->appId);
} catch (NotFoundException $e) {
try {
$this->folder = $appDataRootFolder->newFolder($this->appId);
} catch (NotPermittedException $e) {
throw new \RuntimeException('Could not get appdata folder for ' . $this->appId);
}
}
}
try {
$appDataFolder = $appDataFolder->get($this->appId);
} catch (NotFoundException $e) {
try {
$appDataFolder = $appDataFolder->newFolder($this->appId);
} catch (NotPermittedException $e) {
throw new \RuntimeException('Could not get appdata folder for ' . $this->appId);
}
}
$this->folder = $appDataFolder;
}
return $this->folder;
}
public function getFolder(string $name): ISimpleFolder {
$node = $this->getAppDataFolder()->get($name);
$key = $this->appId . '/' . $name;
if ($cachedFolder = $this->folders->get($key)) {
if ($cachedFolder instanceof \Exception) {
throw $cachedFolder;
} else {
return $cachedFolder;
}
}
try {
$path = $this->getAppDataFolderName() . '/' . $this->appId . '/' . $name;
$node = $this->rootFolder->get($path);
} catch (NotFoundException $e) {
$this->folders->set($key, $e);
throw $e;
}
/** @var Folder $node */
return new SimpleFolder($node);
$folder = new SimpleFolder($node);
$this->folders->set($key, $folder);
return $folder;
}
public function newFolder(string $name): ISimpleFolder {
$key = $this->appId . '/' . $name;
$folder = $this->getAppDataFolder()->newFolder($name);
return new SimpleFolder($folder);
$simpleFolder = new SimpleFolder($folder);
$this->folders->set($key, $simpleFolder);
return $simpleFolder;
}
public function getDirectoryListing(): array {
$listing = $this->getAppDataFolder()->getDirectoryListing();
$fileListing = array_map(function(Node $folder) {
$fileListing = array_map(function (Node $folder) {
if ($folder instanceof Folder) {
return new SimpleFolder($folder);
}

View File

@ -34,6 +34,8 @@ class Factory {
/** @var SystemConfig */
private $config;
private $folders = [];
public function __construct(IRootFolder $rootFolder,
SystemConfig $systemConfig) {
@ -46,6 +48,9 @@ class Factory {
* @return AppData
*/
public function get(string $appId): AppData {
return new AppData($this->rootFolder, $this->config, $appId);
if (!isset($this->folders[$appId])) {
$this->folders[$appId] = new AppData($this->rootFolder, $this->config, $appId);
}
return $this->folders[$appId];
}
}

View File

@ -55,30 +55,22 @@ class AppDataTest extends \Test\TestCase {
}
private function setupAppFolder() {
$dataFolder = $this->createMock(Folder::class);
$appFolder = $this->createMock(Folder::class);
$this->rootFolder->expects($this->once())
$this->rootFolder->expects($this->any())
->method('get')
->with($this->equalTo('appdata_iid'))
->willReturn($dataFolder);
$dataFolder->expects($this->once())
->method('get')
->with($this->equalTo('myApp'))
->with($this->equalTo('appdata_iid/myApp'))
->willReturn($appFolder);
return [$dataFolder, $appFolder];
return $appFolder;
}
public function testGetFolder() {
$folders = $this->setupAppFolder();
$appFolder = $folders[1];
$folder = $this->createMock(Folder::class);
$appFolder->expects($this->once())
$this->rootFolder->expects($this->once())
->method('get')
->with($this->equalTo('folder'))
->with($this->equalTo('appdata_iid/myApp/folder'))
->willReturn($folder);
$result = $this->appData->getFolder('folder');
@ -86,8 +78,7 @@ class AppDataTest extends \Test\TestCase {
}
public function testNewFolder() {
$folders = $this->setupAppFolder();
$appFolder = $folders[1];
$appFolder = $this->setupAppFolder();
$folder = $this->createMock(Folder::class);
@ -101,8 +92,7 @@ class AppDataTest extends \Test\TestCase {
}
public function testGetDirectoryListing() {
$folders = $this->setupAppFolder();
$appFolder = $folders[1];
$appFolder = $this->setupAppFolder();
$file = $this->createMock(File::class);
$folder = $this->createMock(Folder::class);

View File

@ -25,6 +25,7 @@ namespace Test\Preview;
use OC\Files\AppData\Factory;
use OC\Preview\BackgroundCleanupJob;
use OC\PreviewManager;
use OC\SystemConfig;
use OCP\Files\IRootFolder;
use OCP\IDBConnection;
use Test\Traits\MountProviderTrait;
@ -77,7 +78,10 @@ class BackgroundCleanupJobTest extends \Test\TestCase {
$this->trashEnabled = $appManager->isEnabledForUser('files_trashbin', $this->userId);
$appManager->disableApp('files_trashbin');
$this->appDataFactory = \OC::$server->query(Factory::class);
$this->appDataFactory = new Factory(
\OC::$server->getRootFolder(),
\OC::$server->getSystemConfig()
);
$this->connection = \OC::$server->getDatabaseConnection();
$this->previewManager = \OC::$server->getPreviewManager();
$this->rootFolder = \OC::$server->getRootFolder();