Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Bugfix: Copy / move file name exists count duplicate #7119

Merged
merged 8 commits into from
Jun 14, 2022
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion changelog/unreleased/enhancement-copy-move-conflict-dialog
Original file line number Diff line number Diff line change
Expand Up @@ -2,5 +2,6 @@ Enhancement: Copy/Move conflict dialog

We've added a conflict dialog for moving resources via drag&drop in the files list

https://github.com/owncloud/web/pull/7119
https://github.com/owncloud/web/pull/6994
https://github.com/owncloud/web/issues/6996
https://github.com/owncloud/web/issues/6996
28 changes: 22 additions & 6 deletions packages/web-app-files/src/helpers/resource/copyMove.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { Resource } from './index'
import { join } from 'path'
import { buildResource } from '../resources'

export enum ResolveStrategy {
SKIP,
Expand Down Expand Up @@ -64,19 +65,18 @@ export const resolveFileExists = (
createModal(modal)
})
}

export const resolveAllConflicts = async (
resourcesToMove,
targetFolder,
client,
targetFolderItems,
createModal,
hideModal,
$gettext,
$gettextInterpolate,
resolveFileExistsMethod,
copy = false
) => {
// if we implement MERGE, we need to use 'infinity' instead of 1
const targetFolderItems = await client.files.list(targetFolder.webDavPath, 1)
const targetPath = targetFolder.path
const index = targetFolder.webDavPath.lastIndexOf(targetPath)
const webDavPrefix =
Expand Down Expand Up @@ -215,6 +215,14 @@ export const copy = async (
true
)
}

export const resolveFileNameDuplicate = (name, existingFiles, iteration = 1) => {
const potentialName = `${name} (${iteration})`
lookacat marked this conversation as resolved.
Show resolved Hide resolved
const hasConflict = existingFiles.some((f) => f.name === potentialName)
if (!hasConflict) return potentialName
return resolveFileNameDuplicate(name, existingFiles, iteration + 1)
}

export const copyMoveResource = async (
resourcesToMove,
targetFolder,
Expand All @@ -228,10 +236,13 @@ export const copyMoveResource = async (
copy = false
) => {
const errors = []
// if we implement MERGE, we need to use 'infinity' instead of 1
const targetFolderItems = await client.files.list(targetFolder.webDavPath, 1)
const targetFolderResources = targetFolderItems.map((i) => buildResource(i))
const resolvedConflicts = await resolveAllConflicts(
resourcesToMove,
targetFolder,
client,
targetFolderItems,
createModal,
hideModal,
$gettext,
Expand All @@ -241,7 +252,9 @@ export const copyMoveResource = async (
)
const movedResources = []

for (const resource of resourcesToMove) {
for (let resource of resourcesToMove) {
// because javascripts way of cloning is awesome
lookacat marked this conversation as resolved.
Show resolved Hide resolved
resource = JSON.parse(JSON.stringify(resource))
const hasConflict = resolvedConflicts.some((e) => e.resource.id === resource.id)
let targetName = resource.name
let overwriteTarget = false
Expand All @@ -254,7 +267,10 @@ export const copyMoveResource = async (
overwriteTarget = true
}
if (resolveStrategy === ResolveStrategy.KEEP_BOTH) {
targetName = $gettextInterpolate($gettext('%{name} copy'), { name: resource.name }, true)
targetName = resolveFileNameDuplicate(resource.name, [
...movedResources,
...targetFolderResources
])
resource.name = targetName
}
}
Expand Down
60 changes: 32 additions & 28 deletions packages/web-app-files/tests/unit/helpers/resource/copyMove.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,22 @@ describe('copyMove', () => {
webDavPath: '/target'
}
})

it('should name duplicate file correctly', async () => {
lookacat marked this conversation as resolved.
Show resolved Hide resolved
const resultA = Resource.resolveFileNameDuplicate('a', [...resourcesToMove])
const resultA1 = Resource.resolveFileNameDuplicate('a', [
...resourcesToMove,
{
id: 'a1',
name: 'a (1)',
webDavPath: '/a'
}
])
const resultA11 = Resource.resolveFileNameDuplicate('a (1)', [...resourcesToMove])
expect(resultA).toEqual('a (1)')
expect(resultA1).toEqual('a (2)')
expect(resultA11).toEqual('a (1) (1)')
})
it('should copy files if no conflicts exist', async () => {
const client = {
files: {
Expand Down Expand Up @@ -70,27 +86,21 @@ describe('copyMove', () => {
expect(client.files.move).toHaveBeenCalledWith('/b', '/target/b', false)
})
it('should not show message if no conflict exists', async () => {
const client = {
files: {
list: async () => {
return [
{
id: 'c',
path: 'target/c',
webDavPath: '/target/c',
name: '/target/c'
}
]
}
const targetFolderItems = [
{
id: 'c',
path: 'target/c',
webDavPath: '/target/c',
name: '/target/c'
}
}
]
const resolveFileExistsMethod = jest
.fn()
.mockImplementation(() => Promise.resolve({ strategy: 0 } as Resource.ResolveConflict))
await Resource.resolveAllConflicts(
resourcesToMove,
targetFolder,
client,
targetFolderItems,
jest.fn(),
jest.fn(),
jest.fn(),
Expand All @@ -100,27 +110,21 @@ describe('copyMove', () => {
expect(resolveFileExistsMethod).not.toHaveBeenCalled()
})
it('should show message if conflict exists', async () => {
const client = {
files: {
list: async () => {
return [
{
id: 'a',
path: 'target/a',
webDavPath: '/target/a',
name: '/target/a'
}
]
}
const targetFolderItems = [
{
id: 'a',
path: 'target/a',
webDavPath: '/target/a',
name: '/target/a'
}
}
]
const resolveFileExistsMethod = jest
.fn()
.mockImplementation(() => Promise.resolve({ strategy: 0 } as Resource.ResolveConflict))
await Resource.resolveAllConflicts(
resourcesToMove,
targetFolder,
client,
targetFolderItems,
jest.fn(),
jest.fn(),
jest.fn(),
Expand Down