272 lines
7.5 KiB
PHP
272 lines
7.5 KiB
PHP
<?php
|
|
/**
|
|
* @author Lukas Reschke <lukas@owncloud.com>
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
|
* @author Vincent Petry <pvince81@owncloud.com>
|
|
*
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
|
* @license AGPL-3.0
|
|
*
|
|
* This code is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program 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, version 3,
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*/
|
|
namespace OCA\DAV\SystemTag;
|
|
|
|
use OCP\IGroupManager;
|
|
use OCP\IUserSession;
|
|
use Sabre\DAV\Exception\NotFound;
|
|
use Sabre\DAV\PropFind;
|
|
use Sabre\DAV\PropPatch;
|
|
use Sabre\DAV\Exception\BadRequest;
|
|
use Sabre\DAV\Exception\UnsupportedMediaType;
|
|
use Sabre\DAV\Exception\Conflict;
|
|
|
|
use OCP\SystemTag\ISystemTag;
|
|
use OCP\SystemTag\ISystemTagManager;
|
|
use OCP\SystemTag\TagAlreadyExistsException;
|
|
use Sabre\HTTP\RequestInterface;
|
|
use Sabre\HTTP\ResponseInterface;
|
|
|
|
/**
|
|
* Sabre plugin to handle system tags:
|
|
*
|
|
* - makes it possible to create new tags with POST operation
|
|
* - get/set Webdav properties for tags
|
|
*
|
|
*/
|
|
class SystemTagPlugin extends \Sabre\DAV\ServerPlugin {
|
|
|
|
// namespace
|
|
const NS_OWNCLOUD = 'http://owncloud.org/ns';
|
|
const ID_PROPERTYNAME = '{http://owncloud.org/ns}id';
|
|
const DISPLAYNAME_PROPERTYNAME = '{http://owncloud.org/ns}display-name';
|
|
const USERVISIBLE_PROPERTYNAME = '{http://owncloud.org/ns}user-visible';
|
|
const USERASSIGNABLE_PROPERTYNAME = '{http://owncloud.org/ns}user-assignable';
|
|
|
|
/**
|
|
* @var \Sabre\DAV\Server $server
|
|
*/
|
|
private $server;
|
|
|
|
/**
|
|
* @var ISystemTagManager
|
|
*/
|
|
protected $tagManager;
|
|
|
|
/**
|
|
* @var IUserSession
|
|
*/
|
|
protected $userSession;
|
|
|
|
/**
|
|
* @var IGroupManager
|
|
*/
|
|
protected $groupManager;
|
|
|
|
/**
|
|
* @param ISystemTagManager $tagManager tag manager
|
|
* @param IGroupManager $groupManager
|
|
* @param IUserSession $userSession
|
|
*/
|
|
public function __construct(ISystemTagManager $tagManager,
|
|
IGroupManager $groupManager,
|
|
IUserSession $userSession) {
|
|
$this->tagManager = $tagManager;
|
|
$this->userSession = $userSession;
|
|
$this->groupManager = $groupManager;
|
|
}
|
|
|
|
/**
|
|
* This initializes the plugin.
|
|
*
|
|
* This function is called by \Sabre\DAV\Server, after
|
|
* addPlugin is called.
|
|
*
|
|
* This method should set up the required event subscriptions.
|
|
*
|
|
* @param \Sabre\DAV\Server $server
|
|
* @return void
|
|
*/
|
|
public function initialize(\Sabre\DAV\Server $server) {
|
|
|
|
$server->xml->namespaceMap[self::NS_OWNCLOUD] = 'oc';
|
|
|
|
$server->protectedProperties[] = self::ID_PROPERTYNAME;
|
|
|
|
$server->on('propFind', array($this, 'handleGetProperties'));
|
|
$server->on('propPatch', array($this, 'handleUpdateProperties'));
|
|
$server->on('method:POST', [$this, 'httpPost']);
|
|
|
|
$this->server = $server;
|
|
}
|
|
|
|
/**
|
|
* POST operation on system tag collections
|
|
*
|
|
* @param RequestInterface $request request object
|
|
* @param ResponseInterface $response response object
|
|
* @return null|false
|
|
*/
|
|
public function httpPost(RequestInterface $request, ResponseInterface $response) {
|
|
$path = $request->getPath();
|
|
|
|
// Making sure the node exists
|
|
$node = $this->server->tree->getNodeForPath($path);
|
|
if ($node instanceof SystemTagsByIdCollection || $node instanceof SystemTagsObjectMappingCollection) {
|
|
$data = $request->getBodyAsString();
|
|
|
|
$tag = $this->createTag($data, $request->getHeader('Content-Type'));
|
|
|
|
if ($node instanceof SystemTagsObjectMappingCollection) {
|
|
// also add to collection
|
|
$node->createFile($tag->getId());
|
|
$url = $request->getBaseUrl() . 'systemtags/';
|
|
} else {
|
|
$url = $request->getUrl();
|
|
}
|
|
|
|
if ($url[strlen($url) - 1] !== '/') {
|
|
$url .= '/';
|
|
}
|
|
|
|
$response->setHeader('Content-Location', $url . $tag->getId());
|
|
|
|
// created
|
|
$response->setStatus(201);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a new tag
|
|
*
|
|
* @param string $data JSON encoded string containing the properties of the tag to create
|
|
* @param string $contentType content type of the data
|
|
* @return ISystemTag newly created system tag
|
|
*
|
|
* @throws BadRequest if a field was missing
|
|
* @throws Conflict if a tag with the same properties already exists
|
|
* @throws UnsupportedMediaType if the content type is not supported
|
|
*/
|
|
private function createTag($data, $contentType = 'application/json') {
|
|
if (explode(';', $contentType)[0] === 'application/json') {
|
|
$data = json_decode($data, true);
|
|
} else {
|
|
throw new UnsupportedMediaType();
|
|
}
|
|
|
|
if (!isset($data['name'])) {
|
|
throw new BadRequest('Missing "name" attribute');
|
|
}
|
|
|
|
$tagName = $data['name'];
|
|
$userVisible = true;
|
|
$userAssignable = true;
|
|
|
|
if (isset($data['userVisible'])) {
|
|
$userVisible = (bool)$data['userVisible'];
|
|
}
|
|
|
|
if (isset($data['userAssignable'])) {
|
|
$userAssignable = (bool)$data['userAssignable'];
|
|
}
|
|
|
|
if($userVisible === false || $userAssignable === false) {
|
|
if(!$this->userSession->isLoggedIn() || !$this->groupManager->isAdmin($this->userSession->getUser()->getUID())) {
|
|
throw new BadRequest('Not sufficient permissions');
|
|
}
|
|
}
|
|
|
|
try {
|
|
return $this->tagManager->createTag($tagName, $userVisible, $userAssignable);
|
|
} catch (TagAlreadyExistsException $e) {
|
|
throw new Conflict('Tag already exists', 0, $e);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves system tag properties
|
|
*
|
|
* @param PropFind $propFind
|
|
* @param \Sabre\DAV\INode $node
|
|
*/
|
|
public function handleGetProperties(
|
|
PropFind $propFind,
|
|
\Sabre\DAV\INode $node
|
|
) {
|
|
if (!($node instanceof SystemTagNode)) {
|
|
return;
|
|
}
|
|
|
|
$propFind->handle(self::ID_PROPERTYNAME, function() use ($node) {
|
|
return $node->getSystemTag()->getId();
|
|
});
|
|
|
|
$propFind->handle(self::DISPLAYNAME_PROPERTYNAME, function() use ($node) {
|
|
return $node->getSystemTag()->getName();
|
|
});
|
|
|
|
$propFind->handle(self::USERVISIBLE_PROPERTYNAME, function() use ($node) {
|
|
return $node->getSystemTag()->isUserVisible() ? 'true' : 'false';
|
|
});
|
|
|
|
$propFind->handle(self::USERASSIGNABLE_PROPERTYNAME, function() use ($node) {
|
|
return $node->getSystemTag()->isUserAssignable() ? 'true' : 'false';
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Updates tag attributes
|
|
*
|
|
* @param string $path
|
|
* @param PropPatch $propPatch
|
|
*
|
|
* @return void
|
|
*/
|
|
public function handleUpdateProperties($path, PropPatch $propPatch) {
|
|
$propPatch->handle([
|
|
self::DISPLAYNAME_PROPERTYNAME,
|
|
self::USERVISIBLE_PROPERTYNAME,
|
|
self::USERASSIGNABLE_PROPERTYNAME,
|
|
], function($props) use ($path) {
|
|
$node = $this->server->tree->getNodeForPath($path);
|
|
if (!($node instanceof SystemTagNode)) {
|
|
return;
|
|
}
|
|
|
|
$tag = $node->getSystemTag();
|
|
$name = $tag->getName();
|
|
$userVisible = $tag->isUserVisible();
|
|
$userAssignable = $tag->isUserAssignable();
|
|
|
|
if (isset($props[self::DISPLAYNAME_PROPERTYNAME])) {
|
|
$name = $props[self::DISPLAYNAME_PROPERTYNAME];
|
|
}
|
|
|
|
if (isset($props[self::USERVISIBLE_PROPERTYNAME])) {
|
|
$propValue = $props[self::USERVISIBLE_PROPERTYNAME];
|
|
$userVisible = ($propValue !== 'false' && $propValue !== '0');
|
|
}
|
|
|
|
if (isset($props[self::USERASSIGNABLE_PROPERTYNAME])) {
|
|
$propValue = $props[self::USERASSIGNABLE_PROPERTYNAME];
|
|
$userAssignable = ($propValue !== 'false' && $propValue !== '0');
|
|
}
|
|
|
|
$node->update($name, $userVisible, $userAssignable);
|
|
return true;
|
|
});
|
|
}
|
|
}
|