nextcloud/lib/connector/sabre/file.php

195 lines
5.3 KiB
PHP
Raw Normal View History

2011-07-20 17:53:34 +04:00
<?php
2012-05-05 20:13:40 +04:00
2011-07-20 17:53:34 +04:00
/**
2012-05-05 20:13:40 +04:00
* ownCloud
*
* @author Jakob Sack
* @copyright 2011 Jakob Sack kde@jakobsack.de
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
* License as published by the Free Software Foundation; either
* version 3 of the License, or any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU AFFERO GENERAL PUBLIC LICENSE for more details.
*
* You should have received a copy of the GNU Affero General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*
2011-07-20 17:53:34 +04:00
*/
2012-05-05 20:13:40 +04:00
2011-07-22 16:38:42 +04:00
class OC_Connector_Sabre_File extends OC_Connector_Sabre_Node implements Sabre_DAV_IFile {
2011-07-20 17:53:34 +04:00
/**
* Updates the data
*
* The data argument is a readable stream resource.
*
* After a succesful put operation, you may choose to return an ETag. The
* etag must always be surrounded by double-quotes. These quotes must
* appear in the actual string you're returning.
*
* Clients may use the ETag from a PUT request to later on make sure that
* when they update the file, the contents haven't changed in the mean
* time.
*
* If you don't plan to store the file byte-by-byte, and you return a
* different object on a subsequent GET you are strongly recommended to not
* return an ETag, and just return null.
*
2011-07-20 17:53:34 +04:00
* @param resource $data
* @throws Sabre_DAV_Exception_Forbidden
* @return string|null
2011-07-20 17:53:34 +04:00
*/
public function put($data) {
$fs = $this->getFS();
if ($fs->file_exists($this->path) &&
!$fs->isUpdatable($this->path)) {
throw new \Sabre_DAV_Exception_Forbidden();
}
// throw an exception if encryption was disabled but the files are still encrypted
if (\OC_Util::encryptedFiles()) {
throw new \Sabre_DAV_Exception_ServiceUnavailable();
}
2013-09-24 17:14:42 +04:00
// chunked handling
if (isset($_SERVER['HTTP_OC_CHUNKED'])) {
list(, $name) = \Sabre_DAV_URLUtil::splitPath($this->path);
$info = OC_FileChunking::decodeName($name);
if (empty($info)) {
throw new Sabre_DAV_Exception_NotImplemented();
}
$chunk_handler = new OC_FileChunking($info);
$chunk_handler->store($info['index'], $data);
if ($chunk_handler->isComplete()) {
$newPath = $this->path . '/' . $info['name'];
$chunk_handler->file_assemble($newPath);
return $this->getETagPropertyForPath($newPath);
2013-09-24 17:14:42 +04:00
}
return null;
}
2013-02-10 17:16:45 +04:00
// mark file as partial while uploading (ignored by the scanner)
$partpath = $this->path . '.part';
2013-02-22 20:21:57 +04:00
$putOkay = $fs->file_put_contents($partpath, $data);
if ($putOkay === false) {
\OC_Log::write('webdav', '\OC\Files\Filesystem::file_put_contents() failed', \OC_Log::ERROR);
$fs->unlink($partpath);
throw new Sabre_DAV_Exception();
}
2013-02-22 20:21:57 +04:00
//detect aborted upload
2013-09-24 17:14:42 +04:00
if (isset ($_SERVER['REQUEST_METHOD']) && $_SERVER['REQUEST_METHOD'] === 'PUT' ) {
if (isset($_SERVER['CONTENT_LENGTH'])) {
$expected = $_SERVER['CONTENT_LENGTH'];
$actual = $fs->filesize($partpath);
if ($actual != $expected) {
$fs->unlink($partpath);
throw new Sabre_DAV_Exception_BadRequest(
'expected filesize ' . $expected . ' got ' . $actual);
}
}
}
2013-02-22 20:21:57 +04:00
2013-02-10 17:16:45 +04:00
// rename to correct path
$fs->rename($partpath, $this->path);
2013-02-22 20:21:57 +04:00
2013-09-24 17:14:42 +04:00
// allow sync clients to send the mtime along in a header
$mtime = OC_Request::hasModificationTime();
if ($mtime !== false) {
if($fs->touch($this->path, $mtime)) {
header('X-OC-MTime: accepted');
}
}
2011-07-20 17:53:34 +04:00
return $this->getETagPropertyForPath($this->path);
2011-07-20 17:53:34 +04:00
}
/**
* Returns the data
*
* @return string
*/
public function get() {
2013-09-24 17:14:42 +04:00
//throw exception if encryption is disabled but files are still encrypted
if (\OC_Util::encryptedFiles()) {
throw new \Sabre_DAV_Exception_ServiceUnavailable();
} else {
return \OC\Files\Filesystem::fopen($this->path, 'rb');
}
2011-07-20 17:53:34 +04:00
}
/**
* Delete the current file
*
* @return void
* @throws Sabre_DAV_Exception_Forbidden
2011-07-20 17:53:34 +04:00
*/
public function delete() {
if (!\OC\Files\Filesystem::isDeletable($this->path)) {
throw new \Sabre_DAV_Exception_Forbidden();
}
2012-10-10 15:18:36 +04:00
\OC\Files\Filesystem::unlink($this->path);
2011-07-20 17:53:34 +04:00
}
/**
* Returns the size of the node, in bytes
*
* @return int
*/
public function getSize() {
$this->getFileinfoCache();
if ($this->fileinfo_cache['size'] > -1) {
return $this->fileinfo_cache['size'];
} else {
return null;
}
2011-07-20 17:53:34 +04:00
}
/**
* Returns the ETag for a file
*
2013-02-11 20:44:02 +04:00
* An ETag is a unique identifier representing the current version of the
* file. If the file changes, the ETag MUST change. The ETag is an
* arbritrary string, but MUST be surrounded by double-quotes.
2011-07-20 17:53:34 +04:00
*
* Return null if the ETag can not effectively be determined
*
* @return mixed
*/
public function getETag() {
$properties = $this->getProperties(array(self::GETETAG_PROPERTYNAME));
if (isset($properties[self::GETETAG_PROPERTYNAME])) {
return $properties[self::GETETAG_PROPERTYNAME];
}
return null;
2011-07-20 17:53:34 +04:00
}
/**
* Returns the mime-type for a file
*
* If null is returned, we'll assume application/octet-stream
*
* @return mixed
*/
public function getContentType() {
if (isset($this->fileinfo_cache['mimetype'])) {
return $this->fileinfo_cache['mimetype'];
}
2011-07-20 17:53:34 +04:00
2012-10-10 15:18:36 +04:00
return \OC\Files\Filesystem::getMimeType($this->path);
2011-07-20 17:53:34 +04:00
}
}