mirror of
https://github.com/owncloud/ocis.git
synced 2025-04-18 23:44:07 +03:00
542 lines
15 KiB
PHP
542 lines
15 KiB
PHP
<?php declare(strict_types=1);
|
|
/**
|
|
* ownCloud
|
|
*
|
|
* @author Artur Neumann <artur@jankaritech.com>
|
|
* @copyright Copyright (c) 2018, ownCloud GmbH
|
|
*
|
|
* This code 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 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;
|
|
use Behat\Behat\Hook\Scope\BeforeScenarioScope;
|
|
use Behat\Gherkin\Node\TableNode;
|
|
use PHPUnit\Framework\Assert;
|
|
use TestHelpers\HttpRequestHelper;
|
|
use TestHelpers\WebDavHelper;
|
|
use Psr\Http\Message\ResponseInterface;
|
|
|
|
require_once 'bootstrap.php';
|
|
|
|
/**
|
|
* Steps that relate to files_versions app
|
|
*/
|
|
class FilesVersionsContext implements Context {
|
|
private FeatureContext $featureContext;
|
|
|
|
/**
|
|
* @param string $fileId
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getVersionsPathForFileId(string $fileId):string {
|
|
return "/meta/$fileId/v";
|
|
}
|
|
|
|
/**
|
|
* @When user :user tries to get versions of file :file from :fileOwner
|
|
*
|
|
* @param string $user
|
|
* @param string $file
|
|
* @param string $fileOwner
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userTriesToGetFileVersions(string $user, string $file, string $fileOwner):void {
|
|
$this->featureContext->setResponse($this->getFileVersions($user, $file, $fileOwner));
|
|
}
|
|
|
|
/**
|
|
* @When user :user gets the number of versions of file :file
|
|
*
|
|
* @param string $user
|
|
* @param string $file
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userGetsFileVersions(string $user, string $file):void {
|
|
$this->featureContext->setResponse($this->getFileVersions($user, $file));
|
|
}
|
|
|
|
/**
|
|
* @param string $user
|
|
* @param string $file
|
|
* @param string|null $fileOwner
|
|
*
|
|
* @return ResponseInterface
|
|
* @throws JsonException
|
|
* @throws GuzzleException
|
|
*/
|
|
public function getFileVersions(
|
|
string $user,
|
|
string $file,
|
|
?string $fileOwner = null
|
|
): ResponseInterface {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileOwner = $fileOwner ? $this->featureContext->getActualUsername($fileOwner) : $user;
|
|
$fileId = $this->featureContext->getFileIdForPath($fileOwner, $file);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " fileid of file $file user $fileOwner not found (the file may not exist)");
|
|
return $this->featureContext->makeDavRequest(
|
|
$user,
|
|
"PROPFIND",
|
|
$this->getVersionsPathForFileId($fileId),
|
|
null,
|
|
null,
|
|
null,
|
|
'2'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @When user :user gets the number of versions of file :resource using file-id path :endpoint
|
|
* @When user :user tries to get the number of versions of file :resource using file-id path :endpoint
|
|
*
|
|
* @param string $user
|
|
* @param string $endpoint
|
|
*
|
|
* @return void
|
|
*/
|
|
public function userGetsTheNumberOfVersionsOfFileOfTheSpace(string $user, string $endpoint):void {
|
|
$this->featureContext->setResponse(
|
|
$this->featureContext->makeDavRequest(
|
|
$user,
|
|
"PROPFIND",
|
|
$endpoint,
|
|
null,
|
|
null,
|
|
"versions",
|
|
(string)$this->featureContext->getDavPathVersion()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @When user :user gets the version metadata of file :file
|
|
*
|
|
* @param string $user
|
|
* @param string $file
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userGetsVersionMetadataOfFile(string $user, string $file):void {
|
|
$response = $this->getFileVersionMetadata($user, $file);
|
|
$this->featureContext->setResponse($response, $user);
|
|
}
|
|
|
|
/**
|
|
* @param string $user
|
|
* @param string $file
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
public function getFileVersionMetadata(string $user, string $file) : ResponseInterface {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $file);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " fileid of file $file user $user not found (the file may not exist)");
|
|
$body = '<?xml version="1.0"?>
|
|
<d:propfind xmlns:d="DAV:" xmlns:oc="http://owncloud.org/ns">
|
|
<d:prop>
|
|
<oc:meta-version-edited-by />
|
|
<oc:meta-version-edited-by-name />
|
|
</d:prop>
|
|
</d:propfind>';
|
|
return $this->featureContext->makeDavRequest(
|
|
$user,
|
|
"PROPFIND",
|
|
$this->getVersionsPathForFileId($fileId),
|
|
null,
|
|
$body,
|
|
null,
|
|
'2'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param string $user
|
|
* @param int $versionIndex
|
|
* @param string $path
|
|
*
|
|
* @return ResponseInterface
|
|
* @throws Exception
|
|
*/
|
|
public function restoreVersionIndexOfFile(string $user, int $versionIndex, string $path):ResponseInterface {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $path);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " fileid of file $path user $user not found (the file may not exist)");
|
|
$response = $this->listVersionFolder($user, $fileId, 1);
|
|
$responseXml = HttpRequestHelper::getResponseXml(
|
|
$response,
|
|
__METHOD__
|
|
);
|
|
$xmlPart = $responseXml->xpath("//d:response/d:href");
|
|
//restoring the version only works with DAV path v2
|
|
$destinationUrl = $this->featureContext->getBaseUrl() . "/" .
|
|
WebDavHelper::getDavPath($user, 2) . \trim($path, "/");
|
|
$fullUrl = $this->featureContext->getBaseUrlWithoutPath() .
|
|
$xmlPart[$versionIndex];
|
|
return HttpRequestHelper::sendRequest(
|
|
$fullUrl,
|
|
$this->featureContext->getStepLineRef(),
|
|
'COPY',
|
|
$user,
|
|
$this->featureContext->getPasswordForUser($user),
|
|
['Destination' => $destinationUrl]
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @Given user :user has restored version index :versionIndex of file :path
|
|
*
|
|
* @param string $user
|
|
* @param int $versionIndex
|
|
* @param string $path
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userHasRestoredVersionIndexOfFile(string $user, int $versionIndex, string $path):void {
|
|
$response = $this->restoreVersionIndexOfFile($user, $versionIndex, $path);
|
|
$this->featureContext->theHTTPStatusCodeShouldBe(204, "", $response);
|
|
}
|
|
|
|
/**
|
|
* @When user :user restores version index :versionIndex of file :path using the WebDAV API
|
|
*
|
|
* @param string $user
|
|
* @param int $versionIndex
|
|
* @param string $path
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userRestoresVersionIndexOfFile(string $user, int $versionIndex, string $path):void {
|
|
$response = $this->restoreVersionIndexOfFile($user, $versionIndex, $path);
|
|
$this->featureContext->setResponse($response, $user);
|
|
}
|
|
|
|
/**
|
|
* assert file versions count
|
|
*
|
|
* @param string $user
|
|
* @param string $fileId
|
|
* @param int $expectedCount
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function assertFileVersionsCount(string $user, string $fileId, int $expectedCount):void {
|
|
$response = $this->listVersionFolder($user, $fileId, 1);
|
|
$responseXml = HttpRequestHelper::getResponseXml(
|
|
$response,
|
|
__METHOD__
|
|
);
|
|
$actualCount = \count($responseXml->xpath("//d:prop/d:getetag")) - 1;
|
|
if ($actualCount === -1) {
|
|
$actualCount = 0;
|
|
}
|
|
Assert::assertEquals(
|
|
$expectedCount,
|
|
$actualCount,
|
|
"Expected $expectedCount versions but found $actualCount in \n" . $responseXml->asXML()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @Then the version folder of file :path for user :user should contain :count element(s)
|
|
*
|
|
* @param string $path
|
|
* @param string $user
|
|
* @param int $count
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function theVersionFolderOfFileShouldContainElements(
|
|
string $path,
|
|
string $user,
|
|
int $count
|
|
):void {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $path);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " file '$path' for user '$user' not found (the file may not exist)");
|
|
$this->assertFileVersionsCount($user, $fileId, $count);
|
|
}
|
|
|
|
/**
|
|
* @Then the version folder of fileId :fileId for user :user should contain :count element(s)
|
|
*
|
|
* @param string $fileId
|
|
* @param string $user
|
|
* @param int $count
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function theVersionFolderOfFileIdShouldContainElements(
|
|
string $fileId,
|
|
string $user,
|
|
int $count
|
|
):void {
|
|
$this->assertFileVersionsCount($user, $fileId, $count);
|
|
}
|
|
|
|
/**
|
|
* @Then the content length of file :path with version index :index for user :user in versions folder should be :length
|
|
*
|
|
* @param string $path
|
|
* @param int $index
|
|
* @param string $user
|
|
* @param int $length
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function theContentLengthOfFileForUserInVersionsFolderIs(
|
|
string $path,
|
|
int $index,
|
|
string $user,
|
|
int $length
|
|
):void {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $path);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " fileid of file $path user $user not found (the file may not exist)");
|
|
$response = $this->listVersionFolder($user, $fileId, 1, ['d:getcontentlength']);
|
|
$responseXml = HttpRequestHelper::getResponseXml(
|
|
$response,
|
|
__METHOD__
|
|
);
|
|
$xmlPart = $responseXml->xpath("//d:prop/d:getcontentlength");
|
|
Assert::assertEquals(
|
|
$length,
|
|
(int) $xmlPart[$index],
|
|
"The content length of file $path with version $index for user $user was
|
|
expected to be $length but the actual content length is $xmlPart[$index]"
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @Then /^as (?:users|user) "([^"]*)" the authors of the versions of file "([^"]*)" should be:$/
|
|
*
|
|
* @param string $users comma-separated list of usernames
|
|
* @param string $filename
|
|
* @param TableNode $table
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function asUsersAuthorsOfVersionsOfFileShouldBe(
|
|
string $users,
|
|
string $filename,
|
|
TableNode $table
|
|
): void {
|
|
$this->featureContext->verifyTableNodeColumns(
|
|
$table,
|
|
['index', 'author']
|
|
);
|
|
$requiredVersionMetadata = $table->getHash();
|
|
$usersArray = \explode(",", $users);
|
|
foreach ($usersArray as $username) {
|
|
$actualUsername = $this->featureContext->getActualUsername($username);
|
|
$this->featureContext->setResponse(
|
|
$this->getFileVersionMetadata($actualUsername, $filename)
|
|
);
|
|
foreach ($requiredVersionMetadata as $versionMetadata) {
|
|
$this->featureContext->checkAuthorOfAVersionOfFile(
|
|
$versionMetadata['index'],
|
|
$versionMetadata['author']
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param string $user
|
|
* @param string $path
|
|
* @param string $index
|
|
*
|
|
* @return ResponseInterface
|
|
* @throws Exception
|
|
*/
|
|
public function downloadVersion(string $user, string $path, string $index):ResponseInterface {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $path);
|
|
Assert::assertNotNull($fileId, __METHOD__ . " fileid of file $path user $user not found (the file may not exist)");
|
|
$index = (int)$index;
|
|
$response = $this->listVersionFolder($user, $fileId, 1);
|
|
if ($response->getStatusCode() === 403) {
|
|
return $response;
|
|
}
|
|
$responseXml = new SimpleXMLElement($response->getBody()->getContents());
|
|
$xmlPart = $responseXml->xpath("//d:response/d:href");
|
|
if (!isset($xmlPart[$index])) {
|
|
Assert::fail(
|
|
'could not find version of path "' . $path . '" with index "' . $index . '"'
|
|
);
|
|
}
|
|
// the href already contains the path
|
|
$url = WebDavHelper::sanitizeUrl(
|
|
$this->featureContext->getBaseUrlWithoutPath() . $xmlPart[$index]
|
|
);
|
|
return HttpRequestHelper::get(
|
|
$url,
|
|
$this->featureContext->getStepLineRef(),
|
|
$user,
|
|
$this->featureContext->getPasswordForUser($user)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @When user :user downloads the version of file :path with the index :index
|
|
*
|
|
* @param string $user
|
|
* @param string $path
|
|
* @param string $index
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function userDownloadsVersion(string $user, string $path, string $index):void {
|
|
$this->featureContext->setResponse($this->downloadVersion($user, $path, $index), $user);
|
|
}
|
|
|
|
/**
|
|
* @Then /^the content of version index "([^"]*)" of file "([^"]*)" for user "([^"]*)" should be "([^"]*)"$/
|
|
*
|
|
* @param string $index
|
|
* @param string $path
|
|
* @param string $user
|
|
* @param string $content
|
|
*
|
|
* @return void
|
|
* @throws Exception
|
|
*/
|
|
public function theContentOfVersionIndexOfFileForUserShouldBe(
|
|
string $index,
|
|
string $path,
|
|
string $user,
|
|
string $content
|
|
): void {
|
|
$response = $this->downloadVersion($user, $path, $index);
|
|
$this->featureContext->theHTTPStatusCodeShouldBe("200", '', $response);
|
|
$this->featureContext->checkDownloadedContentMatches($content, '', $response);
|
|
}
|
|
|
|
/**
|
|
* @When /^user "([^"]*)" retrieves the meta information of (file|fileId) "([^"]*)" using the meta API$/
|
|
*
|
|
* @param string $user
|
|
* @param string $fileOrFileId
|
|
* @param string $path
|
|
*
|
|
* @return void
|
|
*/
|
|
public function userGetMetaInfo(string $user, string $fileOrFileId, string $path):void {
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$baseUrl = $this->featureContext->getBaseUrl();
|
|
$password = $this->featureContext->getPasswordForUser($user);
|
|
|
|
if ($fileOrFileId === "file") {
|
|
$fileId = $this->featureContext->getFileIdForPath($user, $path);
|
|
$metaPath = "/meta/$fileId/";
|
|
} else {
|
|
$metaPath = "/meta/$path/";
|
|
}
|
|
|
|
$body = '<?xml version="1.0" encoding="utf-8"?>
|
|
<a:propfind xmlns:a="DAV:" xmlns:oc="http://owncloud.org/ns">
|
|
<a:prop>
|
|
<oc:meta-path-for-user />
|
|
</a:prop>
|
|
</a:propfind>';
|
|
|
|
$response = WebDavHelper::makeDavRequest(
|
|
$baseUrl,
|
|
$user,
|
|
$password,
|
|
"PROPFIND",
|
|
$metaPath,
|
|
['Content-Type' => 'text/xml','Depth' => '0'],
|
|
$this->featureContext->getStepLineRef(),
|
|
$body,
|
|
$this->featureContext->getDavPathVersion(),
|
|
null
|
|
);
|
|
$this->featureContext->setResponse($response);
|
|
$responseXml = HttpRequestHelper::getResponseXml(
|
|
$response,
|
|
__METHOD__
|
|
);
|
|
$this->featureContext->setResponseXmlObject($responseXml);
|
|
}
|
|
|
|
/**
|
|
* returns the result parsed into a SimpleXMLElement
|
|
* with a registered namespace with 'd' as prefix and 'DAV:' as namespace
|
|
*
|
|
* @param string $user
|
|
* @param string $fileId
|
|
* @param int $folderDepth
|
|
* @param string[]|null $properties
|
|
*
|
|
* @return ResponseInterface
|
|
* @throws Exception
|
|
*/
|
|
public function listVersionFolder(
|
|
string $user,
|
|
string $fileId,
|
|
int $folderDepth,
|
|
?array $properties = null
|
|
):ResponseInterface {
|
|
if (!$properties) {
|
|
$properties = [
|
|
'd:getetag'
|
|
];
|
|
}
|
|
$user = $this->featureContext->getActualUsername($user);
|
|
$password = $this->featureContext->getPasswordForUser($user);
|
|
$response = WebDavHelper::propfind(
|
|
$this->featureContext->getBaseUrl(),
|
|
$user,
|
|
$password,
|
|
$this->getVersionsPathForFileId($fileId),
|
|
$properties,
|
|
$this->featureContext->getStepLineRef(),
|
|
(string) $folderDepth,
|
|
"versions"
|
|
);
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* This will run before EVERY scenario.
|
|
* It will set the properties for this object.
|
|
*
|
|
* @BeforeScenario
|
|
*
|
|
* @param BeforeScenarioScope $scope
|
|
*
|
|
* @return void
|
|
*/
|
|
public function before(BeforeScenarioScope $scope):void {
|
|
// Get the environment
|
|
$environment = $scope->getEnvironment();
|
|
// Get all the contexts you need in this context
|
|
$this->featureContext = $environment->getContext('FeatureContext');
|
|
}
|
|
}
|