Merge pull request #2471 from nextcloud/harden_files_drop

Harden files drop
This commit is contained in:
Lukas Reschke 2016-12-02 17:43:04 +01:00 committed by GitHub
commit bd2a8e768e
6 changed files with 340 additions and 20 deletions

View File

@ -338,6 +338,15 @@ pipeline:
when: when:
matrix: matrix:
TESTS: integration-setup-features TESTS: integration-setup-features
integration-filesdrop-features:
image: nextcloudci/integration-php7.0:integration-php7.0-2
commands:
- ./occ maintenance:install --admin-pass=admin
- cd build/integration
- ./run.sh filesdrop_features/filesdrop.feature
when:
matrix:
TESTS: integration-filesdrop-features
nodb-codecov: nodb-codecov:
image: nextcloudci/php7.0:php7.0-6 image: nextcloudci/php7.0:php7.0-6
commands: commands:
@ -382,6 +391,7 @@ matrix:
- TESTS: integration-webdav-related - TESTS: integration-webdav-related
- TESTS: integration-sharees-features - TESTS: integration-sharees-features
- TESTS: integration-setup-features - TESTS: integration-setup-features
- TESTS: integration-filesdrop-features
- TESTS: jsunit - TESTS: jsunit
- TESTS: check-autoloader - TESTS: check-autoloader
- TESTS: app-check-code - TESTS: app-check-code

View File

