From 82d3d25e3b57e0c4bfd567b8a0faab582dc26e8f Mon Sep 17 00:00:00 2001 From: David Kocher Date: Wed, 20 Nov 2024 10:55:12 +0100 Subject: [PATCH 1/5] Flip conditions. --- webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java | 2 +- .../src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java | 2 +- webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java index ab5c4aeef62..64b10854ee5 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java @@ -48,7 +48,7 @@ public DAVCopyFeature(final DAVSession session) { public Path copy(final Path source, final Path copy, final TransferStatus status, final ConnectionCallback callback, final StreamListener listener) throws BackgroundException { try { final String target = new DefaultUrlProvider(session.getHost()).toUrl(copy).find(DescriptiveUrl.Type.provider).getUrl(); - if(session.getFeature(Lock.class) != null && status.getLockId() != null) { + if(status.getLockId() != null && session.getFeature(Lock.class) != null) { // Indicate that the client has knowledge of that state token session.getClient().copy(new DAVPathEncoder().encode(source), target, status.isExists(), Collections.singletonMap(HttpHeaders.IF, String.format("(<%s>)", status.getLockId()))); diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java index 3567b231f89..7dffbaa132f 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java @@ -79,7 +79,7 @@ public void delete(final Map files, final PasswordCallback protected HttpRequestBase toRequest(final Path file, final TransferStatus status) { final HttpDelete request = new HttpDelete(new DAVPathEncoder().encode(file)); - if(session.getFeature(Lock.class) != null && status.getLockId() != null) { + if(status.getLockId() != null && session.getFeature(Lock.class) != null) { // Indicate that the client has knowledge of that state token request.setHeader(HttpHeaders.IF, String.format("(<%s>)", status.getLockId())); } diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java index 80295972af0..f81ab238618 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java @@ -49,7 +49,7 @@ public DAVMoveFeature(final DAVSession session) { public Path move(final Path file, final Path renamed, final TransferStatus status, final Delete.Callback callback, final ConnectionCallback connectionCallback) throws BackgroundException { try { final String target = new DefaultUrlProvider(session.getHost()).toUrl(renamed).find(DescriptiveUrl.Type.provider).getUrl(); - if(session.getFeature(Lock.class) != null && status.getLockId() != null) { + if(status.getLockId() != null && session.getFeature(Lock.class) != null) { // Indicate that the client has knowledge of that state token session.getClient().move(new DAVPathEncoder().encode(file), file.isDirectory() ? String.format("%s/", target) : target, status.isExists(), From 157855d69cc22a6076a9e6654a22b6a69ae87a2a Mon Sep 17 00:00:00 2001 From: David Kocher Date: Wed, 20 Nov 2024 11:09:48 +0100 Subject: [PATCH 2/5] Include resource in If: header. --- webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java | 2 +- .../src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java | 2 +- webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java index 64b10854ee5..6e645d0df4c 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java @@ -51,7 +51,7 @@ public Path copy(final Path source, final Path copy, final TransferStatus status if(status.getLockId() != null && session.getFeature(Lock.class) != null) { // Indicate that the client has knowledge of that state token session.getClient().copy(new DAVPathEncoder().encode(source), target, status.isExists(), - Collections.singletonMap(HttpHeaders.IF, String.format("(<%s>)", status.getLockId()))); + Collections.singletonMap(HttpHeaders.IF, String.format("<%s> (<%s>)", new DAVPathEncoder().encode(source), status.getLockId()))); } else { session.getClient().copy(new DAVPathEncoder().encode(source), target, status.isExists()); diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java index 7dffbaa132f..91daad6ab08 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVDeleteFeature.java @@ -81,7 +81,7 @@ protected HttpRequestBase toRequest(final Path file, final TransferStatus status final HttpDelete request = new HttpDelete(new DAVPathEncoder().encode(file)); if(status.getLockId() != null && session.getFeature(Lock.class) != null) { // Indicate that the client has knowledge of that state token - request.setHeader(HttpHeaders.IF, String.format("(<%s>)", status.getLockId())); + request.setHeader(HttpHeaders.IF, String.format("<%s> (<%s>)", new DAVPathEncoder().encode(file), status.getLockId())); } return request; } diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java index f81ab238618..d471fd8b427 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java @@ -53,7 +53,7 @@ public Path move(final Path file, final Path renamed, final TransferStatus statu // Indicate that the client has knowledge of that state token session.getClient().move(new DAVPathEncoder().encode(file), file.isDirectory() ? String.format("%s/", target) : target, status.isExists(), - Collections.singletonMap(HttpHeaders.IF, String.format("(<%s>)", status.getLockId()))); + Collections.singletonMap(HttpHeaders.IF, String.format("<%s> (<%s>)", new DAVPathEncoder().encode(file), status.getLockId()))); } else { session.getClient().move(new DAVPathEncoder().encode(file), file.isDirectory() ? String.format("%s/", target) : target, From 04c23e3dabbcc0048188d23256b1bc442f47328c Mon Sep 17 00:00:00 2001 From: David Kocher Date: Wed, 20 Nov 2024 11:28:53 +0100 Subject: [PATCH 3/5] A COPY method invocation MUST NOT duplicate any write locks active on the source. --- webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java index 6e645d0df4c..cc6abdac5a5 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVCopyFeature.java @@ -20,6 +20,7 @@ import ch.cyberduck.core.ConnectionCallback; import ch.cyberduck.core.DescriptiveUrl; import ch.cyberduck.core.Path; +import ch.cyberduck.core.PathAttributes; import ch.cyberduck.core.exception.BackgroundException; import ch.cyberduck.core.features.Copy; import ch.cyberduck.core.features.Lock; @@ -57,7 +58,7 @@ public Path copy(final Path source, final Path copy, final TransferStatus status session.getClient().copy(new DAVPathEncoder().encode(source), target, status.isExists()); } listener.sent(status.getLength()); - return copy.withAttributes(source.attributes()); + return copy.withAttributes(new PathAttributes(source.attributes()).withLockId(null)); } catch(SardineException e) { throw new DAVExceptionMappingService().map("Cannot copy {0}", e, source); From 7161b8343086728a949848f467ba5ad534fc4ee0 Mon Sep 17 00:00:00 2001 From: David Kocher Date: Wed, 20 Nov 2024 11:29:11 +0100 Subject: [PATCH 4/5] A successful MOVE request on a write locked resource MUST NOT move the write lock with the resource. --- webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java index d471fd8b427..8d2ed622f1e 100644 --- a/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java +++ b/webdav/src/main/java/ch/cyberduck/core/dav/DAVMoveFeature.java @@ -21,6 +21,7 @@ import ch.cyberduck.core.ConnectionCallback; import ch.cyberduck.core.DescriptiveUrl; import ch.cyberduck.core.Path; +import ch.cyberduck.core.PathAttributes; import ch.cyberduck.core.exception.BackgroundException; import ch.cyberduck.core.features.Delete; import ch.cyberduck.core.features.Lock; @@ -60,7 +61,7 @@ public Path move(final Path file, final Path renamed, final TransferStatus statu status.isExists()); } // Copy original file attributes - return renamed.withAttributes(file.attributes()); + return renamed.withAttributes(new PathAttributes(file.attributes()).withLockId(null)); } catch(SardineException e) { throw new DAVExceptionMappingService().map("Cannot rename {0}", e, file); From e82e4c2c5009bc43848da721c4a79aefb33f7a47 Mon Sep 17 00:00:00 2001 From: David Kocher Date: Wed, 20 Nov 2024 11:37:30 +0100 Subject: [PATCH 5/5] Add tests. --- .../core/dav/DAVCopyFeatureTest.java | 8 + .../core/dav/DAVMoveFeatureTest.java | 23 +-- .../MicrosoftIISDAVCopyFeatureTest.java | 129 +++++++++++++++ .../MicrosoftIISDAVMoveFeatureTest.java | 147 ++++++++++++++++++ 4 files changed, 290 insertions(+), 17 deletions(-) create mode 100644 webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVCopyFeatureTest.java create mode 100644 webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVMoveFeatureTest.java diff --git a/webdav/src/test/java/ch/cyberduck/core/dav/DAVCopyFeatureTest.java b/webdav/src/test/java/ch/cyberduck/core/dav/DAVCopyFeatureTest.java index 972feb48c01..58a0d86d251 100644 --- a/webdav/src/test/java/ch/cyberduck/core/dav/DAVCopyFeatureTest.java +++ b/webdav/src/test/java/ch/cyberduck/core/dav/DAVCopyFeatureTest.java @@ -22,6 +22,7 @@ import ch.cyberduck.core.DisabledLoginCallback; import ch.cyberduck.core.Path; import ch.cyberduck.core.exception.ConflictException; +import ch.cyberduck.core.exception.InteroperabilityException; import ch.cyberduck.core.features.Delete; import ch.cyberduck.core.features.Find; import ch.cyberduck.core.io.DisabledStreamListener; @@ -55,6 +56,13 @@ public void testCopyFile() throws Exception { new DAVDeleteFeature(session).delete(Collections.singletonList(copy), new DisabledLoginCallback(), new Delete.DisabledCallback()); } + @Test + public void testCopyWithLock() throws Exception { + final Path test = new DAVTouchFeature(session).touch(new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus()); + assertThrows(InteroperabilityException.class, () -> new DAVLockFeature(session).lock(test)); + new DAVDeleteFeature(session).delete(Collections.singletonMap(test, new TransferStatus()), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + @Test public void testCopyToExistingFile() throws Exception { final Path folder = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); diff --git a/webdav/src/test/java/ch/cyberduck/core/dav/DAVMoveFeatureTest.java b/webdav/src/test/java/ch/cyberduck/core/dav/DAVMoveFeatureTest.java index c9424dedeb1..1e703e41468 100644 --- a/webdav/src/test/java/ch/cyberduck/core/dav/DAVMoveFeatureTest.java +++ b/webdav/src/test/java/ch/cyberduck/core/dav/DAVMoveFeatureTest.java @@ -24,6 +24,7 @@ import ch.cyberduck.core.PathAttributes; import ch.cyberduck.core.exception.ConflictException; import ch.cyberduck.core.exception.InteroperabilityException; +import ch.cyberduck.core.exception.LockedException; import ch.cyberduck.core.exception.NotfoundException; import ch.cyberduck.core.features.Delete; import ch.cyberduck.core.http.HttpResponseOutputStream; @@ -58,26 +59,14 @@ public void testMove() throws Exception { assertTrue(new DAVFindFeature(session).find(target)); assertEquals(status.getResponse(), target.attributes()); assertEquals(attr, new DAVAttributesFinderFeature(session).find(target)); - new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); } @Test public void testMoveWithLock() throws Exception { final Path test = new DAVTouchFeature(session).touch(new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus()); - String lock = null; - try { - lock = new DAVLockFeature(session).lock(test); - } - catch(InteroperabilityException e) { - // Not supported - } - assertEquals(TransferStatus.UNKNOWN_LENGTH, test.attributes().getSize()); - final Path target = new DAVMoveFeature(session).move(test, - new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus().withLockId(lock), new Delete.DisabledCallback(), new DisabledConnectionCallback()); - assertFalse(new DAVFindFeature(session).find(test)); - assertTrue(new DAVFindFeature(session).find(target)); - assertEquals(test.attributes(), target.attributes()); - new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + assertThrows(InteroperabilityException.class, () -> new DAVLockFeature(session).lock(test)); + new DAVDeleteFeature(session).delete(Collections.singletonList(test), new DisabledLoginCallback(), new Delete.DisabledCallback()); } @Test @@ -105,7 +94,7 @@ public void testMoveDirectory() throws Exception { assertEquals(attr, new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file)))); assertEquals(attr.getModificationDate(), new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file))).getModificationDate()); assertNotEquals(attr.getETag(), new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file))).getETag()); - new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); } @Test @@ -118,7 +107,7 @@ public void testMoveOverride() throws Exception { new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(true), new Delete.DisabledCallback(), new DisabledConnectionCallback()); assertFalse(new DAVFindFeature(session).find(test)); assertTrue(new DAVFindFeature(session).find(target)); - new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); } @Test(expected = NotfoundException.class) diff --git a/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVCopyFeatureTest.java b/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVCopyFeatureTest.java new file mode 100644 index 00000000000..7118cee1387 --- /dev/null +++ b/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVCopyFeatureTest.java @@ -0,0 +1,129 @@ +package ch.cyberduck.core.dav.microsoft; + +/* + * Copyright (c) 2002-2013 David Kocher. All rights reserved. + * http://cyberduck.ch/ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 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 General Public License for more details. + * + * Bug fixes, suggestions and comments should be sent to feedback@cyberduck.ch + */ + +import ch.cyberduck.core.AlphanumericRandomStringService; +import ch.cyberduck.core.DisabledConnectionCallback; +import ch.cyberduck.core.DisabledLoginCallback; +import ch.cyberduck.core.Path; +import ch.cyberduck.core.dav.AbstractDAVTest; +import ch.cyberduck.core.dav.DAVAttributesFinderFeature; +import ch.cyberduck.core.dav.DAVCopyFeature; +import ch.cyberduck.core.dav.DAVDeleteFeature; +import ch.cyberduck.core.dav.DAVDirectoryFeature; +import ch.cyberduck.core.dav.DAVLockFeature; +import ch.cyberduck.core.dav.DAVTouchFeature; +import ch.cyberduck.core.exception.ConflictException; +import ch.cyberduck.core.features.Delete; +import ch.cyberduck.core.features.Find; +import ch.cyberduck.core.io.DisabledStreamListener; +import ch.cyberduck.core.shared.DefaultFindFeature; +import ch.cyberduck.core.shared.DefaultHomeFinderService; +import ch.cyberduck.core.transfer.TransferStatus; +import ch.cyberduck.test.IntegrationTest; + +import org.junit.Test; +import org.junit.experimental.categories.Category; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; + +import static org.junit.Assert.*; + +@Category(IntegrationTest.class) +public class MicrosoftIISDAVCopyFeatureTest extends AbstractMicrosoftIISDAVTest { + + @Test + public void testCopyFile() throws Exception { + final Path test = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(test, new TransferStatus()); + final Path copy = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVCopyFeature(session).copy(test, copy, new TransferStatus(), new DisabledConnectionCallback(), new DisabledStreamListener()); + assertEquals(new DAVAttributesFinderFeature(session).find(test), new DAVAttributesFinderFeature(session).find(copy)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(copy)); + new DAVDeleteFeature(session).delete(Collections.singletonList(test), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonList(copy), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testCopyWithLock() throws Exception { + final Path test = new DAVTouchFeature(session).touch(new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus()); + final String lock = new DAVLockFeature(session).lock(test); + assertEquals(TransferStatus.UNKNOWN_LENGTH, test.attributes().getSize()); + final Path target = new DAVCopyFeature(session).copy(test, + new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus().withLockId(lock), new DisabledConnectionCallback(), new DisabledStreamListener()); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + assertEquals(test.attributes(), target.attributes()); + new DAVDeleteFeature(session).delete(Collections.singletonMap(test, new TransferStatus().withLockId(lock)), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonMap(target, new TransferStatus()), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testCopyToExistingFile() throws Exception { + final Path folder = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + new DAVDirectoryFeature(session).mkdir(folder, new TransferStatus()); + final Path test = new Path(folder, new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(test, new TransferStatus()); + final Path copy = new Path(folder, new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(copy, new TransferStatus()); + assertThrows(ConflictException.class, () -> new DAVCopyFeature(session).copy(test, copy, new TransferStatus().exists(false), new DisabledConnectionCallback(), new DisabledStreamListener())); + new DAVCopyFeature(session).copy(test, copy, new TransferStatus().exists(true), new DisabledConnectionCallback(), new DisabledStreamListener()); + final Find find = new DefaultFindFeature(session); + assertTrue(find.find(test)); + assertTrue(find.find(copy)); + new DAVDeleteFeature(session).delete(Arrays.asList(test, copy), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testCopyWithLockToExistingFile() throws Exception { + final Path folder = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + new DAVDirectoryFeature(session).mkdir(folder, new TransferStatus()); + final Path test = new Path(folder, new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(test, new TransferStatus()); + final String lock = new DAVLockFeature(session).lock(test); + final Path copy = new Path(folder, new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(copy, new TransferStatus()); + assertThrows(ConflictException.class, () -> new DAVCopyFeature(session).copy(test, copy, new TransferStatus().exists(false), new DisabledConnectionCallback(), new DisabledStreamListener())); + new DAVCopyFeature(session).copy(test, copy, new TransferStatus().exists(true).withLockId(lock), new DisabledConnectionCallback(), new DisabledStreamListener()); + final Find find = new DefaultFindFeature(session); + assertTrue(find.find(test)); + assertTrue(find.find(copy)); + new DAVDeleteFeature(session).delete(Collections.singletonMap(test, new TransferStatus().withLockId(lock)), new DisabledLoginCallback(), new Delete.DisabledCallback()); + new DAVDeleteFeature(session).delete(Collections.singletonMap(copy, new TransferStatus()), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testCopyDirectory() throws Exception { + final Path directory = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + final String name = new AlphanumericRandomStringService().random(); + final Path file = new Path(directory, name, EnumSet.of(Path.Type.file)); + new DAVDirectoryFeature(session).mkdir(directory, new TransferStatus()); + new DAVTouchFeature(session).touch(file, new TransferStatus()); + final Path copy = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + new DAVDirectoryFeature(session).mkdir(copy, new TransferStatus()); + assertThrows(ConflictException.class, () -> new DAVCopyFeature(session).copy(directory, copy, new TransferStatus().exists(false), new DisabledConnectionCallback(), new DisabledStreamListener())); + new DAVCopyFeature(session).copy(directory, copy, new TransferStatus().exists(true), new DisabledConnectionCallback(), new DisabledStreamListener()); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(file)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(copy)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(new Path(copy, name, EnumSet.of(Path.Type.file)))); + new DAVDeleteFeature(session).delete(Arrays.asList(copy, directory), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } +} diff --git a/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVMoveFeatureTest.java b/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVMoveFeatureTest.java new file mode 100644 index 00000000000..c86bcd87178 --- /dev/null +++ b/webdav/src/test/java/ch/cyberduck/core/dav/microsoft/MicrosoftIISDAVMoveFeatureTest.java @@ -0,0 +1,147 @@ +package ch.cyberduck.core.dav.microsoft; + +/* + * Copyright (c) 2002-2013 David Kocher. All rights reserved. + * http://cyberduck.ch/ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 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 General Public License for more details. + * + * Bug fixes, suggestions and comments should be sent to feedback@cyberduck.ch + */ + +import ch.cyberduck.core.AlphanumericRandomStringService; +import ch.cyberduck.core.DisabledConnectionCallback; +import ch.cyberduck.core.DisabledLoginCallback; +import ch.cyberduck.core.Path; +import ch.cyberduck.core.PathAttributes; +import ch.cyberduck.core.dav.DAVAttributesFinderFeature; +import ch.cyberduck.core.dav.DAVDeleteFeature; +import ch.cyberduck.core.dav.DAVDirectoryFeature; +import ch.cyberduck.core.dav.DAVLockFeature; +import ch.cyberduck.core.dav.DAVMoveFeature; +import ch.cyberduck.core.dav.DAVTimestampFeature; +import ch.cyberduck.core.dav.DAVTouchFeature; +import ch.cyberduck.core.dav.DAVWriteFeature; +import ch.cyberduck.core.exception.ConflictException; +import ch.cyberduck.core.exception.LockedException; +import ch.cyberduck.core.exception.NotfoundException; +import ch.cyberduck.core.features.Delete; +import ch.cyberduck.core.http.HttpResponseOutputStream; +import ch.cyberduck.core.io.StreamCopier; +import ch.cyberduck.core.shared.DefaultHomeFinderService; +import ch.cyberduck.core.transfer.TransferStatus; +import ch.cyberduck.test.IntegrationTest; + +import org.apache.commons.lang3.RandomUtils; +import org.junit.Test; +import org.junit.experimental.categories.Category; + +import java.io.ByteArrayInputStream; +import java.util.Collections; +import java.util.EnumSet; + +import static org.junit.Assert.*; + +@Category(IntegrationTest.class) +public class MicrosoftIISDAVMoveFeatureTest extends AbstractMicrosoftIISDAVTest { + + @Test + public void testMove() throws Exception { + final Path test = new DAVTouchFeature(session).touch(new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus()); + assertEquals(TransferStatus.UNKNOWN_LENGTH, test.attributes().getSize()); + final TransferStatus status = new TransferStatus(); + new DAVTimestampFeature(session).setTimestamp(test, status.withModified(5000L)); + final PathAttributes attr = new DAVAttributesFinderFeature(session).find(test); + final Path target = new DAVMoveFeature(session).move(test.withAttributes(status.getResponse()), + new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus(), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + assertEquals(status.getResponse(), target.attributes()); + assertEquals(attr, new DAVAttributesFinderFeature(session).find(target)); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testMoveWithLock() throws Exception { + final Path test = new DAVTouchFeature(session).touch(new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus()); + final String lock = new DAVLockFeature(session).lock(test); + assertEquals(TransferStatus.UNKNOWN_LENGTH, test.attributes().getSize()); + final Path target = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + assertThrows(LockedException.class, () -> new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(false), new Delete.DisabledCallback(), new DisabledConnectionCallback())); + new DAVMoveFeature(session).move(test, target, new TransferStatus().withLockId(lock), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + assertEquals(test.attributes(), target.attributes()); + new DAVDeleteFeature(session).delete(Collections.singletonMap(target, new TransferStatus()), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testMoveDirectory() throws Exception { + final Path folder = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + new DAVDirectoryFeature(session).mkdir(folder, new TransferStatus()); + final Path test = new Path(folder, new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + { + final byte[] content = RandomUtils.nextBytes(3547); + final TransferStatus status = new TransferStatus(); + status.setOffset(0L); + status.setLength(1024L); + final HttpResponseOutputStream out = new DAVWriteFeature(session).write(test, status, new DisabledConnectionCallback()); + // Write first 1024 + new StreamCopier(status, status).withOffset(status.getOffset()).withLimit(status.getLength()).transfer(new ByteArrayInputStream(content), out); + out.close(); + } + final PathAttributes attr = new DAVAttributesFinderFeature(session).find(test); + final Path target = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.directory)); + new DAVMoveFeature(session).move(folder, target, new TransferStatus(), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(folder)); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file)))); + assertEquals(attr, new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file)))); + assertEquals(attr.getModificationDate(), new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file))).getModificationDate()); + assertEquals(attr.getETag(), new DAVAttributesFinderFeature(session).find(new Path(target, test.getName(), EnumSet.of(Path.Type.file))).getETag()); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testMoveOverride() throws Exception { + final Path test = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(test, new TransferStatus()); + final Path target = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(target, new TransferStatus()); + assertThrows(ConflictException.class, () -> new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(false), new Delete.DisabledCallback(), new DisabledConnectionCallback())); + new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(true), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test + public void testMoveOverrideWithLock() throws Exception { + final Path test = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(test, new TransferStatus()); + final Path target = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVTouchFeature(session).touch(target, new TransferStatus()); + assertThrows(ConflictException.class, () -> new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(false), new Delete.DisabledCallback(), new DisabledConnectionCallback())); + final String lock = new DAVLockFeature(session).lock(test); + assertThrows(LockedException.class, () -> new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(false), new Delete.DisabledCallback(), new DisabledConnectionCallback())); + new DAVMoveFeature(session).move(test, target, new TransferStatus().exists(true).withLockId(lock), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + assertFalse(new MicrosoftIISDAVFindFeature(session).find(test)); + assertTrue(new MicrosoftIISDAVFindFeature(session).find(target)); + new DAVDeleteFeature(session).delete(Collections.singletonList(target), new DisabledLoginCallback(), new Delete.DisabledCallback()); + } + + @Test(expected = NotfoundException.class) + public void testMoveNotFound() throws Exception { + final Path test = new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)); + new DAVMoveFeature(session).move(test, new Path(new DefaultHomeFinderService(session).find(), new AlphanumericRandomStringService().random(), EnumSet.of(Path.Type.file)), new TransferStatus(), new Delete.DisabledCallback(), new DisabledConnectionCallback()); + } +}