summaryrefslogtreecommitdiffstats
path: root/tests
diff options
context:
space:
mode:
Diffstat (limited to 'tests')
-rw-r--r--tests/acceptance/config/behat.yml1
-rw-r--r--tests/acceptance/features/app-files.feature31
-rw-r--r--tests/acceptance/features/bootstrap/FilesAppContext.php138
-rw-r--r--tests/acceptance/features/bootstrap/FilesSharingAppContext.php110
-rw-r--r--tests/acceptance/features/core/Actor.php22
-rw-r--r--tests/acceptance/features/core/ActorContext.php10
-rw-r--r--tests/lib/Share20/ManagerTest.php96
7 files changed, 403 insertions, 5 deletions
diff --git a/tests/acceptance/config/behat.yml b/tests/acceptance/config/behat.yml
index 6c3d9e4a7b9..15310e6883f 100644
--- a/tests/acceptance/config/behat.yml
+++ b/tests/acceptance/config/behat.yml
@@ -11,6 +11,7 @@ default:
- FeatureContext
- FilesAppContext
+ - FilesSharingAppContext
- LoginPageContext
- NotificationContext
- SettingsMenuContext
diff --git a/tests/acceptance/features/app-files.feature b/tests/acceptance/features/app-files.feature
new file mode 100644
index 00000000000..7adc618e02e
--- /dev/null
+++ b/tests/acceptance/features/app-files.feature
@@ -0,0 +1,31 @@
+Feature: app-files
+
+ Scenario: set a password to a shared link
+ Given I am logged in
+ And I share the link for "welcome.txt"
+ When I protect the shared link with the password "abcdef"
+ Then I see that the working icon for password protect is shown
+ And I see that the working icon for password protect is eventually not shown
+
+ Scenario: access a shared link protected by password with a valid password
+ Given I act as John
+ And I am logged in
+ And I share the link for "welcome.txt" protected by the password "abcdef"
+ And I write down the shared link
+ When I act as Jane
+ And I visit the shared link I wrote down
+ And I see that the current page is the Authenticate page for the shared link I wrote down
+ And I authenticate with password "abcdef"
+ Then I see that the current page is the shared link I wrote down
+ And I see that the shared file preview shows the text "Welcome to your Nextcloud account!"
+
+ Scenario: access a shared link protected by password with an invalid password
+ Given I act as John
+ And I am logged in
+ And I share the link for "welcome.txt" protected by the password "abcdef"
+ And I write down the shared link
+ When I act as Jane
+ And I visit the shared link I wrote down
+ And I authenticate with password "fedcba"
+ Then I see that the current page is the Authenticate page for the shared link I wrote down
+ And I see that a wrong password for the shared file message is shown
diff --git a/tests/acceptance/features/bootstrap/FilesAppContext.php b/tests/acceptance/features/bootstrap/FilesAppContext.php
index 9702e64b552..7e7f592a44e 100644
--- a/tests/acceptance/features/bootstrap/FilesAppContext.php
+++ b/tests/acceptance/features/bootstrap/FilesAppContext.php
@@ -28,6 +28,105 @@ class FilesAppContext implements Context, ActorAwareInterface {
use ActorAware;
/**
+ * @return Locator
+ */
+ public static function currentSectionMainView() {
+ return Locator::forThe()->xpath("//*[starts-with(@id, 'app-content-') and not(contains(concat(' ', normalize-space(@class), ' '), ' hidden '))]")->
+ describedAs("Current section main view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function currentSectionDetailsView() {
+ return Locator::forThe()->xpath("/preceding-sibling::*[position() = 1 and @id = 'app-sidebar']")->
+ descendantOf(self::currentSectionMainView())->
+ describedAs("Current section details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function shareLinkCheckbox() {
+ return Locator::forThe()->content("Share link")->descendantOf(self::currentSectionDetailsView())->
+ describedAs("Share link checkbox in the details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function shareLinkField() {
+ return Locator::forThe()->css(".linkText")->descendantOf(self::currentSectionDetailsView())->
+ describedAs("Share link field in the details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function passwordProtectCheckbox() {
+ return Locator::forThe()->content("Password protect")->descendantOf(self::currentSectionDetailsView())->
+ describedAs("Password protect checkbox in the details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function passwordProtectField() {
+ return Locator::forThe()->css(".linkPassText")->descendantOf(self::currentSectionDetailsView())->
+ describedAs("Password protect field in the details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function passwordProtectWorkingIcon() {
+ return Locator::forThe()->css(".linkPass .icon-loading-small")->descendantOf(self::currentSectionDetailsView())->
+ describedAs("Password protect working icon in the details view in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function rowForFile($fileName) {
+ return Locator::forThe()->xpath("//*[@id = 'fileList']//span[contains(concat(' ', normalize-space(@class), ' '), ' nametext ') and normalize-space() = '$fileName']/ancestor::tr")->
+ descendantOf(self::currentSectionMainView())->
+ describedAs("Row for file $fileName in Files app");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function shareActionForFile($fileName) {
+ return Locator::forThe()->css(".action-share")->descendantOf(self::rowForFile($fileName))->
+ describedAs("Share action for file $fileName in Files app");
+ }
+
+ /**
+ * @Given I share the link for :fileName
+ */
+ public function iShareTheLinkFor($fileName) {
+ $this->actor->find(self::shareActionForFile($fileName), 10)->click();
+
+ $this->actor->find(self::shareLinkCheckbox(), 5)->click();
+ }
+
+ /**
+ * @Given I write down the shared link
+ */
+ public function iWriteDownTheSharedLink() {
+ $this->actor->getSharedNotebook()["shared link"] = $this->actor->find(self::shareLinkField(), 10)->getValue();
+ }
+
+ /**
+ * @When I protect the shared link with the password :password
+ */
+ public function iProtectTheSharedLinkWithThePassword($password) {
+ $this->actor->find(self::passwordProtectCheckbox(), 10)->click();
+
+ $this->actor->find(self::passwordProtectField(), 2)->setValue($password . "\r");
+ }
+
+ /**
* @Then I see that the current page is the Files app
*/
public function iSeeThatTheCurrentPageIsTheFilesApp() {
@@ -36,4 +135,43 @@ class FilesAppContext implements Context, ActorAwareInterface {
$this->actor->getSession()->getCurrentUrl());
}
+ /**
+ * @Then I see that the working icon for password protect is shown
+ */
+ public function iSeeThatTheWorkingIconForPasswordProtectIsShown() {
+ PHPUnit_Framework_Assert::assertNotNull($this->actor->find(self::passwordProtectWorkingIcon(), 10));
+ }
+
+ /**
+ * @Then I see that the working icon for password protect is eventually not shown
+ */
+ public function iSeeThatTheWorkingIconForPasswordProtectIsEventuallyNotShown() {
+ $timeout = 10;
+ $timeoutStep = 1;
+
+ $actor = $this->actor;
+ $passwordProtectWorkingIcon = self::passwordProtectWorkingIcon();
+
+ $workingIconNotFoundCallback = function() use ($actor, $passwordProtectWorkingIcon) {
+ try {
+ return !$actor->find($passwordProtectWorkingIcon)->isVisible();
+ } catch (NoSuchElementException $exception) {
+ return true;
+ }
+ };
+ if (!Utils::waitFor($workingIconNotFoundCallback, $timeout, $timeoutStep)) {
+ PHPUnit_Framework_Assert::fail("The working icon for password protect is still shown after $timeout seconds");
+ }
+ }
+
+ /**
+ * @Given I share the link for :fileName protected by the password :password
+ */
+ public function iShareTheLinkForProtectedByThePassword($fileName, $password) {
+ $this->iShareTheLinkFor($fileName);
+ $this->iProtectTheSharedLinkWithThePassword($password);
+ $this->iSeeThatTheWorkingIconForPasswordProtectIsShown();
+ $this->iSeeThatTheWorkingIconForPasswordProtectIsEventuallyNotShown();
+ }
+
}
diff --git a/tests/acceptance/features/bootstrap/FilesSharingAppContext.php b/tests/acceptance/features/bootstrap/FilesSharingAppContext.php
new file mode 100644
index 00000000000..d9d5eca7359
--- /dev/null
+++ b/tests/acceptance/features/bootstrap/FilesSharingAppContext.php
@@ -0,0 +1,110 @@
+<?php
+
+/**
+ *
+ * @copyright Copyright (c) 2017, Daniel Calviño Sánchez (danxuliu@gmail.com)
+ *
+ * @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/>.
+ *
+ */
+
+use Behat\Behat\Context\Context;
+
+class FilesSharingAppContext implements Context, ActorAwareInterface {
+
+ use ActorAware;
+
+ /**
+ * @return Locator
+ */
+ public static function passwordField() {
+ return Locator::forThe()->field("password")->
+ describedAs("Password field in Authenticate page");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function authenticateButton() {
+ return Locator::forThe()->id("password-submit")->
+ describedAs("Authenticate button in Authenticate page");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function wrongPasswordMessage() {
+ return Locator::forThe()->content("The password is wrong. Try again.")->
+ describedAs("Wrong password message in Authenticate page");
+ }
+
+ /**
+ * @return Locator
+ */
+ public static function textPreview() {
+ return Locator::forThe()->css(".text-preview")->
+ describedAs("Text preview in Shared file page");
+ }
+
+ /**
+ * @When I visit the shared link I wrote down
+ */
+ public function iVisitTheSharedLinkIWroteDown() {
+ $this->actor->getSession()->visit($this->actor->getSharedNotebook()["shared link"]);
+ }
+
+ /**
+ * @When I authenticate with password :password
+ */
+ public function iAuthenticateWithPassword($password) {
+ $this->actor->find(self::passwordField(), 10)->setValue($password);
+ $this->actor->find(self::authenticateButton())->click();
+ }
+
+ /**
+ * @Then I see that the current page is the Authenticate page for the shared link I wrote down
+ */
+ public function iSeeThatTheCurrentPageIsTheAuthenticatePageForTheSharedLinkIWroteDown() {
+ PHPUnit_Framework_Assert::assertEquals(
+ $this->actor->getSharedNotebook()["shared link"] . "/authenticate",
+ $this->actor->getSession()->getCurrentUrl());
+ }
+
+ /**
+ * @Then I see that the current page is the shared link I wrote down
+ */
+ public function iSeeThatTheCurrentPageIsTheSharedLinkIWroteDown() {
+ PHPUnit_Framework_Assert::assertEquals(
+ $this->actor->getSharedNotebook()["shared link"],
+ $this->actor->getSession()->getCurrentUrl());
+ }
+
+ /**
+ * @Then I see that a wrong password for the shared file message is shown
+ */
+ public function iSeeThatAWrongPasswordForTheSharedFileMessageIsShown() {
+ PHPUnit_Framework_Assert::assertTrue(
+ $this->actor->find(self::wrongPasswordMessage(), 10)->isVisible());
+ }
+
+ /**
+ * @Then I see that the shared file preview shows the text :text
+ */
+ public function iSeeThatTheSharedFilePreviewShowsTheText($text) {
+ PHPUnit_Framework_Assert::assertContains($text, $this->actor->find(self::textPreview(), 10)->getText());
+ }
+
+}
diff --git a/tests/acceptance/features/core/Actor.php b/tests/acceptance/features/core/Actor.php
index a27e8e6a015..0c23b5f7a40 100644
--- a/tests/acceptance/features/core/Actor.php
+++ b/tests/acceptance/features/core/Actor.php
@@ -48,6 +48,10 @@
* before giving up without modifying the tests themselves. Note that the
* multiplier affects the timeout, but not the timeout step; the rate at which
* find() will try again to find the element does not change.
+ *
+ * All actors share a notebook in which data can be annotated. This makes
+ * possible to share data between different test steps, no matter which Actor
+ * performs them.
*/
class Actor {
@@ -67,15 +71,22 @@ class Actor {
private $findTimeoutMultiplier;
/**
+ * @var array
+ */
+ private $sharedNotebook;
+
+ /**
* Creates a new Actor.
*
* @param \Behat\Mink\Session $session the Mink Session used to control its
* web browser.
* @param string $baseUrl the base URL used when solving relative URLs.
+ * @param array $sharedNotebook the notebook shared between all actors.
*/
- public function __construct(\Behat\Mink\Session $session, $baseUrl) {
+ public function __construct(\Behat\Mink\Session $session, $baseUrl, &$sharedNotebook) {
$this->session = $session;
$this->baseUrl = $baseUrl;
+ $this->sharedNotebook = &$sharedNotebook;
$this->findTimeoutMultiplier = 1;
}
@@ -221,4 +232,13 @@ class Actor {
return $ancestorElement;
}
+ /**
+ * Returns the shared notebook of the Actors.
+ *
+ * @return array the shared notebook of the Actors.
+ */
+ public function &getSharedNotebook() {
+ return $this->sharedNotebook;
+ }
+
}
diff --git a/tests/acceptance/features/core/ActorContext.php b/tests/acceptance/features/core/ActorContext.php
index 9667ef2f01c..86fe3832f66 100644
--- a/tests/acceptance/features/core/ActorContext.php
+++ b/tests/acceptance/features/core/ActorContext.php
@@ -54,6 +54,11 @@ class ActorContext extends RawMinkContext {
private $actors;
/**
+ * @var array
+ */
+ private $sharedNotebook;
+
+ /**
* @var Actor
*/
private $currentActor;
@@ -102,8 +107,9 @@ class ActorContext extends RawMinkContext {
*/
public function initializeActors() {
$this->actors = array();
+ $this->sharedNotebook = array();
- $this->actors["default"] = new Actor($this->getSession(), $this->getMinkParameter("base_url"));
+ $this->actors["default"] = new Actor($this->getSession(), $this->getMinkParameter("base_url"), $this->sharedNotebook);
$this->actors["default"]->setFindTimeoutMultiplier($this->actorFindTimeoutMultiplier);
$this->currentActor = $this->actors["default"];
@@ -127,7 +133,7 @@ class ActorContext extends RawMinkContext {
*/
public function iActAs($actorName) {
if (!array_key_exists($actorName, $this->actors)) {
- $this->actors[$actorName] = new Actor($this->getSession($actorName), $this->getMinkParameter("base_url"));
+ $this->actors[$actorName] = new Actor($this->getSession($actorName), $this->getMinkParameter("base_url"), $this->sharedNotebook);
$this->actors[$actorName]->setFindTimeoutMultiplier($this->actorFindTimeoutMultiplier);
}
diff --git a/tests/lib/Share20/ManagerTest.php b/tests/lib/Share20/ManagerTest.php
index 7de73421d3e..13556285b61 100644
--- a/tests/lib/Share20/ManagerTest.php
+++ b/tests/lib/Share20/ManagerTest.php
@@ -2510,6 +2510,7 @@ class ManagerTest extends \Test\TestCase {
$share->setProviderId('foo')
->setId('42')
->setShareType(\OCP\Share::SHARE_TYPE_LINK)
+ ->setToken('token')
->setSharedBy('owner')
->setShareOwner('owner')
->setPassword('password')
@@ -2520,6 +2521,12 @@ class ManagerTest extends \Test\TestCase {
$manager->expects($this->once())->method('canShare')->willReturn(true);
$manager->expects($this->once())->method('getShareById')->with('foo:42')->willReturn($originalShare);
$manager->expects($this->once())->method('validateExpirationDate')->with($share);
+ $manager->expects($this->once())->method('verifyPassword')->with('password');
+
+ $this->hasher->expects($this->once())
+ ->method('hash')
+ ->with('password')
+ ->willReturn('hashed');
$this->defaultProvider->expects($this->once())
->method('update')
@@ -2536,9 +2543,94 @@ class ManagerTest extends \Test\TestCase {
]);
$hookListner2 = $this->getMockBuilder('Dummy')->setMethods(['post'])->getMock();
- \OCP\Util::connectHook('OCP\Share', 'post_update_permissions', $hookListner2, 'post');
- $hookListner2->expects($this->never())->method('post');
+ \OCP\Util::connectHook('OCP\Share', 'post_update_password', $hookListner2, 'post');
+ $hookListner2->expects($this->once())->method('post')->with([
+ 'itemType' => 'file',
+ 'itemSource' => 100,
+ 'uidOwner' => 'owner',
+ 'token' => 'token',
+ 'disabled' => false,
+ ]);
+
+ $hookListner3 = $this->getMockBuilder('Dummy')->setMethods(['post'])->getMock();
+ \OCP\Util::connectHook('OCP\Share', 'post_update_permissions', $hookListner3, 'post');
+ $hookListner3->expects($this->never())->method('post');
+
+
+ $manager->updateShare($share);
+ }
+
+ public function testUpdateShareMail() {
+ $manager = $this->createManagerMock()
+ ->setMethods([
+ 'canShare',
+ 'getShareById',
+ 'generalCreateChecks',
+ 'verifyPassword',
+ 'pathCreateChecks',
+ 'linkCreateChecks',
+ 'validateExpirationDate',
+ ])
+ ->getMock();
+
+ $originalShare = $this->manager->newShare();
+ $originalShare->setShareType(\OCP\Share::SHARE_TYPE_EMAIL)
+ ->setPermissions(\OCP\Constants::PERMISSION_ALL);
+
+ $tomorrow = new \DateTime();
+ $tomorrow->setTime(0,0,0);
+ $tomorrow->add(new \DateInterval('P1D'));
+
+ $file = $this->createMock(File::class);
+ $file->method('getId')->willReturn(100);
+
+ $share = $this->manager->newShare();
+ $share->setProviderId('foo')
+ ->setId('42')
+ ->setShareType(\OCP\Share::SHARE_TYPE_EMAIL)
+ ->setToken('token')
+ ->setSharedBy('owner')
+ ->setShareOwner('owner')
+ ->setPassword('password')
+ ->setExpirationDate($tomorrow)
+ ->setNode($file)
+ ->setPermissions(\OCP\Constants::PERMISSION_ALL);
+
+ $manager->expects($this->once())->method('canShare')->willReturn(true);
+ $manager->expects($this->once())->method('getShareById')->with('foo:42')->willReturn($originalShare);
+ $manager->expects($this->once())->method('generalCreateChecks')->with($share);
+ $manager->expects($this->once())->method('verifyPassword')->with('password');
+ $manager->expects($this->once())->method('pathCreateChecks')->with($file);
+ $manager->expects($this->never())->method('linkCreateChecks');
+ $manager->expects($this->never())->method('validateExpirationDate');
+
+ $this->hasher->expects($this->once())
+ ->method('hash')
+ ->with('password')
+ ->willReturn('hashed');
+
+ $this->defaultProvider->expects($this->once())
+ ->method('update')
+ ->with($share, 'password')
+ ->willReturn($share);
+
+ $hookListner = $this->getMockBuilder('Dummy')->setMethods(['post'])->getMock();
+ \OCP\Util::connectHook('OCP\Share', 'post_set_expiration_date', $hookListner, 'post');
+ $hookListner->expects($this->never())->method('post');
+
+ $hookListner2 = $this->getMockBuilder('Dummy')->setMethods(['post'])->getMock();
+ \OCP\Util::connectHook('OCP\Share', 'post_update_password', $hookListner2, 'post');
+ $hookListner2->expects($this->once())->method('post')->with([
+ 'itemType' => 'file',
+ 'itemSource' => 100,
+ 'uidOwner' => 'owner',
+ 'token' => 'token',
+ 'disabled' => false,
+ ]);
+ $hookListner3 = $this->getMockBuilder('Dummy')->setMethods(['post'])->getMock();
+ \OCP\Util::connectHook('OCP\Share', 'post_update_permissions', $hookListner3, 'post');
+ $hookListner3->expects($this->never())->method('post');
$manager->updateShare($share);
}