@ -23,6 +23,7 @@
namespace OCA\DAV\Files\Sharing; namespace OCA\DAV\Files\Sharing;
use OC\Files\View; use OC\Files\View;
use Sabre\DAV\Exception\MethodNotAllowed;
use Sabre\DAV\ServerPlugin; use Sabre\DAV\ServerPlugin;
use Sabre\HTTP\RequestInterface; use Sabre\HTTP\RequestInterface;
use Sabre\HTTP\ResponseInterface; use Sabre\HTTP\ResponseInterface;
@ -56,6 +57,7 @@ class FilesDropPlugin extends ServerPlugin {
* @param \Sabre\DAV\Server $server Sabre server * @param \Sabre\DAV\Server $server Sabre server
* *
* @return void * @return void
* @throws MethodNotAllowed
*/ */
public function initialize(\Sabre\DAV\Server $server) { public function initialize(\Sabre\DAV\Server $server) {
$server->on('beforeMethod', [$this, 'beforeMethod'], 999); $server->on('beforeMethod', [$this, 'beforeMethod'], 999);
@ -64,31 +66,19 @@ class FilesDropPlugin extends ServerPlugin {
public function beforeMethod(RequestInterface $request, ResponseInterface $response){ public function beforeMethod(RequestInterface $request, ResponseInterface $response){
if (!$this->enabled || $request->getMethod() !== 'PUT') { if (!$this->enabled) {
return; return;
} }
$path = $request->getPath(); if ($request->getMethod() !== 'PUT') {
throw new MethodNotAllowed('Only PUT is allowed on files drop');
}
$path = explode('/', $request->getPath());
$path = array_pop($path);
if ($this->view->file_exists($path)) {
$newName = \OC_Helper::buildNotExistingFileNameForView('/', $path, $this->view); $newName = \OC_Helper::buildNotExistingFileNameForView('/', $path, $this->view);
$url = $request->getBaseUrl() . $newName;
$url = $request->getBaseUrl() . $newName . '?';
$parms = $request->getQueryParameters();
$first = true;
foreach ($parms as $k => $v) {
if ($first) {
$url .= '?';
$first = false;
} else {
$url .= '&';
}
$url .= $k . '=' . $v;
}
$request->setUrl($url); $request->setUrl($url);
} }
}
} }

View File

@ -0,0 +1,179 @@
<?php
/**
* @copyright Copyright (c) 2016, Roeland Jago Douma <roeland@famdouma.nl>
*
* @author Roeland Jago Douma <roeland@famdouma.nl>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* 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
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\DAV\Tests\Files\Sharing;
use OC\Files\View;
use OCA\DAV\Files\Sharing\FilesDropPlugin;
use Sabre\DAV\Exception\MethodNotAllowed;
use Sabre\DAV\Server;
use Sabre\HTTP\RequestInterface;
use Sabre\HTTP\ResponseInterface;
use Test\TestCase;
class FilesDropPluginTest extends TestCase {
/** @var View|\PHPUnit_Framework_MockObject_MockObject */
private $view;
/** @var Server|\PHPUnit_Framework_MockObject_MockObject */
private $server;
/** @var FilesDropPlugin */
private $plugin;
/** @var RequestInterface|\PHPUnit_Framework_MockObject_MockObject */
private $request;
/** @var ResponseInterface|\PHPUnit_Framework_MockObject_MockObject */
private $response;
public function setUp() {
parent::setUp();
$this->view = $this->createMock(View::class);
$this->server = $this->createMock(Server::class);
$this->plugin = new FilesDropPlugin();
$this->request = $this->createMock(RequestInterface::class);
$this->response = $this->createMock(ResponseInterface::class);
$this->response->expects($this->never())
->method($this->anything());
}
public function testInitialize() {
$this->server->expects($this->once())
->method('on')
->with(
$this->equalTo('beforeMethod'),
$this->equalTo([$this->plugin, 'beforeMethod']),
$this->equalTo(999)
);
$this->plugin->initialize($this->server);
}
public function testNotEnabled() {
$this->view->expects($this->never())
->method($this->anything());
$this->request->expects($this->never())
->method($this->anything());
$this->plugin->beforeMethod($this->request, $this->response);
}
public function testValid() {
$this->plugin->enable();
$this->plugin->setView($this->view);
$this->request->method('getMethod')
->willReturn('PUT');
$this->request->method('getPath')
->willReturn('file.txt');
$this->request->method('getBaseUrl')
->willReturn('https://example.com');
$this->view->method('file_exists')
->with('/file.txt')
->willReturn(false);
$this->request->expects($this->once())
->method('setUrl')
->with('https://example.com/file.txt');
$this->plugin->beforeMethod($this->request, $this->response);
}
public function testFileAlreadyExistsValid() {
$this->plugin->enable();
$this->plugin->setView($this->view);
$this->request->method('getMethod')
->willReturn('PUT');
$this->request->method('getPath')
->willReturn('file.txt');
$this->request->method('getBaseUrl')
->willReturn('https://example.com');
$this->view->method('file_exists')
->will($this->returnCallback(function($path) {
if ($path === 'file.txt' || $path === '/file.txt') {
return true;
} else {
return false;
}
}));
$this->request->expects($this->once())
->method('setUrl')
->with($this->equalTo('https://example.com/file (2).txt'));
$this->plugin->beforeMethod($this->request, $this->response);
}
public function testNoMKCOL() {
$this->plugin->enable();
$this->plugin->setView($this->view);
$this->request->method('getMethod')
->willReturn('MKCOL');
$this->expectException(MethodNotAllowed::class);
$this->plugin->beforeMethod($this->request, $this->response);
}
public function testNoSubdirPut() {
$this->plugin->enable();
$this->plugin->setView($this->view);
$this->request->method('getMethod')
->willReturn('PUT');
$this->request->method('getPath')
->willReturn('folder/file.txt');
$this->request->method('getBaseUrl')
->willReturn('https://example.com');
$this->view->method('file_exists')
->will($this->returnCallback(function($path) {
if ($path === 'file.txt' || $path === '/file.txt') {
return true;
} else {
return false;
}
}));
$this->request->expects($this->once())
->method('setUrl')
->with($this->equalTo('https://example.com/file (2).txt'));
$this->plugin->beforeMethod($this->request, $this->response);
}
}

View File

@ -62,6 +62,16 @@ default:
- admin - admin
- admin - admin
regular_user_password: 123456 regular_user_password: 123456
filesdrop:
paths:
- %paths.base%/../filesdrop_features
contexts:
- FilesDropContext:
baseUrl: http://localhost:8080
admin:
- admin
- admin
regular_user_password: 123456

View File

@ -0,0 +1,72 @@
<?php
use Behat\Behat\Context\Context;
use Behat\Behat\Context\SnippetAcceptingContext;
use Behat\Behat\Tester\Exception\PendingException;
use GuzzleHttp\Client;
require __DIR__ . '/../../vendor/autoload.php';
class FilesDropContext implements Context, SnippetAcceptingContext {
use WebDav;
/**
* @When Dropping file :path with :content
*/
public function droppingFileWith($path, $content) {
$client = new Client();
$options = [];
if (count($this->lastShareData->data->element) > 0){
$token = $this->lastShareData->data[0]->token;
} else {
$token = $this->lastShareData->data[0]->token;
}
$base = substr($this->baseUrl, 0, -4);
$fullUrl = $base . '/public.php/webdav' . $path;
$options['auth'] = [$token, ''];
$options['headers'] = [
'X-REQUESTED-WITH' => 'XMLHttpRequest'
];
$request = $client->createRequest('PUT', $fullUrl, $options);
$file = \GuzzleHttp\Stream\Stream::factory($content);
$request->setBody($file);
try {
$this->response = $client->send($request);
} catch (\GuzzleHttp\Exception\ClientException $e) {
$this->response = $e->getResponse();
}
}
/**
* @When Creating folder :folder in drop
*/
public function creatingFolderInDrop($folder) {
$client = new Client();
$options = [];
if (count($this->lastShareData->data->element) > 0){
$token = $this->lastShareData->data[0]->token;
} else {
$token = $this->lastShareData->data[0]->token;
}
$base = substr($this->baseUrl, 0, -4);
$fullUrl = $base . '/public.php/webdav/' . $folder;
$options['auth'] = [$token, ''];
$options['headers'] = [
'X-REQUESTED-WITH' => 'XMLHttpRequest'
];
$request = $client->createRequest('MKCOL', $fullUrl, $options);
try {
$this->response = $client->send($request);
} catch (\GuzzleHttp\Exception\ClientException $e) {
$this->response = $e->getResponse();
}
}
}

View File

@ -0,0 +1,59 @@
Feature: FilesDrop
Scenario: Put file via files drop
Given user "user0" exists
And As an "user0"
And user "user0" created a folder "/drop"
And as "user0" creating a share with
| path | drop |
| shareType | 3 |
| publicUpload | true |
And Updating last share with
| permissions | 4 |
When Dropping file "/a.txt" with "abc"
And Downloading file "/drop/a.txt"
Then Downloaded content should be "abc"
Scenario: Put file same file multiple times via files drop
Given user "user0" exists
And As an "user0"
And user "user0" created a folder "/drop"
And as "user0" creating a share with
| path | drop |
| shareType | 3 |
| publicUpload | true |
And Updating last share with
| permissions | 4 |
When Dropping file "/a.txt" with "abc"
And Dropping file "/a.txt" with "def"
And Downloading file "/drop/a.txt"
Then Downloaded content should be "abc"
And Downloading file "/drop/a (2).txt"
Then Downloaded content should be "def"
Scenario: Files drop ignores directory
Given user "user0" exists
And As an "user0"
And user "user0" created a folder "/drop"
And as "user0" creating a share with
| path | drop |
| shareType | 3 |
| publicUpload | true |
And Updating last share with
| permissions | 4 |
When Dropping file "/folder/a.txt" with "abc"
And Downloading file "/drop/a.txt"
Then Downloaded content should be "abc"
Scenario: Files drop forbis MKCOL
Given user "user0" exists
And As an "user0"
And user "user0" created a folder "/drop"
And as "user0" creating a share with
| path | drop |
| shareType | 3 |
| publicUpload | true |
And Updating last share with
| permissions | 4 |
When Creating folder "folder" in drop
Then the HTTP status code should be "405"