diff --git a/src/main/java/org/cryptomator/cryptofs/migration/v7/FilePathMigration.java b/src/main/java/org/cryptomator/cryptofs/migration/v7/FilePathMigration.java
index 4a08097d..7f33177a 100644
--- a/src/main/java/org/cryptomator/cryptofs/migration/v7/FilePathMigration.java
+++ b/src/main/java/org/cryptomator/cryptofs/migration/v7/FilePathMigration.java
@@ -44,7 +44,7 @@ class FilePathMigration {
private final String oldCanonicalName;
/**
- * @param oldPath The actual file path before migration
+ * @param oldPath The actual file path before migration
* @param oldCanonicalName The inflated old filename without any conflicting pre- or suffixes but including the file type prefix
*/
FilePathMigration(Path oldPath, String oldCanonicalName) {
@@ -57,7 +57,7 @@ class FilePathMigration {
* Starts a migration of the given file.
*
* @param vaultRoot Path to the vault's base directory (parent of d/
and m/
).
- * @param oldPath Path of an existing file inside the d/
directory of a vault. May be a normal file, directory file or symlink as well as conflicting copies.
+ * @param oldPath Path of an existing file inside the d/
directory of a vault. May be a normal file, directory file or symlink as well as conflicting copies.
* @return A new instance of FileNameMigration
* @throws IOException Non-recoverable I/O error, such as {@link UninflatableFileException}s
*/
@@ -89,7 +89,7 @@ public static Optional parse(Path vaultRoot, Path oldPath) th
/**
* Resolves the canonical name of a deflated file represented by the given longFileName
.
*
- * @param vaultRoot Path to the vault's base directory (parent of d/
and m/
).
+ * @param vaultRoot Path to the vault's base directory (parent of d/
and m/
).
* @param longFileName Canonical name of the {@value #OLD_SHORTENED_FILENAME_SUFFIX} file.
* @return The inflated filename
* @throws UninflatableFileException If the file could not be inflated due to missing or malformed metadata.
@@ -151,7 +151,7 @@ public Path migrate() throws IOException {
* @return The path after successful migration of {@link #oldPath} if migration is successful for the given attemptSuffix
*/
// visible for testing
- Path getTargetPath(String attemptSuffix) {
+ Path getTargetPath(String attemptSuffix) throws InvalidOldFilenameException {
final String canonicalInflatedName = getNewInflatedName();
final String canonicalDeflatedName = getNewDeflatedName();
final boolean isShortened = !canonicalInflatedName.equals(canonicalDeflatedName);
@@ -177,7 +177,7 @@ Path getTargetPath(String attemptSuffix) {
}
}
}
-
+
public Path getOldPath() {
return oldPath;
}
@@ -202,19 +202,34 @@ String getOldCanonicalNameWithoutTypePrefix() {
}
/**
- * @return BASE64-encode(BASE32-decode({@link #getOldCanonicalNameWithoutTypePrefix oldCanonicalNameWithoutPrefix})) + {@value #NEW_REGULAR_SUFFIX}
+ * @return BASE64-encode({@link #getDecodedCiphertext oldDecodedCiphertext}) + {@value #NEW_REGULAR_SUFFIX}
+ * @throws InvalidOldFilenameException if failing to base32-decode the old filename
*/
// visible for testing
- String getNewInflatedName() {
- byte[] decoded = BASE32.decode(getOldCanonicalNameWithoutTypePrefix());
+ String getNewInflatedName() throws InvalidOldFilenameException {
+ byte[] decoded = getDecodedCiphertext();
return BASE64.encode(decoded) + NEW_REGULAR_SUFFIX;
}
+ /**
+ * @return BASE32-decode({@link #getOldCanonicalNameWithoutTypePrefix oldCanonicalNameWithoutPrefix})
+ * @throws InvalidOldFilenameException if failing to base32-decode the old filename
+ */
+ // visible for testing
+ byte[] getDecodedCiphertext() throws InvalidOldFilenameException {
+ String encodedCiphertext = getOldCanonicalNameWithoutTypePrefix();
+ try {
+ return BASE32.decode(encodedCiphertext);
+ } catch (IllegalArgumentException e) {
+ throw new InvalidOldFilenameException("Can't base32-decode '" + encodedCiphertext + "' in file " + oldPath.toString(), e);
+ }
+ }
+
/**
* @return {@link #getNewInflatedName() newInflatedName} if it is shorter than {@link #SHORTENING_THRESHOLD}, else BASE64(SHA1(newInflatedName)) + ".c9s"
*/
// visible for testing
- String getNewDeflatedName() {
+ String getNewDeflatedName() throws InvalidOldFilenameException {
String inflatedName = getNewInflatedName();
if (inflatedName.length() > SHORTENING_THRESHOLD) {
byte[] longFileNameBytes = inflatedName.getBytes(UTF_8);
diff --git a/src/main/java/org/cryptomator/cryptofs/migration/v7/InvalidOldFilenameException.java b/src/main/java/org/cryptomator/cryptofs/migration/v7/InvalidOldFilenameException.java
new file mode 100644
index 00000000..fe45fde9
--- /dev/null
+++ b/src/main/java/org/cryptomator/cryptofs/migration/v7/InvalidOldFilenameException.java
@@ -0,0 +1,10 @@
+package org.cryptomator.cryptofs.migration.v7;
+
+import java.io.IOException;
+
+public class InvalidOldFilenameException extends IOException {
+
+ public InvalidOldFilenameException(String message, Throwable cause) {
+ super(message, cause);
+ }
+}
\ No newline at end of file
diff --git a/src/test/java/org/cryptomator/cryptofs/migration/v7/FilePathMigrationTest.java b/src/test/java/org/cryptomator/cryptofs/migration/v7/FilePathMigrationTest.java
index e2962915..120592bb 100644
--- a/src/test/java/org/cryptomator/cryptofs/migration/v7/FilePathMigrationTest.java
+++ b/src/test/java/org/cryptomator/cryptofs/migration/v7/FilePathMigrationTest.java
@@ -15,6 +15,9 @@
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestInstance;
import org.junit.jupiter.params.ParameterizedTest;
+import org.junit.jupiter.params.converter.ArgumentConversionException;
+import org.junit.jupiter.params.converter.ConvertWith;
+import org.junit.jupiter.params.converter.SimpleArgumentConverter;
import org.junit.jupiter.params.provider.CsvSource;
import org.junit.jupiter.params.provider.ValueSource;
import org.mockito.Mockito;
@@ -25,6 +28,7 @@
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.nio.file.Paths;
+import java.util.Base64;
import java.util.Optional;
import static java.nio.charset.StandardCharsets.UTF_8;
@@ -44,6 +48,31 @@ public void testGetOldCanonicalNameWithoutTypePrefix(String oldCanonicalName, St
Assertions.assertEquals(expectedResult, migration.getOldCanonicalNameWithoutTypePrefix());
}
+
+ @ParameterizedTest(name = "getDecodedCiphertext() expected to be {1} for {0}")
+ @CsvSource({
+ "ORSXG5A=,dGVzdA==",
+ "0ORSXG5A=,dGVzdA==",
+ "1SORSXG5A=,dGVzdA==",
+ })
+ public void testGetDecodedCiphertext(String oldCanonicalName, @ConvertWith(ByteArrayArgumentConverter.class) byte[] expected) throws InvalidOldFilenameException {
+ FilePathMigration migration = new FilePathMigration(oldPath, oldCanonicalName);
+ Assertions.assertArrayEquals(expected, migration.getDecodedCiphertext());
+ }
+
+ @ParameterizedTest(name = "getDecodedCiphertext() throws InvalidOldFilenameException for {0}")
+ @ValueSource(strings = {
+ "ORSXG5=A",
+ "ORSX=G5A"
+ })
+ public void testMalformedGetDecodedCiphertext(String oldCanonicalName) {
+ FilePathMigration migration = new FilePathMigration(oldPath, oldCanonicalName);
+
+ InvalidOldFilenameException e = Assertions.assertThrows(InvalidOldFilenameException.class, () -> {
+ migration.getDecodedCiphertext();
+ });
+ Assertions.assertTrue(e.getMessage().contains(oldPath.toString()));
+ }
@ParameterizedTest(name = "getNewInflatedName() expected to be {1} for {0}")
@CsvSource({
@@ -51,7 +80,7 @@ public void testGetOldCanonicalNameWithoutTypePrefix(String oldCanonicalName, St
"0ORSXG5A=,dGVzdA==.c9r",
"1SORSXG5A=,dGVzdA==.c9r",
})
- public void testGetNewInflatedName(String oldCanonicalName, String expectedResult) {
+ public void testGetNewInflatedName(String oldCanonicalName, String expectedResult) throws InvalidOldFilenameException {
FilePathMigration migration = new FilePathMigration(oldPath, oldCanonicalName);
Assertions.assertEquals(expectedResult, migration.getNewInflatedName());
@@ -63,7 +92,7 @@ public void testGetNewInflatedName(String oldCanonicalName, String expectedResul
"ORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSQ====,dGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRlc3QgdGVzdCB0ZXN0IHRl.c9r",
"ORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG===,30xtS3YjsiMJRwu1oAVc_0S2aAU=.c9s",
})
- public void testGetNewDeflatedName(String oldCanonicalName, String expectedResult) {
+ public void testGetNewDeflatedName(String oldCanonicalName, String expectedResult) throws InvalidOldFilenameException {
FilePathMigration migration = new FilePathMigration(oldPath, oldCanonicalName);
Assertions.assertEquals(expectedResult, migration.getNewDeflatedName());
@@ -104,7 +133,7 @@ public void testIsSymlink(String oldCanonicalName, boolean expectedResult) {
"ORSXG5A=,'_1',dGVzdA==_1.c9r",
"ORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG5BAORSXG===,'_123',30xtS3YjsiMJRwu1oAVc_0S2aAU=_123.c9s/contents.c9r",
})
- public void testGetTargetPath(String oldCanonicalName, String attemptSuffix, String expected) {
+ public void testGetTargetPath(String oldCanonicalName, String attemptSuffix, String expected) throws InvalidOldFilenameException {
Path old = Paths.get("/tmp/foo");
FilePathMigration migration = new FilePathMigration(old, oldCanonicalName);
@@ -325,4 +354,12 @@ public void testMigrateShortened(String oldPathStr, String metadataFilePath, Str
}
+ public static class ByteArrayArgumentConverter extends SimpleArgumentConverter {
+
+ @Override
+ protected Object convert(Object source, Class> targetType) throws ArgumentConversionException {
+ Assertions.assertEquals(byte[].class, targetType, "Can only convert to byte[]");
+ return Base64.getUrlDecoder().decode(String.valueOf(source));
+ }
+ }
}