Skip to content

Commit c3b1501

Browse files
committed
feat: Support X-NC-Skip-Trashbin header
This is useful for clients that want to directly and permanently delete a file. Signed-off-by: Louis Chemineau <louis@chmn.me>
1 parent 4febc10 commit c3b1501

File tree

1 file changed

+25
-19
lines changed

1 file changed

+25
-19
lines changed

apps/files_trashbin/lib/Storage.php

+25-19
Original file line numberDiff line numberDiff line change
@@ -32,12 +32,15 @@
3232
use OC\Files\Storage\Wrapper\Wrapper;
3333
use OCA\Files_Trashbin\Events\MoveToTrashEvent;
3434
use OCA\Files_Trashbin\Trash\ITrashManager;
35+
use OCP\App\IAppManager;
3536
use OCP\Encryption\Exceptions\GenericEncryptionException;
3637
use OCP\EventDispatcher\IEventDispatcher;
3738
use OCP\Files\IRootFolder;
3839
use OCP\Files\Node;
3940
use OCP\Files\Storage\IStorage;
41+
use OCP\IRequest;
4042
use OCP\IUserManager;
43+
use OCP\Server;
4144
use Psr\Log\LoggerInterface;
4245

4346
class Storage extends Wrapper {
@@ -65,7 +68,8 @@ public function __construct(
6568
?IUserManager $userManager = null,
6669
?LoggerInterface $logger = null,
6770
?IEventDispatcher $eventDispatcher = null,
68-
?IRootFolder $rootFolder = null
71+
?IRootFolder $rootFolder = null,
72+
private ?IRequest $request = null,
6973
) {
7074
$this->mountPoint = $parameters['mountPoint'];
7175
$this->trashManager = $trashManager;
@@ -173,26 +177,26 @@ protected function createMoveToTrashEvent(Node $node) {
173177
* @return bool true if the operation succeeded, false otherwise
174178
*/
175179
private function doDelete($path, $method) {
176-
if (
177-
!\OC::$server->getAppManager()->isEnabledForUser('files_trashbin')
178-
|| (pathinfo($path, PATHINFO_EXTENSION) === 'part')
179-
|| $this->shouldMoveToTrash($path) === false
180-
) {
181-
return call_user_func([$this->storage, $method], $path);
182-
}
180+
$isTrashbinEnabled = Server::get(IAppManager::class)->isEnabledForUser('files_trashbin');
181+
$isPartFile = pathinfo($path, PATHINFO_EXTENSION) === 'part';
182+
$isSkipTrashHeaderSet = $this->request !== null && $this->request->getHeader('X-NC-Skip-Trashbin') === 'true';
183+
// We keep the shouldMoveToTrash call at the end to prevent emitting unnecessary event.
184+
$shouldMoveToTrash = $isTrashbinEnabled && !$isPartFile && !$isSkipTrashHeaderSet && $this->shouldMoveToTrash($path);
183185

184-
// check permissions before we continue, this is especially important for
185-
// shared files
186-
if (!$this->isDeletable($path)) {
187-
return false;
188-
}
186+
if ($shouldMoveToTrash) {
187+
// check permissions before we continue, this is especially important for
188+
// shared files
189+
if (!$this->isDeletable($path)) {
190+
return false;
191+
}
189192

190-
$isMovedToTrash = $this->trashManager->moveToTrash($this, $path);
191-
if (!$isMovedToTrash) {
192-
return call_user_func([$this->storage, $method], $path);
193-
} else {
194-
return true;
193+
$isMovedToTrash = $this->trashManager->moveToTrash($this, $path);
194+
if ($isMovedToTrash) {
195+
return true;
196+
}
195197
}
198+
199+
return call_user_func([$this->storage, $method], $path);
196200
}
197201

198202
/**
@@ -204,16 +208,18 @@ public static function setupStorage() {
204208
$logger = \OC::$server->get(LoggerInterface::class);
205209
$eventDispatcher = \OC::$server->get(IEventDispatcher::class);
206210
$rootFolder = \OC::$server->get(IRootFolder::class);
211+
$request = \OC::$server->get(IRequest::class);
207212
Filesystem::addStorageWrapper(
208213
'oc_trashbin',
209-
function (string $mountPoint, IStorage $storage) use ($trashManager, $userManager, $logger, $eventDispatcher, $rootFolder) {
214+
function (string $mountPoint, IStorage $storage) use ($trashManager, $userManager, $logger, $eventDispatcher, $rootFolder, $request) {
210215
return new Storage(
211216
['storage' => $storage, 'mountPoint' => $mountPoint],
212217
$trashManager,
213218
$userManager,
214219
$logger,
215220
$eventDispatcher,
216221
$rootFolder,
222+
$request,
217223
);
218224
},
219225
1);

0 commit comments

Comments
 (0)