mirror of
https://github.com/nextcloud/server.git
synced 2026-04-28 17:48:40 -04:00
Merge pull request #49840 from nextcloud/revert-49825-revert-49650-backport/49293/stable30
Revert "Revert "[stable30] fix: Handle copy of folders containing live photos""
This commit is contained in:
commit
76486b7a20
14 changed files with 327 additions and 160 deletions
4
.github/workflows/cypress.yml
vendored
4
.github/workflows/cypress.yml
vendored
|
|
@ -146,7 +146,7 @@ jobs:
|
|||
|
||||
- name: Extract NC logs
|
||||
if: failure() && matrix.containers != 'component'
|
||||
run: docker logs nextcloud-cypress-tests-${{ env.APP_NAME }} > nextcloud.log
|
||||
run: docker logs nextcloud-cypress-tests_${{ env.APP_NAME }} > nextcloud.log
|
||||
|
||||
- name: Upload NC logs
|
||||
uses: actions/upload-artifact@0b2256b8c012f0828dc542b3febcab082c67f72b # v4.3.4
|
||||
|
|
@ -157,7 +157,7 @@ jobs:
|
|||
|
||||
- name: Create data dir archive
|
||||
if: failure() && matrix.containers != 'component'
|
||||
run: docker exec nextcloud-cypress-tests-server tar -cvjf - data > data.tar
|
||||
run: docker exec nextcloud-cypress-tests_${{ env.APP_NAME }} tar -cvjf - data > data.tar
|
||||
|
||||
- name: Upload data dir archive
|
||||
uses: actions/upload-artifact@0b2256b8c012f0828dc542b3febcab082c67f72b # v4.3.4
|
||||
|
|
|
|||
|
|
@ -445,7 +445,13 @@ class Directory extends \OCA\DAV\Connector\Sabre\Node implements \Sabre\DAV\ICol
|
|||
throw new InvalidPath($ex->getMessage());
|
||||
}
|
||||
|
||||
return $this->fileView->copy($sourcePath, $destinationPath);
|
||||
$copyOkay = $this->fileView->copy($sourcePath, $destinationPath);
|
||||
|
||||
if (!$copyOkay) {
|
||||
throw new \Sabre\DAV\Exception\Forbidden('Copy did not proceed');
|
||||
}
|
||||
|
||||
return true;
|
||||
} catch (StorageNotAvailableException $e) {
|
||||
throw new ServiceUnavailable($e->getMessage(), $e->getCode(), $e);
|
||||
} catch (ForbiddenException $ex) {
|
||||
|
|
|
|||
|
|
@ -8,18 +8,23 @@ declare(strict_types=1);
|
|||
|
||||
namespace OCA\Files\Listener;
|
||||
|
||||
use Exception;
|
||||
use OC\Files\Node\NonExistingFile;
|
||||
use OC\Files\Node\NonExistingFolder;
|
||||
use OC\Files\View;
|
||||
use OC\FilesMetadata\Model\FilesMetadata;
|
||||
use OCA\Files\Service\LivePhotosService;
|
||||
use OCP\EventDispatcher\Event;
|
||||
use OCP\EventDispatcher\IEventListener;
|
||||
use OCP\Exceptions\AbortedEventException;
|
||||
use OCP\Files\Cache\CacheEntryRemovedEvent;
|
||||
use OCP\Files\Events\Node\AbstractNodesEvent;
|
||||
use OCP\Files\Events\Node\BeforeNodeCopiedEvent;
|
||||
use OCP\Files\Events\Node\BeforeNodeDeletedEvent;
|
||||
use OCP\Files\Events\Node\BeforeNodeRenamedEvent;
|
||||
use OCP\Files\Events\Node\NodeCopiedEvent;
|
||||
use OCP\Files\File;
|
||||
use OCP\Files\Folder;
|
||||
use OCP\Files\IRootFolder;
|
||||
use OCP\Files\Node;
|
||||
use OCP\Files\NotFoundException;
|
||||
use OCP\FilesMetadata\IFilesMetadataManager;
|
||||
|
|
@ -37,6 +42,8 @@ class SyncLivePhotosListener implements IEventListener {
|
|||
private ?Folder $userFolder,
|
||||
private IFilesMetadataManager $filesMetadataManager,
|
||||
private LivePhotosService $livePhotosService,
|
||||
private IRootFolder $rootFolder,
|
||||
private View $view,
|
||||
) {
|
||||
}
|
||||
|
||||
|
|
@ -45,61 +52,47 @@ class SyncLivePhotosListener implements IEventListener {
|
|||
return;
|
||||
}
|
||||
|
||||
$peerFileId = null;
|
||||
if ($event instanceof BeforeNodeCopiedEvent || $event instanceof NodeCopiedEvent) {
|
||||
$this->handleCopyRecursive($event, $event->getSource(), $event->getTarget());
|
||||
} else {
|
||||
$peerFileId = null;
|
||||
|
||||
if ($event instanceof BeforeNodeRenamedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getSource()->getId());
|
||||
} elseif ($event instanceof BeforeNodeDeletedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getNode()->getId());
|
||||
} elseif ($event instanceof CacheEntryRemovedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getFileId());
|
||||
} elseif ($event instanceof BeforeNodeCopiedEvent || $event instanceof NodeCopiedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getSource()->getId());
|
||||
}
|
||||
if ($event instanceof BeforeNodeRenamedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getSource()->getId());
|
||||
} elseif ($event instanceof BeforeNodeDeletedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getNode()->getId());
|
||||
} elseif ($event instanceof CacheEntryRemovedEvent) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($event->getFileId());
|
||||
}
|
||||
|
||||
if ($peerFileId === null) {
|
||||
return; // Not a live photo.
|
||||
}
|
||||
if ($peerFileId === null) {
|
||||
return; // Not a live photo.
|
||||
}
|
||||
|
||||
// Check the user's folder.
|
||||
$peerFile = $this->userFolder->getFirstNodeById($peerFileId);
|
||||
// Check the user's folder.
|
||||
$peerFile = $this->userFolder->getFirstNodeById($peerFileId);
|
||||
|
||||
if ($peerFile === null) {
|
||||
return; // Peer file not found.
|
||||
}
|
||||
if ($peerFile === null) {
|
||||
return; // Peer file not found.
|
||||
}
|
||||
|
||||
if ($event instanceof BeforeNodeRenamedEvent) {
|
||||
$this->handleMove($event, $peerFile, false);
|
||||
} elseif ($event instanceof BeforeNodeDeletedEvent) {
|
||||
$this->handleDeletion($event, $peerFile);
|
||||
} elseif ($event instanceof CacheEntryRemovedEvent) {
|
||||
$peerFile->delete();
|
||||
} elseif ($event instanceof BeforeNodeCopiedEvent) {
|
||||
$this->handleMove($event, $peerFile, true);
|
||||
} elseif ($event instanceof NodeCopiedEvent) {
|
||||
$this->handleCopy($event, $peerFile);
|
||||
if ($event instanceof BeforeNodeRenamedEvent) {
|
||||
$this->runMoveOrCopyChecks($event->getSource(), $event->getTarget(), $peerFile);
|
||||
$this->handleMove($event->getSource(), $event->getTarget(), $peerFile);
|
||||
} elseif ($event instanceof BeforeNodeDeletedEvent) {
|
||||
$this->handleDeletion($event, $peerFile);
|
||||
} elseif ($event instanceof CacheEntryRemovedEvent) {
|
||||
$peerFile->delete();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* During rename events, which also include move operations,
|
||||
* we rename the peer file using the same name.
|
||||
* The event listener being singleton, we can store the current state
|
||||
* of pending renames inside the 'pendingRenames' property,
|
||||
* to prevent infinite recursive.
|
||||
*/
|
||||
private function handleMove(AbstractNodesEvent $event, Node $peerFile, bool $prepForCopyOnly = false): void {
|
||||
if (!($event instanceof BeforeNodeCopiedEvent) &&
|
||||
!($event instanceof BeforeNodeRenamedEvent)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$sourceFile = $event->getSource();
|
||||
$targetFile = $event->getTarget();
|
||||
private function runMoveOrCopyChecks(Node $sourceFile, Node $targetFile, Node $peerFile): void {
|
||||
$targetParent = $targetFile->getParent();
|
||||
$sourceExtension = $sourceFile->getExtension();
|
||||
$peerFileExtension = $peerFile->getExtension();
|
||||
$targetName = $targetFile->getName();
|
||||
$peerTargetName = substr($targetName, 0, -strlen($sourceExtension)) . $peerFileExtension;
|
||||
|
||||
if (!str_ends_with($targetName, "." . $sourceExtension)) {
|
||||
throw new AbortedEventException('Cannot change the extension of a Live Photo');
|
||||
|
|
@ -111,15 +104,31 @@ class SyncLivePhotosListener implements IEventListener {
|
|||
} catch (NotFoundException) {
|
||||
}
|
||||
|
||||
$peerTargetName = substr($targetName, 0, -strlen($sourceExtension)) . $peerFileExtension;
|
||||
try {
|
||||
$targetParent->get($peerTargetName);
|
||||
throw new AbortedEventException('A file already exist at destination path of the Live Photo');
|
||||
} catch (NotFoundException) {
|
||||
if (!($targetParent instanceof NonExistingFolder)) {
|
||||
try {
|
||||
$targetParent->get($peerTargetName);
|
||||
throw new AbortedEventException('A file already exist at destination path of the Live Photo');
|
||||
} catch (NotFoundException) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* During rename events, which also include move operations,
|
||||
* we rename the peer file using the same name.
|
||||
* The event listener being singleton, we can store the current state
|
||||
* of pending renames inside the 'pendingRenames' property,
|
||||
* to prevent infinite recursive.
|
||||
*/
|
||||
private function handleMove(Node $sourceFile, Node $targetFile, Node $peerFile): void {
|
||||
$targetParent = $targetFile->getParent();
|
||||
$sourceExtension = $sourceFile->getExtension();
|
||||
$peerFileExtension = $peerFile->getExtension();
|
||||
$targetName = $targetFile->getName();
|
||||
$peerTargetName = substr($targetName, 0, -strlen($sourceExtension)) . $peerFileExtension;
|
||||
|
||||
// in case the rename was initiated from this listener, we stop right now
|
||||
if ($prepForCopyOnly || in_array($peerFile->getId(), $this->pendingRenames)) {
|
||||
if (in_array($peerFile->getId(), $this->pendingRenames)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
|
@ -130,39 +139,37 @@ class SyncLivePhotosListener implements IEventListener {
|
|||
throw new AbortedEventException($ex->getMessage());
|
||||
}
|
||||
|
||||
array_diff($this->pendingRenames, [$sourceFile->getId()]);
|
||||
$this->pendingRenames = array_diff($this->pendingRenames, [$sourceFile->getId()]);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* handle copy, we already know if it is doable from BeforeNodeCopiedEvent, so we just copy the linked file
|
||||
*
|
||||
* @param NodeCopiedEvent $event
|
||||
* @param Node $peerFile
|
||||
*/
|
||||
private function handleCopy(NodeCopiedEvent $event, Node $peerFile): void {
|
||||
$sourceFile = $event->getSource();
|
||||
private function handleCopy(File $sourceFile, File $targetFile, File $peerFile): void {
|
||||
$sourceExtension = $sourceFile->getExtension();
|
||||
$peerFileExtension = $peerFile->getExtension();
|
||||
$targetFile = $event->getTarget();
|
||||
$targetParent = $targetFile->getParent();
|
||||
$targetName = $targetFile->getName();
|
||||
$peerTargetName = substr($targetName, 0, -strlen($sourceExtension)) . $peerFileExtension;
|
||||
|
||||
/**
|
||||
* let's use freshly set variable.
|
||||
* we copy the file and get its id. We already have the id of the current copy
|
||||
* We have everything to update metadata and keep the link between the 2 copies.
|
||||
*/
|
||||
$newPeerFile = $peerFile->copy($targetParent->getPath() . '/' . $peerTargetName);
|
||||
|
||||
if ($targetParent->nodeExists($peerTargetName)) {
|
||||
// If the copy was a folder copy, then the peer file already exists.
|
||||
$targetPeerFile = $targetParent->get($peerTargetName);
|
||||
} else {
|
||||
// If the copy was a file copy, then we need to create the peer file.
|
||||
$targetPeerFile = $peerFile->copy($targetParent->getPath() . '/' . $peerTargetName);
|
||||
}
|
||||
|
||||
/** @var FilesMetadata $targetMetadata */
|
||||
$targetMetadata = $this->filesMetadataManager->getMetadata($targetFile->getId(), true);
|
||||
$targetMetadata->setStorageId($targetFile->getStorage()->getCache()->getNumericStorageId());
|
||||
$targetMetadata->setString('files-live-photo', (string)$newPeerFile->getId());
|
||||
$targetMetadata->setString('files-live-photo', (string)$targetPeerFile->getId());
|
||||
$this->filesMetadataManager->saveMetadata($targetMetadata);
|
||||
/** @var FilesMetadata $peerMetadata */
|
||||
$peerMetadata = $this->filesMetadataManager->getMetadata($newPeerFile->getId(), true);
|
||||
$peerMetadata->setStorageId($newPeerFile->getStorage()->getCache()->getNumericStorageId());
|
||||
$peerMetadata = $this->filesMetadataManager->getMetadata($targetPeerFile->getId(), true);
|
||||
$peerMetadata->setStorageId($targetPeerFile->getStorage()->getCache()->getNumericStorageId());
|
||||
$peerMetadata->setString('files-live-photo', (string)$targetFile->getId());
|
||||
$this->filesMetadataManager->saveMetadata($peerMetadata);
|
||||
}
|
||||
|
|
@ -193,4 +200,47 @@ class SyncLivePhotosListener implements IEventListener {
|
|||
}
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* Recursively get all the peer ids of a live photo.
|
||||
* Needed when coping a folder.
|
||||
*
|
||||
* @param BeforeNodeCopiedEvent|NodeCopiedEvent $event
|
||||
*/
|
||||
private function handleCopyRecursive(Event $event, Node $sourceNode, Node $targetNode): void {
|
||||
if ($sourceNode instanceof Folder && $targetNode instanceof Folder) {
|
||||
foreach ($sourceNode->getDirectoryListing() as $sourceChild) {
|
||||
if ($event instanceof BeforeNodeCopiedEvent) {
|
||||
if ($sourceChild instanceof Folder) {
|
||||
$targetChild = new NonExistingFolder($this->rootFolder, $this->view, $targetNode->getPath() . '/' . $sourceChild->getName(), null, $targetNode);
|
||||
} else {
|
||||
$targetChild = new NonExistingFile($this->rootFolder, $this->view, $targetNode->getPath() . '/' . $sourceChild->getName(), null, $targetNode);
|
||||
}
|
||||
} elseif ($event instanceof NodeCopiedEvent) {
|
||||
$targetChild = $targetNode->get($sourceChild->getName());
|
||||
} else {
|
||||
throw new Exception('Event is type is not supported');
|
||||
}
|
||||
|
||||
$this->handleCopyRecursive($event, $sourceChild, $targetChild);
|
||||
}
|
||||
} elseif ($sourceNode instanceof File && $targetNode instanceof File) {
|
||||
$peerFileId = $this->livePhotosService->getLivePhotoPeerId($sourceNode->getId());
|
||||
if ($peerFileId === null) {
|
||||
return;
|
||||
}
|
||||
$peerFile = $this->userFolder->getFirstNodeById($peerFileId);
|
||||
if ($peerFile === null) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ($event instanceof BeforeNodeCopiedEvent) {
|
||||
$this->runMoveOrCopyChecks($sourceNode, $targetNode, $peerFile);
|
||||
} elseif ($event instanceof NodeCopiedEvent) {
|
||||
$this->handleCopy($sourceNode, $targetNode, $peerFile);
|
||||
}
|
||||
} else {
|
||||
throw new Exception('Source and target type are not matching');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -11,6 +11,7 @@ namespace OCA\Files_Versions\Listener;
|
|||
|
||||
use Exception;
|
||||
use OC\Files\Node\NonExistingFile;
|
||||
use OC\Files\Node\NonExistingFolder;
|
||||
use OCA\Files_Versions\Versions\IVersionBackend;
|
||||
use OCA\Files_Versions\Versions\IVersionManager;
|
||||
use OCA\Files_Versions\Versions\IVersionsImporterBackend;
|
||||
|
|
@ -130,7 +131,7 @@ class VersionStorageMoveListener implements IEventListener {
|
|||
}
|
||||
|
||||
private function getNodeStorage(Node $node): IStorage {
|
||||
if ($node instanceof NonExistingFile) {
|
||||
if ($node instanceof NonExistingFile || $node instanceof NonExistingFolder) {
|
||||
return $node->getParent()->getStorage();
|
||||
} else {
|
||||
return $node->getStorage();
|
||||
|
|
|
|||
|
|
@ -68,6 +68,19 @@ export default defineConfig({
|
|||
|
||||
on('task', { removeDirectory })
|
||||
|
||||
// This allows to store global data (e.g. the name of a snapshot)
|
||||
// because Cypress.env() and other options are local to the current spec file.
|
||||
const data = {}
|
||||
on('task', {
|
||||
setVariable({ key, value }) {
|
||||
data[key] = value
|
||||
return null
|
||||
},
|
||||
getVariable({ key }) {
|
||||
return data[key] ?? null
|
||||
},
|
||||
})
|
||||
|
||||
// Disable spell checking to prevent rendering differences
|
||||
on('before:browser:launch', (browser, launchOptions) => {
|
||||
if (browser.family === 'chromium' && browser.name !== 'electron') {
|
||||
|
|
|
|||
|
|
@ -9,13 +9,13 @@
|
|||
import Docker from 'dockerode'
|
||||
import waitOn from 'wait-on'
|
||||
import { c as createTar } from 'tar'
|
||||
import path from 'path'
|
||||
import path, { basename } from 'path'
|
||||
import { execSync } from 'child_process'
|
||||
import { existsSync } from 'fs'
|
||||
|
||||
export const docker = new Docker()
|
||||
|
||||
const CONTAINER_NAME = 'nextcloud-cypress-tests-server'
|
||||
const CONTAINER_NAME = `nextcloud-cypress-tests_${basename(process.cwd()).replace(' ', '')}`
|
||||
const SERVER_IMAGE = 'ghcr.io/nextcloud/continuous-integration-shallow-server'
|
||||
|
||||
/**
|
||||
|
|
|
|||
|
|
@ -183,7 +183,7 @@ export const createFolder = (folderName: string) => {
|
|||
|
||||
// TODO: replace by proper data-cy selectors
|
||||
cy.get('[data-cy-upload-picker] .action-item__menutoggle').first().click()
|
||||
cy.contains('.upload-picker__menu-entry button', 'New folder').click()
|
||||
cy.get('[data-cy-upload-picker-menu-entry="newFolder"] button').click()
|
||||
cy.get('[data-cy-files-new-node-dialog]').should('be.visible')
|
||||
cy.get('[data-cy-files-new-node-dialog-input]').type(`{selectall}${folderName}`)
|
||||
cy.get('[data-cy-files-new-node-dialog-submit]').click()
|
||||
|
|
|
|||
107
cypress/e2e/files/LivePhotosUtils.ts
Normal file
107
cypress/e2e/files/LivePhotosUtils.ts
Normal file
|
|
@ -0,0 +1,107 @@
|
|||
/**
|
||||
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
|
||||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
*/
|
||||
|
||||
import type { User } from '@nextcloud/cypress'
|
||||
|
||||
type SetupInfo = {
|
||||
snapshot: string
|
||||
jpgFileId: number
|
||||
movFileId: number
|
||||
fileName: string
|
||||
user: User
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param user
|
||||
* @param fileName
|
||||
* @param domain
|
||||
* @param requesttoken
|
||||
* @param metadata
|
||||
*/
|
||||
function setMetadata(user: User, fileName: string, requesttoken: string, metadata: object) {
|
||||
cy.url().then(url => {
|
||||
const hostname = new URL(url).hostname
|
||||
cy.request({
|
||||
method: 'PROPPATCH',
|
||||
url: `http://${hostname}/remote.php/dav/files/${user.userId}/${fileName}`,
|
||||
auth: { user: user.userId, pass: user.password },
|
||||
headers: {
|
||||
requesttoken,
|
||||
},
|
||||
body: `<?xml version="1.0"?>
|
||||
<d:propertyupdate xmlns:d="DAV:" xmlns:nc="http://nextcloud.org/ns">
|
||||
<d:set>
|
||||
<d:prop>
|
||||
${Object.entries(metadata).map(([key, value]) => `<${key}>${value}</${key}>`).join('\n')}
|
||||
</d:prop>
|
||||
</d:set>
|
||||
</d:propertyupdate>`,
|
||||
})
|
||||
})
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param enable
|
||||
*/
|
||||
export function setShowHiddenFiles(enable: boolean) {
|
||||
cy.get('[data-cy-files-navigation-settings-button]').click()
|
||||
// Force:true because the checkbox is hidden by the pretty UI.
|
||||
if (enable) {
|
||||
cy.get('[data-cy-files-settings-setting="show_hidden"] input').check({ force: true })
|
||||
} else {
|
||||
cy.get('[data-cy-files-settings-setting="show_hidden"] input').uncheck({ force: true })
|
||||
}
|
||||
cy.get('[data-cy-files-navigation-settings]').type('{esc}')
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
export function setupLivePhotos(): Cypress.Chainable<SetupInfo> {
|
||||
return cy.task('getVariable', { key: 'live-photos-data' })
|
||||
.then((_setupInfo) => {
|
||||
const setupInfo = _setupInfo as SetupInfo || {}
|
||||
if (setupInfo.snapshot) {
|
||||
cy.restoreState(setupInfo.snapshot)
|
||||
} else {
|
||||
let requesttoken: string
|
||||
|
||||
setupInfo.fileName = Math.random().toString(36).replace(/[^a-z]+/g, '').substring(0, 10)
|
||||
|
||||
cy.createRandomUser().then(_user => { setupInfo.user = _user })
|
||||
|
||||
cy.then(() => {
|
||||
cy.uploadContent(setupInfo.user, new Blob(['jpg file'], { type: 'image/jpg' }), 'image/jpg', `/${setupInfo.fileName}.jpg`)
|
||||
.then(response => { setupInfo.jpgFileId = parseInt(response.headers['oc-fileid']) })
|
||||
cy.uploadContent(setupInfo.user, new Blob(['mov file'], { type: 'video/mov' }), 'video/mov', `/${setupInfo.fileName}.mov`)
|
||||
.then(response => { setupInfo.movFileId = parseInt(response.headers['oc-fileid']) })
|
||||
|
||||
cy.login(setupInfo.user)
|
||||
})
|
||||
|
||||
cy.visit('/apps/files')
|
||||
|
||||
cy.get('head').invoke('attr', 'data-requesttoken').then(_requesttoken => { requesttoken = _requesttoken as string })
|
||||
|
||||
cy.then(() => {
|
||||
setMetadata(setupInfo.user, `${setupInfo.fileName}.jpg`, requesttoken, { 'nc:metadata-files-live-photo': setupInfo.movFileId })
|
||||
setMetadata(setupInfo.user, `${setupInfo.fileName}.mov`, requesttoken, { 'nc:metadata-files-live-photo': setupInfo.jpgFileId })
|
||||
})
|
||||
|
||||
cy.then(() => {
|
||||
cy.saveState().then((value) => { setupInfo.snapshot = value })
|
||||
cy.task('setVariable', { key: 'live-photos-data', value: setupInfo })
|
||||
})
|
||||
}
|
||||
return cy.then(() => {
|
||||
cy.login(setupInfo.user)
|
||||
cy.visit('/apps/files')
|
||||
return cy.wrap(setupInfo)
|
||||
})
|
||||
})
|
||||
}
|
||||
|
|
@ -4,75 +4,34 @@
|
|||
*/
|
||||
|
||||
import type { User } from '@nextcloud/cypress'
|
||||
import { clickOnBreadcrumbs, closeSidebar, copyFile, getRowForFile, getRowForFileId, renameFile, triggerActionForFile, triggerInlineActionForFileId } from './FilesUtils'
|
||||
|
||||
/**
|
||||
*
|
||||
* @param user
|
||||
* @param fileName
|
||||
* @param domain
|
||||
* @param requesttoken
|
||||
* @param metadata
|
||||
*/
|
||||
function setMetadata(user: User, fileName: string, domain: string, requesttoken: string, metadata: object) {
|
||||
cy.request({
|
||||
method: 'PROPPATCH',
|
||||
url: `http://${domain}/remote.php/dav/files/${user.userId}/${fileName}`,
|
||||
auth: { user: user.userId, pass: user.password },
|
||||
headers: {
|
||||
requesttoken,
|
||||
},
|
||||
body: `<?xml version="1.0"?>
|
||||
<d:propertyupdate xmlns:d="DAV:" xmlns:nc="http://nextcloud.org/ns">
|
||||
<d:set>
|
||||
<d:prop>
|
||||
${Object.entries(metadata).map(([key, value]) => `<${key}>${value}</${key}>`).join('\n')}
|
||||
</d:prop>
|
||||
</d:set>
|
||||
</d:propertyupdate>`,
|
||||
})
|
||||
}
|
||||
import {
|
||||
clickOnBreadcrumbs,
|
||||
copyFile,
|
||||
createFolder,
|
||||
getRowForFile,
|
||||
getRowForFileId,
|
||||
moveFile,
|
||||
navigateToFolder,
|
||||
renameFile,
|
||||
triggerActionForFile,
|
||||
triggerInlineActionForFileId,
|
||||
} from './FilesUtils'
|
||||
import { setShowHiddenFiles, setupLivePhotos } from './LivePhotosUtils'
|
||||
|
||||
describe('Files: Live photos', { testIsolation: true }, () => {
|
||||
let currentUser: User
|
||||
let user: User
|
||||
let randomFileName: string
|
||||
let jpgFileId: number
|
||||
let movFileId: number
|
||||
let hostname: string
|
||||
let requesttoken: string
|
||||
|
||||
before(() => {
|
||||
cy.createRandomUser().then((user) => {
|
||||
currentUser = user
|
||||
cy.login(currentUser)
|
||||
cy.visit('/apps/files')
|
||||
})
|
||||
|
||||
cy.url().then(url => { hostname = new URL(url).hostname })
|
||||
})
|
||||
|
||||
beforeEach(() => {
|
||||
randomFileName = Math.random().toString(36).replace(/[^a-z]+/g, '').substring(0, 10)
|
||||
|
||||
cy.uploadContent(currentUser, new Blob(['jpg file'], { type: 'image/jpg' }), 'image/jpg', `/${randomFileName}.jpg`)
|
||||
.then(response => { jpgFileId = parseInt(response.headers['oc-fileid']) })
|
||||
cy.uploadContent(currentUser, new Blob(['mov file'], { type: 'video/mov' }), 'video/mov', `/${randomFileName}.mov`)
|
||||
.then(response => { movFileId = parseInt(response.headers['oc-fileid']) })
|
||||
|
||||
cy.login(currentUser)
|
||||
cy.visit('/apps/files')
|
||||
|
||||
cy.get('head').invoke('attr', 'data-requesttoken').then(_requesttoken => { requesttoken = _requesttoken as string })
|
||||
|
||||
cy.then(() => {
|
||||
setMetadata(currentUser, `${randomFileName}.jpg`, hostname, requesttoken, { 'nc:metadata-files-live-photo': movFileId })
|
||||
setMetadata(currentUser, `${randomFileName}.mov`, hostname, requesttoken, { 'nc:metadata-files-live-photo': jpgFileId })
|
||||
})
|
||||
|
||||
cy.then(() => {
|
||||
cy.visit(`/apps/files/files/${jpgFileId}`) // Refresh and scroll to the .jpg file.
|
||||
closeSidebar()
|
||||
})
|
||||
setupLivePhotos()
|
||||
.then((setupInfo) => {
|
||||
user = setupInfo.user
|
||||
randomFileName = setupInfo.fileName
|
||||
jpgFileId = setupInfo.jpgFileId
|
||||
movFileId = setupInfo.movFileId
|
||||
})
|
||||
})
|
||||
|
||||
it('Only renders the .jpg file', () => {
|
||||
|
|
@ -81,12 +40,8 @@ describe('Files: Live photos', { testIsolation: true }, () => {
|
|||
})
|
||||
|
||||
context("'Show hidden files' is enabled", () => {
|
||||
before(() => {
|
||||
cy.login(currentUser)
|
||||
cy.visit('/apps/files')
|
||||
cy.get('[data-cy-files-navigation-settings-button]').click()
|
||||
// Force:true because the checkbox is hidden by the pretty UI.
|
||||
cy.get('[data-cy-files-settings-setting="show_hidden"] input').check({ force: true })
|
||||
beforeEach(() => {
|
||||
setShowHiddenFiles(true)
|
||||
})
|
||||
|
||||
it("Shows both files when 'Show hidden files' is enabled", () => {
|
||||
|
|
@ -113,6 +68,35 @@ describe('Files: Live photos', { testIsolation: true }, () => {
|
|||
getRowForFile(`${randomFileName} (copy).mov`).should('have.length', 1)
|
||||
})
|
||||
|
||||
it('Keeps live photo link when copying folder', () => {
|
||||
createFolder('folder')
|
||||
moveFile(`${randomFileName}.jpg`, 'folder')
|
||||
copyFile('folder', '.')
|
||||
navigateToFolder('folder (copy)')
|
||||
|
||||
getRowForFile(`${randomFileName}.jpg`).should('have.length', 1)
|
||||
getRowForFile(`${randomFileName}.mov`).should('have.length', 1)
|
||||
|
||||
setShowHiddenFiles(false)
|
||||
|
||||
getRowForFile(`${randomFileName}.jpg`).should('have.length', 1)
|
||||
getRowForFile(`${randomFileName}.mov`).should('have.length', 0)
|
||||
})
|
||||
|
||||
it('Block copying live photo in a folder containing a mov file with the same name', () => {
|
||||
createFolder('folder')
|
||||
cy.uploadContent(user, new Blob(['mov file'], { type: 'video/mov' }), 'video/mov', `/folder/${randomFileName}.mov`)
|
||||
cy.login(user)
|
||||
cy.visit('/apps/files')
|
||||
copyFile(`${randomFileName}.jpg`, 'folder')
|
||||
navigateToFolder('folder')
|
||||
|
||||
cy.get('[data-cy-files-list-row-fileid]').should('have.length', 1)
|
||||
getRowForFile(`${randomFileName}.mov`).should('have.length', 1)
|
||||
getRowForFile(`${randomFileName}.jpg`).should('have.length', 0)
|
||||
getRowForFile(`${randomFileName} (copy).jpg`).should('have.length', 0)
|
||||
})
|
||||
|
||||
it('Moves files when moving the .jpg', () => {
|
||||
renameFile(`${randomFileName}.jpg`, `${randomFileName}_moved.jpg`)
|
||||
clickOnBreadcrumbs('All files')
|
||||
|
|
|
|||
|
|
@ -21,7 +21,7 @@ describe('Versions expiration', () => {
|
|||
})
|
||||
|
||||
it('Expire all versions', () => {
|
||||
cy.runOccCommand('config:system:set versions_retention_obligation --value "0, 0"')
|
||||
cy.runOccCommand("config:system:set versions_retention_obligation --value '0, 0'")
|
||||
cy.runOccCommand('versions:expire')
|
||||
cy.runOccCommand('config:system:set versions_retention_obligation --value auto')
|
||||
cy.visit('/apps/files')
|
||||
|
|
@ -38,7 +38,7 @@ describe('Versions expiration', () => {
|
|||
it('Expire versions v2', () => {
|
||||
nameVersion(2, 'v1')
|
||||
|
||||
cy.runOccCommand('config:system:set versions_retention_obligation --value "0, 0"')
|
||||
cy.runOccCommand("config:system:set versions_retention_obligation --value '0, 0'")
|
||||
cy.runOccCommand('versions:expire')
|
||||
cy.runOccCommand('config:system:set versions_retention_obligation --value auto')
|
||||
cy.visit('/apps/files')
|
||||
|
|
|
|||
|
|
@ -110,7 +110,7 @@ describe('Accessibility of Nextcloud theming colors', () => {
|
|||
before(() => {
|
||||
cy.createRandomUser().then(($user) => {
|
||||
// set user theme
|
||||
cy.runOccCommand(`user:setting -- '${$user.userId}' theming enabled-themes '["${theme}"]'`)
|
||||
cy.runOccCommand(`user:setting -- '${$user.userId}' theming enabled-themes '[\\"${theme}\\"]'`)
|
||||
cy.login($user)
|
||||
cy.visit('/')
|
||||
cy.injectAxe({ axeCorePath: 'node_modules/axe-core/axe.min.js' })
|
||||
|
|
|
|||
|
|
@ -60,11 +60,6 @@ declare global {
|
|||
* **Warning**: Providing a user will reset the previous session.
|
||||
*/
|
||||
resetUserTheming(user?: User): Cypress.Chainable<void>,
|
||||
|
||||
/**
|
||||
* Run an occ command in the docker container.
|
||||
*/
|
||||
runOccCommand(command: string, options?: Partial<Cypress.ExecOptions>): Cypress.Chainable<Cypress.Exec>,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -295,8 +290,3 @@ Cypress.Commands.add('resetUserTheming', (user?: User) => {
|
|||
cy.clearCookies()
|
||||
}
|
||||
})
|
||||
|
||||
Cypress.Commands.add('runOccCommand', (command: string, options?: Partial<Cypress.ExecOptions>) => {
|
||||
const env = Object.entries(options?.env ?? {}).map(([name, value]) => `-e '${name}=${value}'`).join(' ')
|
||||
return cy.exec(`docker exec --user www-data ${env} nextcloud-cypress-tests-server php ./occ ${command}`, options)
|
||||
})
|
||||
|
|
|
|||
|
|
@ -3,6 +3,8 @@
|
|||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
*/
|
||||
|
||||
import { basename } from 'path'
|
||||
|
||||
/**
|
||||
* Get the header navigation bar
|
||||
*/
|
||||
|
|
@ -49,8 +51,12 @@ export function installTestApp() {
|
|||
cy.runOccCommand('-V').then((output) => {
|
||||
const version = output.stdout.match(/(\d\d+)\.\d+\.\d+/)?.[1]
|
||||
cy.wrap(version).should('not.be.undefined')
|
||||
cy.exec(`docker cp '${testAppPath}' nextcloud-cypress-tests-server:/var/www/html/apps`, { log: true })
|
||||
cy.exec(`docker exec nextcloud-cypress-tests-server sed -i -e 's|-version="[0-9]\\+|-version="${version}|g' apps/testapp/appinfo/info.xml`)
|
||||
getContainerName()
|
||||
.then(containerName => {
|
||||
cy.exec(`docker cp '${testAppPath}' ${containerName}:/var/www/html/apps`, { log: true })
|
||||
cy.exec(`docker exec --workdir /var/www/html ${containerName} chown -R www-data:www-data /var/www/html/apps/testapp`)
|
||||
})
|
||||
cy.runCommand(`sed -i -e 's|-version=\\"[0-9]\\+|-version=\\"${version}|g' apps/testapp/appinfo/info.xml`)
|
||||
cy.runOccCommand('app:enable --force testapp')
|
||||
})
|
||||
}
|
||||
|
|
@ -60,5 +66,15 @@ export function installTestApp() {
|
|||
*/
|
||||
export function uninstallTestApp() {
|
||||
cy.runOccCommand('app:remove testapp', { failOnNonZeroExit: false })
|
||||
cy.exec('docker exec nextcloud-cypress-tests-server rm -fr apps/testapp/appinfo/info.xml')
|
||||
cy.runCommand('rm -fr apps/testapp/appinfo/info.xml')
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
export function getContainerName(): Cypress.Chainable<string> {
|
||||
return cy.exec('pwd')
|
||||
.then(({ stdout }) => {
|
||||
return cy.wrap(`nextcloud-cypress-tests_${basename(stdout).replace(' ', '')}`)
|
||||
})
|
||||
}
|
||||
|
|
|
|||
|
|
@ -171,7 +171,7 @@ class HookConnector {
|
|||
|
||||
public function copy($arguments) {
|
||||
$source = $this->getNodeForPath($arguments['oldpath']);
|
||||
$target = $this->getNodeForPath($arguments['newpath']);
|
||||
$target = $this->getNodeForPath($arguments['newpath'], $source instanceof Folder);
|
||||
$this->root->emit('\OC\Files', 'preCopy', [$source, $target]);
|
||||
$this->dispatcher->dispatch('\OCP\Files::preCopy', new GenericEvent([$source, $target]));
|
||||
|
||||
|
|
@ -203,7 +203,7 @@ class HookConnector {
|
|||
$this->dispatcher->dispatchTyped($event);
|
||||
}
|
||||
|
||||
private function getNodeForPath(string $path): Node {
|
||||
private function getNodeForPath(string $path, bool $isDir = false): Node {
|
||||
$info = Filesystem::getView()->getFileInfo($path);
|
||||
if (!$info) {
|
||||
$fullPath = Filesystem::getView()->getAbsolutePath($path);
|
||||
|
|
@ -212,7 +212,7 @@ class HookConnector {
|
|||
} else {
|
||||
$info = null;
|
||||
}
|
||||
if (Filesystem::is_dir($path)) {
|
||||
if ($isDir || Filesystem::is_dir($path)) {
|
||||
return new NonExistingFolder($this->root, $this->view, $fullPath, $info);
|
||||
} else {
|
||||
return new NonExistingFile($this->root, $this->view, $fullPath, $info);
|
||||
|
|
|
|||
Loading…
Reference in a new issue