From 2035a179bc319cf3f339e90e14dc01c7a980bc78 Mon Sep 17 00:00:00 2001 From: Roeland Jago Douma Date: Fri, 29 Jan 2016 21:50:48 +0100 Subject: [PATCH] Add store/retrieve checksums * Add extra db column to filecache * Bump version * Update filecache code to actually handle checksum * Webdav code to store/retrieve checksums --- apps/dav/lib/connector/sabre/file.php | 15 ++++++++++++ apps/dav/lib/connector/sabre/filesplugin.php | 24 ++++++++++++++++++-- db_structure.xml | 8 +++++++ lib/private/files/cache/cache.php | 17 ++++++++------ lib/private/files/fileinfo.php | 7 ++++++ lib/private/files/node/file.php | 7 ++++++ lib/private/files/node/node.php | 4 ++++ lib/public/files/fileinfo.php | 8 +++++++ version.php | 2 +- 9 files changed, 82 insertions(+), 10 deletions(-) diff --git a/apps/dav/lib/connector/sabre/file.php b/apps/dav/lib/connector/sabre/file.php index b925a6704055..be313a91e8cd 100644 --- a/apps/dav/lib/connector/sabre/file.php +++ b/apps/dav/lib/connector/sabre/file.php @@ -214,7 +214,13 @@ public function put($data) { header('X-OC-MTime: accepted'); } } + + if (isset($request->server['HTTP_OC_CHECKSUM'])) { + $checksum = trim($request->server['HTTP_OC_CHECKSUM']); + $this->fileView->putFileInfo($this->path, ['checksum' => $checksum]); + } $this->refreshInfo(); + } catch (StorageNotAvailableException $e) { throw new ServiceUnavailable("Failed to check file size: " . $e->getMessage()); } @@ -528,4 +534,13 @@ private function convertToSabreException(\Exception $e) { throw new \Sabre\DAV\Exception($e->getMessage(), 0, $e); } + + /** + * Get the checksum for this file + * + * @return string + */ + public function getChecksum() { + return $this->info->getChecksum(); + } } diff --git a/apps/dav/lib/connector/sabre/filesplugin.php b/apps/dav/lib/connector/sabre/filesplugin.php index ef139eae94ae..82d000149050 100644 --- a/apps/dav/lib/connector/sabre/filesplugin.php +++ b/apps/dav/lib/connector/sabre/filesplugin.php @@ -47,6 +47,7 @@ class FilesPlugin extends \Sabre\DAV\ServerPlugin { const LASTMODIFIED_PROPERTYNAME = '{DAV:}lastmodified'; const OWNER_ID_PROPERTYNAME = '{http://owncloud.org/ns}owner-id'; const OWNER_DISPLAY_NAME_PROPERTYNAME = '{http://owncloud.org/ns}owner-display-name'; + const CHECKSUM_PROPERTYNAME = '{http://owncloud.org/ns}checksum'; /** * Reference to main server object @@ -107,6 +108,7 @@ public function initialize(\Sabre\DAV\Server $server) { $server->protectedProperties[] = self::DOWNLOADURL_PROPERTYNAME; $server->protectedProperties[] = self::OWNER_ID_PROPERTYNAME; $server->protectedProperties[] = self::OWNER_DISPLAY_NAME_PROPERTYNAME; + $server->protectedProperties[] = self::CHECKSUM_PROPERTYNAME; // normally these cannot be changed (RFC4918), but we want them modifiable through PROPPATCH $allowedProperties = ['{DAV:}getetag']; @@ -178,8 +180,8 @@ function handleDownloadToken(RequestInterface $request, ResponseInterface $respo } /** - * Plugin that adds a 'Content-Disposition: attachment' header to all files - * delivered by SabreDAV. + * Add headers to file download + * * @param RequestInterface $request * @param ResponseInterface $response */ @@ -188,7 +190,15 @@ function httpGet(RequestInterface $request, ResponseInterface $response) { $node = $this->tree->getNodeForPath($request->getPath()); if (!($node instanceof IFile)) return; + // adds a 'Content-Disposition: attachment' header $response->addHeader('Content-Disposition', 'attachment'); + + //Add OC-Checksum header + /** @var $node File */ + $checksum = $node->getChecksum(); + if ($checksum !== null) { + $response->addHeader('OC-Checksum', $checksum); + } } /** @@ -237,6 +247,16 @@ public function handleGetProperties(PropFind $propFind, \Sabre\DAV\INode $node) } return false; }); + + $propFind->handle(self::CHECKSUM_PROPERTYNAME, function() use ($node) { + $checksum = $node->getChecksum(); + + if ($checksum === null) { + return ''; + } + return $checksum; + }); + } if ($node instanceof \OCA\DAV\Connector\Sabre\Directory) { diff --git a/db_structure.xml b/db_structure.xml index ea1b89e28da0..dbbfa8c7a4d3 100644 --- a/db_structure.xml +++ b/db_structure.xml @@ -397,6 +397,14 @@ 4 + + checksum + text + + false + 255 + + fs_storage_path_hash diff --git a/lib/private/files/cache/cache.php b/lib/private/files/cache/cache.php index 30e00b6080c0..cbe48f21bd8f 100644 --- a/lib/private/files/cache/cache.php +++ b/lib/private/files/cache/cache.php @@ -121,7 +121,7 @@ public function get($file) { $params = array($file); } $sql = 'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`, - `storage_mtime`, `encrypted`, `etag`, `permissions` + `storage_mtime`, `encrypted`, `etag`, `permissions`, `checksum` FROM `*PREFIX*filecache` ' . $where; $result = $this->connection->executeQuery($sql, $params); $data = $result->fetch(); @@ -177,7 +177,7 @@ public function getFolderContents($folder) { public function getFolderContentsById($fileId) { if ($fileId > -1) { $sql = 'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`, - `storage_mtime`, `encrypted`, `etag`, `permissions` + `storage_mtime`, `encrypted`, `etag`, `permissions`, `checksum` FROM `*PREFIX*filecache` WHERE `parent` = ? ORDER BY `name` ASC'; $result = $this->connection->executeQuery($sql, [$fileId]); $files = $result->fetchAll(); @@ -287,7 +287,10 @@ public function update($id, array $data) { // don't update if the data we try to set is the same as the one in the record // some databases (Postgres) don't like superfluous updates $sql = 'UPDATE `*PREFIX*filecache` SET ' . implode(' = ?, ', $queryParts) . '=? ' . - 'WHERE (' . implode(' <> ? OR ', $queryParts) . ' <> ? ) AND `fileid` = ? '; + 'WHERE (' . + implode(' <> ? OR ', $queryParts) . ' <> ? OR ' . + implode(' IS NULL OR ', $queryParts) . ' IS NULL' . + ') AND `fileid` = ? '; $this->connection->executeQuery($sql, $params); } @@ -303,7 +306,7 @@ public function update($id, array $data) { protected function buildParts(array $data) { $fields = array( 'path', 'parent', 'name', 'mimetype', 'size', 'mtime', 'storage_mtime', 'encrypted', - 'etag', 'permissions'); + 'etag', 'permissions', 'checksum'); $doNotCopyStorageMTime = false; if (array_key_exists('mtime', $data) && $data['mtime'] === null) { @@ -567,7 +570,7 @@ public function search($pattern) { $sql = ' SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`, `encrypted`, - `etag`, `permissions` + `etag`, `permissions`, `checksum` FROM `*PREFIX*filecache` WHERE `storage` = ? AND `name` ILIKE ?'; $result = $this->connection->executeQuery($sql, @@ -598,7 +601,7 @@ public function searchByMime($mimetype) { } else { $where = '`mimepart` = ?'; } - $sql = 'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`, `encrypted`, `etag`, `permissions` + $sql = 'SELECT `fileid`, `storage`, `path`, `parent`, `name`, `mimetype`, `mimepart`, `size`, `mtime`, `encrypted`, `etag`, `permissions`, `checksum` FROM `*PREFIX*filecache` WHERE ' . $where . ' AND `storage` = ?'; $mimetype = $this->mimetypeLoader->getId($mimetype); $result = $this->connection->executeQuery($sql, array($mimetype, $this->getNumericStorageId())); @@ -625,7 +628,7 @@ public function searchByMime($mimetype) { public function searchByTag($tag, $userId) { $sql = 'SELECT `fileid`, `storage`, `path`, `parent`, `name`, ' . '`mimetype`, `mimepart`, `size`, `mtime`, ' . - '`encrypted`, `etag`, `permissions` ' . + '`encrypted`, `etag`, `permissions`, `checksum` ' . 'FROM `*PREFIX*filecache` `file`, ' . '`*PREFIX*vcategory_to_object` `tagmap`, ' . '`*PREFIX*vcategory` `tag` ' . diff --git a/lib/private/files/fileinfo.php b/lib/private/files/fileinfo.php index 1e6fe474f7b7..f22e1099e26e 100644 --- a/lib/private/files/fileinfo.php +++ b/lib/private/files/fileinfo.php @@ -327,4 +327,11 @@ public function addSubEntry($data, $entryPath) { $this->childEtags[] = $relativeEntryPath . '/' . $data['etag'] . $permissions; } } + + /** + * @inheritdoc + */ + public function getChecksum() { + return $this->data['checksum']; + } } diff --git a/lib/private/files/node/file.php b/lib/private/files/node/file.php index c3d18cdb3587..cf163b9b7634 100644 --- a/lib/private/files/node/file.php +++ b/lib/private/files/node/file.php @@ -164,4 +164,11 @@ public function move($targetPath) { public function hash($type, $raw = false) { return $this->view->hash($type, $this->path, $raw); } + + /** + * @inheritdoc + */ + public function getChecksum() { + return $this->fileInfo->getChecksum(); + } } diff --git a/lib/private/files/node/node.php b/lib/private/files/node/node.php index 7769f15ee598..9feccac50bc6 100644 --- a/lib/private/files/node/node.php +++ b/lib/private/files/node/node.php @@ -351,4 +351,8 @@ public function getMountPoint() { public function getOwner() { return $this->getFileInfo()->getOwner(); } + + public function getChecksum() { + return; + } } diff --git a/lib/public/files/fileinfo.php b/lib/public/files/fileinfo.php index 77e37d53ab98..aa4aa605d323 100644 --- a/lib/public/files/fileinfo.php +++ b/lib/public/files/fileinfo.php @@ -237,4 +237,12 @@ public function getMountPoint(); * @since 9.0.0 */ public function getOwner(); + + /** + * Get the stored checksum for this file + * + * @return string + * @since 9.0.0 + */ + public function getChecksum(); } diff --git a/version.php b/version.php index 0b7eb6f79d2f..f807b01d7d04 100644 --- a/version.php +++ b/version.php @@ -25,7 +25,7 @@ // We only can count up. The 4. digit is only for the internal patchlevel to trigger DB upgrades // between betas, final and RCs. This is _not_ the public version number. Reset minor/patchlevel // when updating major/minor version number. -$OC_Version = array(9, 0, 0, 8); +$OC_Version = array(9, 0, 0, 9); // The human readable string $OC_VersionString = '9.0 pre alpha';