414 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			414 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * ownCloud - Richdocuments App
 | |
|  *
 | |
|  * @author Victor Dubiniuk
 | |
|  * @copyright 2014 Victor Dubiniuk victor.dubiniuk@gmail.com
 | |
|  *
 | |
|  * This file is licensed under the Affero General Public License version 3 or
 | |
|  * later.
 | |
|  */
 | |
| 
 | |
| namespace OCA\Richdocuments\Controller;
 | |
| 
 | |
| use \OCP\AppFramework\Controller;
 | |
| use \OCP\IRequest;
 | |
| use \OCP\IConfig;
 | |
| use \OCP\IL10N;
 | |
| use \OCP\AppFramework\Http\ContentSecurityPolicy;
 | |
| use \OCP\AppFramework\Http\JSONResponse;
 | |
| use \OCP\AppFramework\Http\TemplateResponse;
 | |
| 
 | |
| use \OCA\Richdocuments\Db;
 | |
| use \OCA\Richdocuments\Helper;
 | |
| use \OCA\Richdocuments\Storage;
 | |
| use \OCA\Richdocuments\Download;
 | |
| use \OCA\Richdocuments\DownloadResponse;
 | |
| use \OCA\Richdocuments\File;
 | |
| use \OCA\Richdocuments\Genesis;
 | |
| use \OC\Files\View;
 | |
| use \OCP\ICacheFactory;
 | |
| use \OCP\ILogger;
 | |
| 
 | |
