2012-09-16 14:52:32 +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 OC\Files\Cache;
|
|
|
|
|
|
|
|
class Cache {
|
|
|
|
/**
|
|
|
|
* @var array partial data for the cache
|
|
|
|
*/
|
|
|
|
private static $partial = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get the stored metadata of a file or folder
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File or int $file
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
static public function get($file) {
|
|
|
|
if ($file instanceof \OC\Files\File) {
|
|
|
|
$where = 'WHERE `storage` = ? AND `path_hash` = ?';
|
2012-09-22 13:43:48 +00:00
|
|
|
$params = array($file->getStorageId(), md5($file->getInternalPath()));
|
2012-09-16 14:52:32 +00:00
|
|
|
} else { //file id
|
|
|
|
$where = 'WHERE `fileid` = ?';
|
|
|
|
$params = array($file);
|
|
|
|
}
|
|
|
|
$query = \OC_DB::prepare(
|
2012-09-22 13:43:48 +00:00
|
|
|
'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`
|
2012-09-16 14:52:32 +00:00
|
|
|
FROM `*PREFIX*filecache` ' . $where);
|
2012-09-22 12:40:04 +00:00
|
|
|
$result = $query->execute($params);
|
2012-09-22 13:43:48 +00:00
|
|
|
$data = $result->fetchRow();
|
2012-09-22 12:40:04 +00:00
|
|
|
|
|
|
|
//merge partial data
|
2012-09-22 13:43:48 +00:00
|
|
|
if (!$data and $file instanceof \OC\Files\File) {
|
|
|
|
$key = $file->getStorageId() . '::' . $file->getInternalPath();
|
|
|
|
if (isset(self::$partial[$key])) {
|
|
|
|
$data = self::$partial[$key];
|
|
|
|
}
|
2012-09-22 12:40:04 +00:00
|
|
|
}
|
2012-09-22 13:43:48 +00:00
|
|
|
return $data;
|
2012-09-16 14:52:32 +00:00
|
|
|
}
|
|
|
|
|
2012-09-23 13:25:03 +00:00
|
|
|
/**
|
|
|
|
* get the metadata of all files stored in $folder
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $folder
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
static public function getFolderContents($folder) {
|
|
|
|
$fileId = self::getId($folder);
|
|
|
|
if ($fileId > -1) {
|
|
|
|
$query = \OC_DB::prepare(
|
|
|
|
'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`
|
|
|
|
FROM `*PREFIX*filecache` WHERE parent = ?');
|
|
|
|
$result = $query->execute(array($fileId));
|
|
|
|
return $result->fetchAll();
|
|
|
|
} else {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-16 14:52:32 +00:00
|
|
|
/**
|
|
|
|
* store meta data for a file or folder
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $file
|
|
|
|
* @param array $data
|
|
|
|
*
|
|
|
|
* @return int file id
|
|
|
|
*/
|
|
|
|
static public function put(\OC\Files\File $file, array $data) {
|
2012-09-22 13:43:48 +00:00
|
|
|
if (($id = self::getId($file)) > -1) {
|
2012-09-16 14:52:32 +00:00
|
|
|
self::update($id, $data);
|
|
|
|
return $id;
|
|
|
|
} else {
|
|
|
|
$key = $file->getStorageId() . '::' . $file->getInternalPath();
|
|
|
|
if (isset(self::$partial[$key])) { //add any saved partial data
|
2012-09-22 13:43:48 +00:00
|
|
|
$data = array_merge(self::$partial[$key], $data);
|
2012-09-16 14:52:32 +00:00
|
|
|
unset(self::$partial[$key]);
|
|
|
|
}
|
|
|
|
|
|
|
|
$requiredFields = array('size', 'mtime', 'mimetype');
|
|
|
|
foreach ($requiredFields as $field) {
|
|
|
|
if (!isset($data[$field])) { //data not complete save as partial and return
|
|
|
|
self::$partial[$key] = $data;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$data['path'] = $file->getInternalPath();
|
|
|
|
$data['parent'] = self::getParentId($file);
|
|
|
|
$data['name'] = basename($file->getInternalPath());
|
|
|
|
|
|
|
|
list($queryParts, $params) = self::buildParts($data);
|
|
|
|
$queryParts[] = '`storage`';
|
|
|
|
$params[] = $file->getStorageId();
|
|
|
|
$valuesPlaceholder = array_fill(0, count($queryParts), '?');
|
|
|
|
|
2012-09-22 13:43:48 +00:00
|
|
|
$query = \OC_DB::prepare('INSERT INTO `*PREFIX*filecache`(' . implode(', ', $queryParts) . ') VALUES(' . implode(', ', $valuesPlaceholder) . ')');
|
2012-09-16 14:52:32 +00:00
|
|
|
$query->execute($params);
|
|
|
|
|
|
|
|
return \OC_DB::insertid('*PREFIX*filecache');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* update the metadata in the cache
|
|
|
|
*
|
|
|
|
* @param int $id
|
|
|
|
* @param array $data
|
|
|
|
*/
|
|
|
|
static public function update($id, array $data) {
|
|
|
|
list($queryParts, $params) = self::buildParts($data);
|
|
|
|
$params[] = $id;
|
|
|
|
|
|
|
|
$query = \OC_DB::prepare('UPDATE `*PREFIX*filecache` SET ' . implode(' = ?, ', $queryParts) . '=? WHERE fileid = ?');
|
|
|
|
$query->execute($params);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* extract query parts and params array from data array
|
|
|
|
*
|
|
|
|
* @param array $data
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private static function buildParts(array $data) {
|
|
|
|
$fields = array('path', 'parent', 'name', 'mimetype', 'size', 'mtime');
|
|
|
|
|
|
|
|
$params = array();
|
|
|
|
$queryParts = array();
|
|
|
|
foreach ($data as $name => $value) {
|
|
|
|
if (array_search($name, $fields) !== false) {
|
|
|
|
$params[] = $value;
|
|
|
|
$queryParts[] = '`' . $name . '`';
|
|
|
|
if ($name === 'path') {
|
|
|
|
$params[] = md5($value);
|
|
|
|
$queryParts[] = '`path_hash`';
|
|
|
|
} elseif ($name === 'mimetype') {
|
|
|
|
$params[] = substr($value, 0, strpos($value, '/'));
|
|
|
|
$queryParts[] = '`mimepart`';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return array($queryParts, $params);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get the file id for a file
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $file
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
static public function getId(\OC\Files\File $file) {
|
|
|
|
$storageId = $file->getStorageId();
|
|
|
|
$pathHash = md5($file->getInternalPath());
|
|
|
|
|
2012-09-22 13:43:48 +00:00
|
|
|
$query = \OC_DB::prepare('SELECT `fileid` FROM `*PREFIX*filecache` WHERE `storage` = ? AND `path_hash` = ?');
|
2012-09-16 14:52:32 +00:00
|
|
|
$result = $query->execute(array($storageId, $pathHash));
|
|
|
|
|
|
|
|
if ($row = $result->fetchRow()) {
|
2012-09-22 13:43:48 +00:00
|
|
|
return $row['fileid'];
|
2012-09-16 14:52:32 +00:00
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get the id of the parent folder of a file
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $file
|
2012-09-22 13:48:39 +00:00
|
|
|
* @return int
|
2012-09-16 14:52:32 +00:00
|
|
|
*/
|
|
|
|
static public function getParentId(\OC\Files\File $file) {
|
|
|
|
$path = $file->getInternalPath();
|
|
|
|
if ($path === '/' or $path === '') {
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return self::getId(new \OC\Files\File($file->getStorage(), dirname($path)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* check if a file is available in the cache
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $file
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
static public function inCache(\OC\Files\File $file) {
|
|
|
|
return self::getId($file) != -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* remove a file or folder from the cache
|
|
|
|
*
|
|
|
|
* @param \OC\Files\File $file
|
|
|
|
*/
|
2012-09-22 13:43:48 +00:00
|
|
|
static public function remove(\OC\Files\File $file) {
|
2012-09-16 14:52:32 +00:00
|
|
|
$storageId = $file->getStorageId();
|
|
|
|
$pathHash = md5($file->getInternalPath());
|
|
|
|
$query = \OC_DB::prepare('DELETE FROM `*PREFIX*filecache` WHERE `storage` = ? AND `path_hash` = ?');
|
|
|
|
$query->execute(array($storageId, $pathHash));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* remove all entries for files that are stored on $storage form the cache
|
|
|
|
*
|
|
|
|
* @param \OC\Files\Storage\Storage $storage
|
|
|
|
*/
|
2012-09-22 13:43:48 +00:00
|
|
|
static public function removeStorage(\OC\Files\Storage\Storage $storage) {
|
2012-09-16 14:52:32 +00:00
|
|
|
$storageId = $storage->getId();
|
|
|
|
$query = \OC_DB::prepare('DELETE FROM `*PREFIX*filecache` WHERE storage=?');
|
|
|
|
$query->execute(array($storageId));
|
|
|
|
}
|
|
|
|
}
|