summaryrefslogtreecommitdiffstats
path: root/tests/acceptance
diff options
context:
space:
mode:
authorRoeland Jago Douma <rullzer@users.noreply.github.com>2017-04-25 14:12:44 +0200
committerGitHub <noreply@github.com>2017-04-25 14:12:44 +0200
commit82c9eb1c5654562e8057953356af49b7295a7561 (patch)
tree852acfbf17793033f932fed8da3a39b5f242c812 /tests/acceptance
parent026070a2fc3b766f9d686c50c358b6f03462ad18 (diff)
parent58cc1251be33b43a5bb9163e1b042970b8e81b4b (diff)
downloadnextcloud-server-82c9eb1c5654562e8057953356af49b7295a7561.tar.gz
nextcloud-server-82c9eb1c5654562e8057953356af49b7295a7561.zip
Merge pull request #4462 from danxuliu/fix-sharing-password-protected-link
Fix sharing a password protected link
Diffstat (limited to 'tests/acceptance')
-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
6 files changed, 309 insertions, 3 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);
}