| class DocumentController extends Controller{
 | |
| 
 | |
| 	private $uid;
 | |
| 	private $l10n;
 | |
| 	private $settings;
 | |
| 	private $cache;
 | |
| 	private $logger;
 | |
| 
 | |
| 	const ODT_TEMPLATE_PATH = '/assets/odttemplate.odt';
 | |
| 
 | |
| 	public function __construct($appName, IRequest $request, IConfig $settings, IL10N $l10n, $uid, ICacheFactory $cache, ILogger $logger){
 | |
| 		parent::__construct($appName, $request);
 | |
| 		$this->uid = $uid;
 | |
| 		$this->l10n = $l10n;
 | |
| 		$this->settings = $settings;
 | |
| 		$this->cache = $cache->create($appName);
 | |
| 		$this->logger = $logger;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param \SimpleXMLElement $discovery
 | |
| 	 * @param string $mimetype
 | |
| 	 * @param string $action
 | |
| 	 */
 | |
| 	private function getWopiSrcUrl($discovery, $mimetype, $action) {
 | |
| 		if(is_null($discovery) || $discovery == false) {
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		$result = $discovery->xpath(sprintf('/wopi-discovery/net-zone/app[@name=\'%s\']/action[@name=\'%s\']', $mimetype, $action));
 | |
| 		if ($result && count($result) > 0) {
 | |
| 			return (string)$result[0]['urlsrc'];
 | |
| 		}
 | |
| 
 | |
| 		return null;
 | |
| 	}
 | |
| 
 | |
| 	private function responseError($message, $hint = ''){
 | |
| 		$errors = array('errors' => array(array('error' => $message, 'hint' => $hint)));
 | |
| 		$response = new TemplateResponse('', 'error', $errors, 'error');
 | |
| 		return $response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * @NoCSRFRequired
 | |
| 	 */
 | |
| 	public function index(){
 | |
| 		$wopiRemote = $this->settings->getAppValue('richdocuments', 'wopi_url');
 | |
| 		if (($parts = parse_url($wopiRemote))) {
 | |
| 			// Provides access to information about the capabilities of a WOPI client
 | |
| 			// and the mechanisms for invoking those abilities through URIs.
 | |
| 			// HTTP://server/hosting/discovery
 | |
| 			$wopiDiscovery = sprintf(
 | |
| 				"%s%s%s%s",
 | |
| 				isset($parts['scheme']) ? $parts['scheme'] . "://" : '',
 | |
| 				isset($parts['host']) ? $parts['host'] : "",
 | |
| 				isset($parts['port']) ? ":" . $parts['port'] : "",
 | |
| 				"/hosting/discovery" );
 | |
| 			$webSocketProtocol = "ws://";
 | |
| 			if ($parts['scheme'] == "https") {
 | |
| 				$webSocketProtocol = "wss://";
 | |
| 			}
 | |
| 			$webSocket = sprintf(
 | |
| 				"%s%s%s",
 | |
| 				$webSocketProtocol,
 | |
| 				isset($parts['host']) ? $parts['host'] : "",
 | |
| 				isset($parts['port']) ? ":" . $parts['port'] : "");
 | |
| 		}
 | |
| 		else {
 | |
| 			return $this->responseError('Invalid Collabora Online', $wopiRemote);
 | |
| 		}
 | |
| 
 | |
| 		$memcache = \OC::$server->getMemCacheFactory();
 | |
| 		if (!$memcache->isAvailable()) {
 | |
| 			return $this->responseError('MemCache is not enabled', $wopiRemote);
 | |
| 		}
 | |
| 
 | |
| 		$discovery = $this->cache->get('discovery.xml');
 | |
| 		if (is_null($discovery)) {
 | |
| 			try {
 | |
| 				$wopiClient = \OC::$server->getHTTPClientService()->newClient();
 | |
| 				$xmlBody = $wopiClient->get($wopiDiscovery)->getBody();
 | |
| 			}
 | |
| 			catch (\Exception $e) {
 | |
| 				return $this->responseError($e->getMessage(), $wopiRemote);
 | |
| 			}
 | |
| 
 | |
| 			if (!$xmlBody) {
 | |
| 				return $this->responseError('failure body content', $wopiRemote);
 | |
| 			}
 | |
| 
 | |
| 			$loadEntities = libxml_disable_entity_loader(true);
 | |
| 			$data = simplexml_load_string($xmlBody);
 | |
| 			libxml_disable_entity_loader($loadEntities);
 | |
| 			if ($data !== false) {
 | |
| 				$this->cache->set('discovery.xml', $xmlBody, 3600);
 | |
| 			}
 | |
| 			else {
 | |
| 				return $this->responseError('failure discovery.xml not well-formed XML string', $wopiRemote);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		\OC::$server->getNavigationManager()->setActiveEntry( 'richdocuments_index' );
 | |
| 		$maxUploadFilesize = \OCP\Util::maxUploadFilesize("/");
 | |
| 		$response = new TemplateResponse('richdocuments', 'documents', [
 | |
| 			'enable_previews' => 		$this->settings->getSystemValue('enable_previews', true),
 | |
| 			'savePath' => 			$this->settings->getUserValue($this->uid, 'richdocuments', 'save_path', '/'),
 | |
| 			'uploadMaxFilesize' =>		$maxUploadFilesize,
 | |
| 			'uploadMaxHumanFilesize' =>	\OCP\Util::humanFileSize($maxUploadFilesize),
 | |
| 			'allowShareWithLink' => 	$this->settings->getAppValue('core', 'shareapi_allow_links', 'yes'),
 | |
| 			'wopi_url' => 			$webSocket,
 | |
| 		]);
 | |
| 
 | |
| 		$policy = new ContentSecurityPolicy();
 | |
| 		$policy->addAllowedScriptDomain('\'self\' http://ajax.googleapis.com/ajax/libs/jquery/2.1.0/jquery.min.js http://cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.12/jquery.mousewheel.min.js \'unsafe-eval\' ' . $wopiRemote);
 | |
| 		$policy->addAllowedFrameDomain('\'self\' http://ajax.googleapis.com/ajax/libs/jquery/2.1.0/jquery.min.js http://cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.12/jquery.mousewheel.min.js \'unsafe-eval\' ' . $wopiRemote);
 | |
| 		$policy->addAllowedConnectDomain($webSocket);
 | |
| 		$policy->addAllowedImageDomain('*');
 | |
| 		$policy->allowInlineScript(true);
 | |
| 		$policy->addAllowedFontDomain('data:');
 | |
| 		$response->setContentSecurityPolicy($policy);
 | |
| 
 | |
| 		return $response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 */
 | |
| 	public function create(){
 | |
| 		$mimetype = $this->request->post['mimetype'];
 | |
| 
 | |
| 		$view = new View('/' . $this->uid . '/files');
 | |
| 		$dir = $this->settings->getUserValue($this->uid, $this->appName, 'save_path', '/');
 | |
| 		if (!$view->is_dir($dir)){
 | |
| 			$dir = '/';
 | |
| 		}
 | |
| 
 | |
| 		$basename = $this->l10n->t('New Document.odt');
 | |
| 		switch ($mimetype) {
 | |
| 			case 'application/vnd.oasis.opendocument.spreadsheet':
 | |
| 				$basename = $this->l10n->t('New Spreadsheet.ods');
 | |
| 				break;
 | |
| 			case 'application/vnd.oasis.opendocument.presentation':
 | |
| 				$basename = $this->l10n->t('New Presentation.odp');
 | |
| 				break;
 | |
| 			default:
 | |
| 				// to be safe
 | |
| 				$mimetype = 'application/vnd.oasis.opendocument.text';
 | |
| 				break;
 | |
| 		}
 | |
| 
 | |
| 		$path = Helper::getNewFileName($view, $dir . '/' . $basename);
 | |
| 
 | |
| 		$content = '';
 | |
| 		if (class_exists('\OC\Files\Type\TemplateManager')){
 | |
| 			$manager = \OC_Helper::getFileTemplateManager();
 | |
| 			$content = $manager->getTemplate($mimetype);
 | |
| 		}
 | |
| 
 | |
| 		if (!$content){
 | |
| 			$content = file_get_contents(dirname(__DIR__) . self::ODT_TEMPLATE_PATH);
 | |
| 		}
 | |
| 
 | |
| 		if ($content && $view->file_put_contents($path, $content)){
 | |
| 			$info = $view->getFileInfo($path);
 | |
| 			$response =  array(
 | |
| 				'status' => 'success',
 | |
| 				'fileid' => $info['fileid']
 | |
| 			);
 | |
| 		} else {
 | |
| 			$response =  array(
 | |
| 				'status' => 'error',
 | |
| 				'message' => (string) $this->l10n->t('Can\'t create document')
 | |
| 			);
 | |
| 		}
 | |
| 		return $response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * Generates and returns an access token for a given fileId.
 | |
| 	 * Only for authenticated users!
 | |
| 	 */
 | |
| 	public function wopiGetToken($fileId){
 | |
| 		\OC::$server->getLogger()->debug('Generating WOPI Token for file {fileId}.', [ 'app' => $this->appName, 'fileId' => $fileId ]);
 | |
| 
 | |
| 		$row = new Db\Wopi();
 | |
| 		$token = $row->generateFileToken($fileId);
 | |
| 
 | |
| 		// Return the token.
 | |
| 		return array(
 | |
| 			'status' => 'success',
 | |
| 			'token' => $token
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * @NoCSRFRequired
 | |
| 	 * @PublicPage
 | |
| 	 * Returns general info about a file.
 | |
| 	 */
 | |
| 	public function wopiCheckFileInfo($fileId){
 | |
| 		$token = $this->request->getParam('access_token');
 | |
| 
 | |
| 		\OC::$server->getLogger()->debug('Getting info about file {fileId} by token {token}.', [ 'app' => $this->appName, 'fileId' => $fileId, 'token' => $token ]);
 | |
| 
 | |
| 		$row = new Db\Wopi();
 | |
| 		$row->loadBy('token', $token);
 | |
| 
 | |
| 		$res = $row->getPathForToken($fileId, $token);
 | |
| 		if ($res == false || http_response_code() != 200)
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		$view = new \OC\Files\View('/' . $res['user'] . '/');
 | |
| 		$info = $view->getFileInfo($res['path']);
 | |
| 
 | |
| 		\OC::$server->getLogger()->debug('File info: {info}.', [ 'app' => $this->appName, 'info' => $info ]);
 | |
| 
 | |
| 		$baseFileName = $info['name'];
 | |
| 		$size = $info['size'];
 | |
| 
 | |
| 		return array(
 | |
| 			'BaseFileName' => $baseFileName,
 | |
| 			'Size' => $size,
 | |
| 			//'DownloadUrl' => '',
 | |
| 			//'FileUrl' => '',
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * @NoCSRFRequired
 | |
| 	 * @PublicPage
 | |
| 	 * Given an access token and a fileId, returns the contents of the file.
 | |
| 	 * Expects a valid token in access_token parameter.
 | |
| 	 */
 | |
| 	public function wopiGetFile($fileId){
 | |
| 		$token = $this->request->getParam('access_token');
 | |
| 
 | |
| 		\OC::$server->getLogger()->debug('Getting contents of file {fileId} by token {token}.', [ 'app' => $this->appName, 'fileId' => $fileId, 'token' => $token ]);
 | |
| 
 | |
| 		$row = new Db\Wopi();
 | |
| 		$row->loadBy('token', $token);
 | |
| 
 | |
| 		//TODO: Support X-WOPIMaxExpectedSize header.
 | |
| 		$res = $row->getPathForToken($fileId, $token);
 | |
| 		return new DownloadResponse($this->request, $res['user'], $res['path']);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * @NoCSRFRequired
 | |
| 	 * @PublicPage
 | |
| 	 * Given an access token and a fileId, replaces the files with the request body.
 | |
| 	 * Expects a valid token in access_token parameter.
 | |
| 	 */
 | |
| 	public function wopiPutFile($fileId){
 | |
| 		$token = $this->request->getParam('access_token');
 | |
| 
 | |
| 		\OC::$server->getLogger()->debug('Putting contents of file {fileId} by token {token}.', [ 'app' => $this->appName, 'fileId' => $fileId, 'token' => $token ]);
 | |
| 
 | |
| 		$row = new Db\Wopi();
 | |
| 		$row->loadBy('token', $token);
 | |
| 
 | |
| 		$res = $row->getPathForToken($fileId, $token);
 | |
| 		$view = new \OC\Files\View('/' . $res['user'] . '/');
 | |
| 
 | |
| 		// Read the contents of the file from the POST body and store.
 | |
| 		$content = file_get_contents('php://input');
 | |
| 		\OC::$server->getLogger()->debug('Putting {size} bytes.', [ 'app' => $this->appName, 'size' => strlen($content) ]);
 | |
| 
 | |
| 		$view->file_put_contents($res['path'], $content);
 | |
| 
 | |
| 		return array(
 | |
| 			'status' => 'success'
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * @PublicPage
 | |
| 	 * Process partial/complete file download
 | |
| 	 */
 | |
| 	public function serve($esId){
 | |
| 		$session = new Db\Session();
 | |
| 		$session->load($esId);
 | |
| 
 | |
| 		$filename = $session->getGenesisUrl() ? $session->getGenesisUrl() : '';
 | |
| 		return new DownloadResponse($this->request, $session->getOwner(), $filename);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 */
 | |
| 	public function download($path){
 | |
| 		if (!$path){
 | |
| 			$response = new JSONResponse();
 | |
| 			$response->setStatus(Http::STATUS_BAD_REQUEST);
 | |
| 			return $response;
 | |
| 		}
 | |
| 
 | |
| 		$fullPath = '/files' . $path;
 | |
| 		$fileInfo = \OC\Files\Filesystem::getFileInfo($path);
 | |
| 		if ($fileInfo){
 | |
| 			$file = new File($fileInfo->getId());
 | |
| 			$genesis = new Genesis($file);
 | |
| 			$fullPath = $genesis->getPath();
 | |
| 		}
 | |
| 		return new DownloadResponse($this->request, $this->uid, $fullPath);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 */
 | |
| 	public function rename($fileId){
 | |
| 		$name = $this->request->post['name'];
 | |
| 
 | |
| 		$view = \OC\Files\Filesystem::getView();
 | |
| 		$path = $view->getPath($fileId);
 | |
| 
 | |
| 		if ($name && $view->is_file($path) && $view->isUpdatable($path)) {
 | |
| 			$newPath = dirname($path) . '/' . $name;
 | |
| 			if ($view->rename($path, $newPath)) {
 | |
| 						return array('status' => 'success');
 | |
| 			}
 | |
| 		}
 | |
| 		return array(
 | |
| 			'status' => 'error',
 | |
| 			'message' => (string) $this->l10n->t('You don\'t have permission to rename this document')
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @NoAdminRequired
 | |
| 	 * lists the documents the user has access to (including shared files, once the code in core has been fixed)
 | |
| 	 * also adds session and member info for these files
 | |
| 	 */
 | |
| 	public function listAll(){
 | |
| 		$found = Storage::getDocuments();
 | |
| 		$data = $this->cache->get('discovery.xml');
 | |
| 		$loadEntities = libxml_disable_entity_loader(true);
 | |
| 		$discovery = simplexml_load_string($data);
 | |
| 		libxml_disable_entity_loader($loadEntities);
 | |
| 
 | |
| 		$fileIds = array();
 | |
| 		$documents = array();
 | |
| 		foreach ($found as $key=>$document) {
 | |
| 			if (is_object($document)){
 | |
| 				$documents[] = $document->getData();
 | |
| 			} else {
 | |
| 				$documents[$key] = $document;
 | |
| 			}
 | |
| 			$documents[$key]['icon'] = preg_replace('/\.png$/', '.svg', \OCP\Template::mimetype_icon($document['mimetype']));
 | |
| 			$documents[$key]['hasPreview'] = \OC::$server->getPreviewManager()->isMimeSupported($document['mimetype']);
 | |
| 			$documents[$key]['urlsrc'] = $this->getWopiSrcUrl($discovery, $document['mimetype'], 'edit');
 | |
| 			$fileIds[] = $document['fileid'];
 | |
| 		}
 | |
| 
 | |
| 		usort($documents, function($a, $b){
 | |
| 			return @$b['mtime']-@$a['mtime'];
 | |
| 		});
 | |
| 
 | |
| 		$session = new Db\Session();
 | |
| 		$sessions = $session->getCollectionBy('file_id', $fileIds);
 | |
| 
 | |
| 		$members = array();
 | |
| 		$member = new Db\Member();
 | |
| 		foreach ($sessions as $session) {
 | |
| 			$members[$session['es_id']] = $member->getActiveCollection($session['es_id']);
 | |
| 		}
 | |
| 
 | |
| 		return array(
 | |
| 			'status' => 'success', 'documents' => $documents,'sessions' => $sessions,'members' => $members
 | |
| 		);
 | |
| 	}
 | |
| }
 |