aboutsummaryrefslogtreecommitdiffstats
path: root/apps/dav/lib/Files/Sharing/FilesDropPlugin.php
blob: b364c4ebbfc10e732f093ba9eb35f09c33a86ed7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
<?php
/**
 * SPDX-FileCopyrightText: 2016 Nextcloud GmbH and Nextcloud contributors
 * SPDX-License-Identifier: AGPL-3.0-or-later
 */
namespace OCA\DAV\Files\Sharing;

use OC\Files\View;
use OCP\Share\IShare;
use Sabre\DAV\Exception\MethodNotAllowed;
use Sabre\DAV\ServerPlugin;
use Sabre\HTTP\RequestInterface;
use Sabre\HTTP\ResponseInterface;

/**
 * Make sure that the destination is writable
 */
class FilesDropPlugin extends ServerPlugin {

	private ?View $view = null;
	private ?IShare $share = null;
	private bool $enabled = false;

	public function setView(View $view): void {
		$this->view = $view;
	}

	public function setShare(IShare $share): void {
		$this->share = $share;
	}

	public function enable(): void {
		$this->enabled = true;
	}


	/**
	 * This initializes the plugin.
	 *
	 * @param \Sabre\DAV\Server $server Sabre server
	 *
	 * @return void
	 * @throws MethodNotAllowed
	 */
	public function initialize(\Sabre\DAV\Server $server): void {
		$server->on('beforeMethod:*', [$this, 'beforeMethod'], 999);
		$this->enabled = false;
	}

	public function beforeMethod(RequestInterface $request, ResponseInterface $response): void {
		if (!$this->enabled || $this->share === null || $this->view === null) {
			return;
		}

		// Only allow file drop
		if ($request->getMethod() !== 'PUT') {
			throw new MethodNotAllowed('Only PUT is allowed on files drop');
		}

		// Always upload at the root level
		$path = explode('/', $request->getPath());
		$path = array_pop($path);

		// Extract the attributes for the file request
		$attributes = $this->share->getAttributes();
		if ($attributes === null) {
			return;
		}

		// Prepare file request
		$nickName = $request->getHeader('X-NC-Nickname');
		$isFileRequest = $attributes->getAttribute('fileRequest', 'enabled') === true;

		// We need a valid nickname for file requests
		if ($isFileRequest && ($nickName == null || trim($nickName) === '')) {
			throw new MethodNotAllowed('Nickname is required for file requests');
		}
		
		// If this is a file request we need to create a folder for the user
		if ($isFileRequest) {
			// Check if the folder already exists
			if (!($this->view->file_exists($nickName) === true)) {
				$this->view->mkdir($nickName);
			}
			// Put all files in the subfolder
			$path = $nickName . '/' . $path;
		}
		
		$newName = \OC_Helper::buildNotExistingFileNameForView('/', $path, $this->view);
		$url = $request->getBaseUrl() . $newName;
		$request->setUrl($url);
	}

}