NoDirFileDialog not shown using Local storage if dir file missing

Fixes #437
This commit is contained in:
Julian Raufelder 2022-04-20 17:29:51 +02:00
parent c5d7a22f2f
commit a39e5698a9
No known key found for this signature in database
GPG Key ID: 17EE71F6634E381D
5 changed files with 125 additions and 98 deletions

View File

@ -85,10 +85,13 @@ abstract class CryptoImplDecorator(
abstract fun write(cryptoFile: CryptoFile, data: DataSource, progressAware: ProgressAware<UploadState>, replace: Boolean, length: Long): CryptoFile abstract fun write(cryptoFile: CryptoFile, data: DataSource, progressAware: ProgressAware<UploadState>, replace: Boolean, length: Long): CryptoFile
@Throws(BackendException::class, EmptyDirFileException::class) @Throws(BackendException::class, EmptyDirFileException::class)
abstract fun loadDirId(folder: CryptoFolder): String abstract fun loadDirId(folder: CryptoFolder): String?
@Throws(BackendException::class) @Throws(BackendException::class)
abstract fun createDirIdInfo(folder: CryptoFolder): DirIdInfo abstract fun getOrCreateDirIdInfo(folder: CryptoFolder): DirIdInfo
@Throws(BackendException::class)
abstract fun getDirIdInfo(folder: CryptoFolder): DirIdInfo?
private fun dirHash(directoryId: String): String { private fun dirHash(directoryId: String): String {
return cryptor().fileNameCryptor().hashDirectoryId(directoryId) return cryptor().fileNameCryptor().hashDirectoryId(directoryId)
@ -162,7 +165,7 @@ abstract class CryptoImplDecorator(
@Throws(BackendException::class) @Throws(BackendException::class)
private fun file(cryptoParent: CryptoFolder, cleartextName: String, ciphertextName: String, cleartextSize: Long?): CryptoFile { private fun file(cryptoParent: CryptoFolder, cleartextName: String, ciphertextName: String, cleartextSize: Long?): CryptoFile {
val ciphertextSize = cleartextSize?.let { cryptor().fileContentCryptor().ciphertextSize(it) + cryptor().fileHeaderCryptor().headerSize() } val ciphertextSize = cleartextSize?.let { cryptor().fileContentCryptor().ciphertextSize(it) + cryptor().fileHeaderCryptor().headerSize() }
val cloudFile = cloudContentRepository.file(dirIdInfo(cryptoParent).cloudFolder, ciphertextName, ciphertextSize) val cloudFile = cloudContentRepository.file(getOrCreateCachingAwareDirIdInfo(cryptoParent).cloudFolder, ciphertextName, ciphertextSize)
return file(cryptoParent, cleartextName, cloudFile, cleartextSize) return file(cryptoParent, cleartextName, cloudFile, cleartextSize)
} }
@ -212,7 +215,9 @@ abstract class CryptoImplDecorator(
@Throws(BackendException::class) @Throws(BackendException::class)
private fun exists(folder: CryptoFolder): Boolean { private fun exists(folder: CryptoFolder): Boolean {
requireNotNull(folder.dirFile) requireNotNull(folder.dirFile)
return cloudContentRepository.exists(folder.dirFile) && cloudContentRepository.exists(dirIdInfo(folder).cloudFolder) return cloudContentRepository.exists(folder.dirFile) && getCachingAwareDirIdInfo(folder)?.let {
cloudContentRepository.exists(it.cloudFolder)
} ?: false
} }
@Throws(BackendException::class) @Throws(BackendException::class)
@ -352,8 +357,13 @@ abstract class CryptoImplDecorator(
} }
@Throws(BackendException::class) @Throws(BackendException::class)
fun dirIdInfo(folder: CryptoFolder): DirIdInfo { fun getOrCreateCachingAwareDirIdInfo(folder: CryptoFolder): DirIdInfo {
return dirIdCache[folder] ?: return createDirIdInfo(folder) return dirIdCache[folder] ?: return getOrCreateDirIdInfo(folder)
}
@Throws(BackendException::class)
fun getCachingAwareDirIdInfo(folder: CryptoFolder): DirIdInfo? {
return dirIdCache[folder] ?: return getDirIdInfo(folder)
} }
@Throws(BackendException::class) @Throws(BackendException::class)

View File

@ -64,7 +64,7 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
@Throws(BackendException::class) @Throws(BackendException::class)
override fun folder(cryptoParent: CryptoFolder, cleartextName: String): CryptoFolder { override fun folder(cryptoParent: CryptoFolder, cleartextName: String): CryptoFolder {
val dirFileName = encryptFolderName(cryptoParent, cleartextName) val dirFileName = encryptFolderName(cryptoParent, cleartextName)
val dirFolder = cloudContentRepository.folder(dirIdInfo(cryptoParent).cloudFolder, dirFileName) val dirFolder = cloudContentRepository.folder(getOrCreateCachingAwareDirIdInfo(cryptoParent).cloudFolder, dirFileName)
val dirFile = cloudContentRepository.file(dirFolder, CLOUD_FOLDER_DIR_FILE_PRE + CLOUD_NODE_EXT) val dirFile = cloudContentRepository.file(dirFolder, CLOUD_FOLDER_DIR_FILE_PRE + CLOUD_NODE_EXT)
return folder(cryptoParent, cleartextName, dirFile) return folder(cryptoParent, cleartextName, dirFile)
} }
@ -73,7 +73,7 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
override fun encryptName(cryptoParent: CryptoFolder, name: String): String { override fun encryptName(cryptoParent: CryptoFolder, name: String): String {
var ciphertextName: String = cryptor() // var ciphertextName: String = cryptor() //
.fileNameCryptor() // .fileNameCryptor() //
.encryptFilename(BaseEncoding.base64Url(), name, dirIdInfo(cryptoParent).id.toByteArray(StandardCharsets.UTF_8)) + CLOUD_NODE_EXT .encryptFilename(BaseEncoding.base64Url(), name, getOrCreateCachingAwareDirIdInfo(cryptoParent).id.toByteArray(StandardCharsets.UTF_8)) + CLOUD_NODE_EXT
if (ciphertextName.length > shorteningThreshold) { if (ciphertextName.length > shorteningThreshold) {
ciphertextName = deflate(cryptoParent, ciphertextName) ciphertextName = deflate(cryptoParent, ciphertextName)
} }
@ -85,7 +85,7 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
val longFilenameBytes = longFileName.toByteArray(StandardCharsets.UTF_8) val longFilenameBytes = longFileName.toByteArray(StandardCharsets.UTF_8)
val hash = MessageDigestSupplier.SHA1.get().digest(longFilenameBytes) val hash = MessageDigestSupplier.SHA1.get().digest(longFilenameBytes)
val shortFileName = BaseEncoding.base64Url().encode(hash) + LONG_NODE_FILE_EXT val shortFileName = BaseEncoding.base64Url().encode(hash) + LONG_NODE_FILE_EXT
var dirFolder = cloudContentRepository.folder(dirIdInfo(cryptoParent).cloudFolder, shortFileName) var dirFolder = cloudContentRepository.folder(getOrCreateCachingAwareDirIdInfo(cryptoParent).cloudFolder, shortFileName)
// if folder already exists in case of renaming // if folder already exists in case of renaming
if (!cloudContentRepository.exists(dirFolder)) { if (!cloudContentRepository.exists(dirFolder)) {
@ -98,6 +98,14 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
return shortFileName return shortFileName
} }
@Throws(BackendException::class)
private fun inflate(cloudNode: CloudNode): String {
val metadataFile = metadataFile(cloudNode)
val out = ByteArrayOutputStream()
cloudContentRepository.read(metadataFile, null, out, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD)
return String(out.toByteArray(), StandardCharsets.UTF_8)
}
@Throws(BackendException::class) @Throws(BackendException::class)
private fun metadataFile(cloudNode: CloudNode): CloudFile { private fun metadataFile(cloudNode: CloudNode): CloudFile {
val cloudFolder = when (cloudNode) { val cloudFolder = when (cloudNode) {
@ -114,14 +122,6 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
return cloudContentRepository.file(cloudFolder, LONG_NODE_FILE_CONTENT_NAME + LONG_NODE_FILE_EXT) return cloudContentRepository.file(cloudFolder, LONG_NODE_FILE_CONTENT_NAME + LONG_NODE_FILE_EXT)
} }
@Throws(BackendException::class)
private fun inflate(cloudNode: CloudNode): String {
val metadataFile = metadataFile(cloudNode)
val out = ByteArrayOutputStream()
cloudContentRepository.read(metadataFile, null, out, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD)
return String(out.toByteArray(), StandardCharsets.UTF_8)
}
override fun decryptName(dirId: String, encryptedName: String): String? { override fun decryptName(dirId: String, encryptedName: String): String? {
return extractEncryptedName(encryptedName)?.let { return extractEncryptedName(encryptedName)?.let {
return cryptor().fileNameCryptor().decryptFilename(BaseEncoding.base64Url(), it, dirId.toByteArray(StandardCharsets.UTF_8)) return cryptor().fileNameCryptor().decryptFilename(BaseEncoding.base64Url(), it, dirId.toByteArray(StandardCharsets.UTF_8))
@ -132,34 +132,41 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
override fun list(cryptoFolder: CryptoFolder): List<CryptoNode> { override fun list(cryptoFolder: CryptoFolder): List<CryptoNode> {
dirIdCache.evictSubFoldersOf(cryptoFolder) dirIdCache.evictSubFoldersOf(cryptoFolder)
val dirIdInfo = dirIdInfo(cryptoFolder) val dirIdInfo = getCachingAwareDirIdInfo(cryptoFolder)
val dirId = dirIdInfo(cryptoFolder).id ?: when (cryptoFolder.dirFile) {
null -> {
Timber.tag("CryptoFs").e(String.format("Dir-file of folder is null %s", cryptoFolder.path))
throw FatalBackendException(String.format("Dir-file of folder is null %s", cryptoFolder.path))
}
else -> {
Timber.tag("CryptoFs").e("No dir file exists in %s", cryptoFolder.dirFile.path)
throw NoDirFileException(cryptoFolder.name, cryptoFolder.dirFile.path)
}
}
val dirId = dirIdInfo.id
val lvl2Dir = dirIdInfo.cloudFolder val lvl2Dir = dirIdInfo.cloudFolder
val ciphertextNodes: List<CloudNode> = try { return try {
cloudContentRepository.list(lvl2Dir) cloudContentRepository.list(lvl2Dir)
} catch (e: NoSuchCloudFileException) { } catch (e: NoSuchCloudFileException) {
if (cryptoFolder is RootCryptoFolder) { when {
Timber.tag("CryptoFs").e("No lvl2Dir exists for root folder in %s", lvl2Dir.path) cryptoFolder is RootCryptoFolder -> {
throw FatalBackendException(String.format("No lvl2Dir exists for root folder in %s", lvl2Dir.path), e) Timber.tag("CryptoFs").e("No lvl2Dir exists for root folder in %s", lvl2Dir.path)
} else if (cryptoFolder.dirFile == null) { throw FatalBackendException(String.format("No lvl2Dir exists for root folder in %s", lvl2Dir.path), e)
Timber.tag("CryptoFs").e(String.format("Dir-file of folder is null %s", lvl2Dir.path)) }
throw FatalBackendException(String.format("Dir-file of folder is null %s", lvl2Dir.path)) cryptoFolder.dirFile == null -> {
} else if (cloudContentRepository.exists(cloudContentRepository.file(cryptoFolder.dirFile.parent, CLOUD_NODE_SYMLINK_PRE + CLOUD_NODE_EXT))) { Timber.tag("CryptoFs").e(String.format("Dir-file of folder is null %s", lvl2Dir.path))
throw SymLinkException() throw FatalBackendException(String.format("Dir-file of folder is null %s", lvl2Dir.path))
} else if (!cloudContentRepository.exists(cryptoFolder.dirFile)) { }
Timber.tag("CryptoFs").e("No dir file exists in %s", cryptoFolder.dirFile.path) cloudContentRepository.exists(cloudContentRepository.file(cryptoFolder.dirFile.parent, CLOUD_NODE_SYMLINK_PRE + CLOUD_NODE_EXT)) -> {
throw NoDirFileException(cryptoFolder.name, cryptoFolder.dirFile.path) throw SymLinkException()
}
else -> return emptyList()
} }
return emptyList() }.map { node ->
} ciphertextToCleartextNode(cryptoFolder, dirId, node)
}.toList().filterNotNull()
return ciphertextNodes
.map { node ->
ciphertextToCleartextNode(cryptoFolder, dirId, node)
}
.toList()
.filterNotNull()
} }
@Throws(BackendException::class) @Throws(BackendException::class)
@ -247,11 +254,17 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
} }
@Throws(BackendException::class) @Throws(BackendException::class)
override fun createDirIdInfo(folder: CryptoFolder): DirIdInfo { override fun getOrCreateDirIdInfo(folder: CryptoFolder): DirIdInfo {
val dirId = loadDirId(folder) val dirId = loadDirId(folder) ?: newDirId()
return dirIdCache.put(folder, createDirIdInfoFor(dirId)) return dirIdCache.put(folder, createDirIdInfoFor(dirId))
} }
override fun getDirIdInfo(folder: CryptoFolder): DirIdInfo? {
return loadDirId(folder)?.let {
dirIdCache.put(folder, createDirIdInfoFor(it))
}
}
@Throws(BackendException::class) @Throws(BackendException::class)
override fun encryptFolderName(cryptoFolder: CryptoFolder, name: String): String { override fun encryptFolderName(cryptoFolder: CryptoFolder, name: String): String {
return encryptName(cryptoFolder, name) return encryptName(cryptoFolder, name)
@ -263,17 +276,13 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
} }
@Throws(BackendException::class, EmptyDirFileException::class) @Throws(BackendException::class, EmptyDirFileException::class)
override fun loadDirId(folder: CryptoFolder): String { override fun loadDirId(folder: CryptoFolder): String? {
var dirFile: CloudFile? = null
if (folder.dirFile != null) {
dirFile = folder.dirFile
}
return if (RootCryptoFolder.isRoot(folder)) { return if (RootCryptoFolder.isRoot(folder)) {
CryptoConstants.ROOT_DIR_ID CryptoConstants.ROOT_DIR_ID
} else if (dirFile != null && cloudContentRepository.exists(dirFile)) { } else if (folder.dirFile != null && cloudContentRepository.exists(folder.dirFile)) {
String(loadContentsOfDirFile(dirFile), StandardCharsets.UTF_8) String(loadContentsOfDirFile(folder.dirFile), StandardCharsets.UTF_8)
} else { } else {
newDirId() null
} }
} }
@ -302,7 +311,7 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
assertCryptoFolderAlreadyExists(folder) assertCryptoFolderAlreadyExists(folder)
shortName = true shortName = true
} }
val dirIdInfo = dirIdInfo(folder) val dirIdInfo = getOrCreateCachingAwareDirIdInfo(folder)
val createdCloudFolder = cloudContentRepository.create(dirIdInfo.cloudFolder) val createdCloudFolder = cloudContentRepository.create(dirIdInfo.cloudFolder)
var dirFolder = folder.dirFile.parent var dirFolder = folder.dirFile.parent
var dirFile = folder.dirFile var dirFile = folder.dirFile
@ -407,17 +416,13 @@ open class CryptoImplVaultFormat7 : CryptoImplDecorator {
requireNotNull(node.dirFile) requireNotNull(node.dirFile)
val cryptoSubfolders = deepCollectSubfolders(node) val cryptoSubfolders = deepCollectSubfolders(node)
for (cryptoSubfolder in cryptoSubfolders) { for (cryptoSubfolder in cryptoSubfolders) {
try { getCachingAwareDirIdInfo(cryptoSubfolder)?.let {
cloudContentRepository.delete(dirIdInfo(cryptoSubfolder).cloudFolder) cloudContentRepository.delete(it.cloudFolder)
} catch (e: NoSuchCloudFileException) { } ?: Timber.tag("CryptoFs").w("Dir file doesn't exists of a sub folder while deleting the parent, continue anyway")
// Ignoring because nothing can be done if the dir-file doesn't exists in the cloud
}
}
try {
cloudContentRepository.delete(dirIdInfo(node).cloudFolder)
} catch (e: NoSuchCloudFileException) {
// Ignoring because nothing can be done if the dir-file doesn't exists in the cloud
} }
getCachingAwareDirIdInfo(node)?.let {
cloudContentRepository.delete(it.cloudFolder)
} ?: Timber.tag("CryptoFs").w("Dir file doesn't exists while deleting the folder, continue anyway")
cloudContentRepository.delete(node.dirFile.parent) cloudContentRepository.delete(node.dirFile.parent)
evictFromCache(node) evictFromCache(node)
} else if (node is CryptoFile) { } else if (node is CryptoFile) {

View File

@ -16,6 +16,7 @@ import org.cryptomator.domain.CloudNode
import org.cryptomator.domain.exception.AlreadyExistException import org.cryptomator.domain.exception.AlreadyExistException
import org.cryptomator.domain.exception.BackendException import org.cryptomator.domain.exception.BackendException
import org.cryptomator.domain.exception.EmptyDirFileException import org.cryptomator.domain.exception.EmptyDirFileException
import org.cryptomator.domain.exception.NoDirFileException
import org.cryptomator.domain.exception.NoSuchCloudFileException import org.cryptomator.domain.exception.NoSuchCloudFileException
import org.cryptomator.domain.exception.ParentFolderIsNullException import org.cryptomator.domain.exception.ParentFolderIsNullException
import org.cryptomator.domain.repository.CloudContentRepository import org.cryptomator.domain.repository.CloudContentRepository
@ -27,7 +28,6 @@ import java.io.ByteArrayOutputStream
import java.nio.charset.StandardCharsets import java.nio.charset.StandardCharsets
import java.util.function.Supplier import java.util.function.Supplier
import java.util.regex.Pattern import java.util.regex.Pattern
import kotlin.streams.toList
import timber.log.Timber import timber.log.Timber
internal class CryptoImplVaultFormatPre7( internal class CryptoImplVaultFormatPre7(
@ -44,7 +44,7 @@ internal class CryptoImplVaultFormatPre7(
@Throws(BackendException::class) @Throws(BackendException::class)
override fun folder(cryptoParent: CryptoFolder, cleartextName: String): CryptoFolder { override fun folder(cryptoParent: CryptoFolder, cleartextName: String): CryptoFolder {
val dirFileName = encryptFolderName(cryptoParent, cleartextName) val dirFileName = encryptFolderName(cryptoParent, cleartextName)
val dirFile = cloudContentRepository.file(dirIdInfo(cryptoParent).cloudFolder, dirFileName) val dirFile = cloudContentRepository.file(getOrCreateCachingAwareDirIdInfo(cryptoParent).cloudFolder, dirFileName)
return folder(cryptoParent, cleartextName, dirFile) return folder(cryptoParent, cleartextName, dirFile)
} }
@ -52,7 +52,7 @@ internal class CryptoImplVaultFormatPre7(
override fun create(folder: CryptoFolder): CryptoFolder { override fun create(folder: CryptoFolder): CryptoFolder {
requireNotNull(folder.dirFile) requireNotNull(folder.dirFile)
assertCryptoFolderAlreadyExists(folder) assertCryptoFolderAlreadyExists(folder)
val dirIdInfo = dirIdInfo(folder) val dirIdInfo = getOrCreateCachingAwareDirIdInfo(folder)
val createdCloudFolder = cloudContentRepository.create(dirIdInfo.cloudFolder) val createdCloudFolder = cloudContentRepository.create(dirIdInfo.cloudFolder)
val dirId = dirIdInfo.id.toByteArray(StandardCharsets.UTF_8) val dirId = dirIdInfo.id.toByteArray(StandardCharsets.UTF_8)
val createdDirFile = cloudContentRepository.write(folder.dirFile, from(dirId), ProgressAware.NO_OP_PROGRESS_AWARE_UPLOAD, false, dirId.size.toLong()) val createdDirFile = cloudContentRepository.write(folder.dirFile, from(dirId), ProgressAware.NO_OP_PROGRESS_AWARE_UPLOAD, false, dirId.size.toLong())
@ -68,7 +68,7 @@ internal class CryptoImplVaultFormatPre7(
@Throws(BackendException::class) @Throws(BackendException::class)
private fun encryptName(cryptoParent: CryptoFolder, name: String, prefix: String): String { private fun encryptName(cryptoParent: CryptoFolder, name: String, prefix: String): String {
var ciphertextName = prefix + cryptor().fileNameCryptor().encryptFilename(BaseEncoding.base32(), name, dirIdInfo(cryptoParent).id.toByteArray(StandardCharsets.UTF_8)) var ciphertextName = prefix + cryptor().fileNameCryptor().encryptFilename(BaseEncoding.base32(), name, getOrCreateCachingAwareDirIdInfo(cryptoParent).id.toByteArray(StandardCharsets.UTF_8))
if (ciphertextName.length > shorteningThreshold) { if (ciphertextName.length > shorteningThreshold) {
ciphertextName = deflate(ciphertextName) ciphertextName = deflate(ciphertextName)
} }
@ -120,8 +120,8 @@ internal class CryptoImplVaultFormatPre7(
@Throws(BackendException::class) @Throws(BackendException::class)
override fun list(cryptoFolder: CryptoFolder): List<CryptoNode> { override fun list(cryptoFolder: CryptoFolder): List<CryptoNode> {
val dirIdInfo = dirIdInfo(cryptoFolder) val dirIdInfo = getDirIdInfo(cryptoFolder) ?: throw NoDirFileException(cryptoFolder.name, cryptoFolder.dirFile?.path)
val dirId = dirIdInfo(cryptoFolder).id val dirId = dirIdInfo.id
val lvl2Dir = dirIdInfo.cloudFolder val lvl2Dir = dirIdInfo.cloudFolder
return cloudContentRepository return cloudContentRepository
.list(lvl2Dir) .list(lvl2Dir)
@ -198,7 +198,7 @@ internal class CryptoImplVaultFormatPre7(
@Throws(BackendException::class) @Throws(BackendException::class)
override fun symlink(cryptoParent: CryptoFolder, cleartextName: String, target: String): CryptoSymlink { override fun symlink(cryptoParent: CryptoFolder, cleartextName: String, target: String): CryptoSymlink {
val ciphertextName = encryptSymlinkName(cryptoParent, cleartextName) val ciphertextName = encryptSymlinkName(cryptoParent, cleartextName)
val cloudFile = cloudContentRepository.file(dirIdInfo(cryptoParent).cloudFolder, ciphertextName) val cloudFile = cloudContentRepository.file(getOrCreateCachingAwareDirIdInfo(cryptoParent).cloudFolder, ciphertextName)
return CryptoSymlink(cryptoParent, cleartextName, path(cryptoParent, cleartextName), target, cloudFile) return CryptoSymlink(cryptoParent, cleartextName, path(cryptoParent, cleartextName), target, cloudFile)
} }
@ -237,9 +237,13 @@ internal class CryptoImplVaultFormatPre7(
requireNotNull(node.dirFile) requireNotNull(node.dirFile)
val cryptoSubfolders = deepCollectSubfolders(node) val cryptoSubfolders = deepCollectSubfolders(node)
for (cryptoSubfolder in cryptoSubfolders) { for (cryptoSubfolder in cryptoSubfolders) {
cloudContentRepository.delete(dirIdInfo(cryptoSubfolder).cloudFolder) getCachingAwareDirIdInfo(cryptoSubfolder)?.let {
cloudContentRepository.delete(it.cloudFolder)
} ?: Timber.tag("CryptoFs").w("Dir file doesn't exists of a sub folder while deleting the parent, continue anyway")
} }
cloudContentRepository.delete(dirIdInfo(node).cloudFolder) getCachingAwareDirIdInfo(node)?.let {
cloudContentRepository.delete(it.cloudFolder)
} ?: Timber.tag("CryptoFs").w("Dir file doesn't exists while deleting the folder, continue anyway")
cloudContentRepository.delete(node.dirFile) cloudContentRepository.delete(node.dirFile)
evictFromCache(node) evictFromCache(node)
} else if (node is CryptoFile) { } else if (node is CryptoFile) {
@ -248,22 +252,28 @@ internal class CryptoImplVaultFormatPre7(
} }
@Throws(BackendException::class, EmptyDirFileException::class) @Throws(BackendException::class, EmptyDirFileException::class)
override fun loadDirId(folder: CryptoFolder): String { override fun loadDirId(folder: CryptoFolder): String? {
return if (isRoot(folder)) { return if (isRoot(folder)) {
CryptoConstants.ROOT_DIR_ID CryptoConstants.ROOT_DIR_ID
} else if (folder.dirFile != null && cloudContentRepository.exists(folder.dirFile)) { } else if (folder.dirFile != null && cloudContentRepository.exists(folder.dirFile)) {
String(loadContentsOfDirFile(folder), StandardCharsets.UTF_8) String(loadContentsOfDirFile(folder), StandardCharsets.UTF_8)
} else { } else {
newDirId() null
} }
} }
@Throws(BackendException::class) @Throws(BackendException::class)
override fun createDirIdInfo(folder: CryptoFolder): DirIdInfo { override fun getOrCreateDirIdInfo(folder: CryptoFolder): DirIdInfo {
val dirId = loadDirId(folder) val dirId = loadDirId(folder) ?: newDirId()
return dirIdCache.put(folder, createDirIdInfoFor(dirId)) return dirIdCache.put(folder, createDirIdInfoFor(dirId))
} }
override fun getDirIdInfo(folder: CryptoFolder): DirIdInfo? {
return loadDirId(folder)?.let {
dirIdCache.put(folder, createDirIdInfoFor(it))
}
}
@Throws(BackendException::class) @Throws(BackendException::class)
override fun write(cryptoFile: CryptoFile, data: DataSource, progressAware: ProgressAware<UploadState>, replace: Boolean, length: Long): CryptoFile { override fun write(cryptoFile: CryptoFile, data: DataSource, progressAware: ProgressAware<UploadState>, replace: Boolean, length: Long): CryptoFile {
return writeShortNameFile(cryptoFile, data, progressAware, replace, length) return writeShortNameFile(cryptoFile, data, progressAware, replace, length)

View File

@ -42,7 +42,6 @@ import java.io.InputStreamReader
import java.io.OutputStream import java.io.OutputStream
import java.nio.ByteBuffer import java.nio.ByteBuffer
import java.nio.charset.StandardCharsets import java.nio.charset.StandardCharsets
import java.util.ArrayList
import kotlin.io.path.createTempDirectory import kotlin.io.path.createTempDirectory
import kotlin.io.path.deleteExisting import kotlin.io.path.deleteExisting
@ -730,6 +729,7 @@ class CryptoImplVaultFormat7Test {
whenever(cloudContentRepository.folder(aaFolder, shortenedFileName)).thenReturn(testDir3) whenever(cloudContentRepository.folder(aaFolder, shortenedFileName)).thenReturn(testDir3)
whenever(cloudContentRepository.exists(testDir3)).thenReturn(false) whenever(cloudContentRepository.exists(testDir3)).thenReturn(false)
whenever(dirIdCache.put(eq(cryptoFolder3), any())).thenReturn(DirIdInfo("dir3-id", ddFolder)) whenever(dirIdCache.put(eq(cryptoFolder3), any())).thenReturn(DirIdInfo("dir3-id", ddFolder))
whenever(dirIdCache[cryptoFolder3]).thenReturn(DirIdInfo("dir3-id", ddFolder))
whenever(cloudContentRepository.file(testDir3, "dir.c9r")).thenReturn(testDir3DirFile) whenever(cloudContentRepository.file(testDir3, "dir.c9r")).thenReturn(testDir3DirFile)
whenever(cloudContentRepository.file(testDir3, "name.c9s", 257L)).thenReturn(testDir3NameFile) whenever(cloudContentRepository.file(testDir3, "name.c9s", 257L)).thenReturn(testDir3NameFile)
whenever<List<*>>(cloudContentRepository.list(ddFolder)).thenReturn(ArrayList<CloudNode>()) whenever<List<*>>(cloudContentRepository.list(ddFolder)).thenReturn(ArrayList<CloudNode>())

View File

@ -33,6 +33,7 @@ import org.mockito.AdditionalMatchers
import org.mockito.Mockito import org.mockito.Mockito
import org.mockito.invocation.InvocationOnMock import org.mockito.invocation.InvocationOnMock
import org.mockito.kotlin.any import org.mockito.kotlin.any
import org.mockito.kotlin.anyOrNull
import org.mockito.kotlin.eq import org.mockito.kotlin.eq
import org.mockito.kotlin.mock import org.mockito.kotlin.mock
import org.mockito.kotlin.whenever import org.mockito.kotlin.whenever
@ -43,7 +44,6 @@ import java.io.InputStreamReader
import java.io.OutputStream import java.io.OutputStream
import java.nio.ByteBuffer import java.nio.ByteBuffer
import java.nio.charset.StandardCharsets import java.nio.charset.StandardCharsets
import java.util.ArrayList
import java.util.function.Supplier import java.util.function.Supplier
import kotlin.io.path.createTempDirectory import kotlin.io.path.createTempDirectory
import kotlin.io.path.deleteExisting import kotlin.io.path.deleteExisting
@ -138,11 +138,11 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(cloudContentRepository.folder(lvl2Dir, "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")).thenReturn(aaFolder) whenever(cloudContentRepository.folder(lvl2Dir, "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")).thenReturn(aaFolder)
whenever(cloudContentRepository.file(aaFolder, "0dir1")).thenReturn(testDir1) whenever(cloudContentRepository.file(aaFolder, "0dir1")).thenReturn(testDir1)
whenever(cloudContentRepository.exists(testDir1)).thenReturn(true) whenever(cloudContentRepository.exists(testDir1)).thenReturn(true)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(cryptoFolder1.dirFile!!), any(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(dirId1.toByteArray()), out) copyStreamToStream(ByteArrayInputStream(dirId1.toByteArray()), out)
null null
}.`when`(cloudContentRepository).read(eq(cryptoFolder1.dirFile!!), any(), any(), any()) }
whenever<List<*>>(cloudContentRepository.list(aaFolder)).thenReturn(rootItems) whenever<List<*>>(cloudContentRepository.list(aaFolder)).thenReturn(rootItems)
whenever(dirIdCache.put(eq(root), any())).thenReturn(DirIdInfo("", aaFolder)) whenever(dirIdCache.put(eq(root), any())).thenReturn(DirIdInfo("", aaFolder))
} }
@ -193,11 +193,11 @@ internal class CryptoImplVaultFormatPre7Test {
val cryptoFolder3 = CryptoFolder(cryptoFolder1, dir3Name, "/Directory 1/$dir3Name", testDir3DirFile) val cryptoFolder3 = CryptoFolder(cryptoFolder1, dir3Name, "/Directory 1/$dir3Name", testDir3DirFile)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(cryptoFolder3.dirFile!!), anyOrNull(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream("dir3-id".toByteArray()), out) copyStreamToStream(ByteArrayInputStream("dir3-id".toByteArray()), out)
null null
}.`when`(cloudContentRepository).read(eq(cryptoFolder3.dirFile!!), any(), any(), any()) }
/* /*
* Directory 3x250 * Directory 3x250
@ -217,11 +217,11 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(cloudContentRepository.file(directory4x250, "name.c9s")).thenReturn(testDir4NameFile) whenever(cloudContentRepository.file(directory4x250, "name.c9s")).thenReturn(testDir4NameFile)
whenever(fileNameCryptor.encryptFilename(BaseEncoding.base32(), dir4Name, "dir3-id".toByteArray())).thenReturn(dir4Cipher) whenever(fileNameCryptor.encryptFilename(BaseEncoding.base32(), dir4Name, "dir3-id".toByteArray())).thenReturn(dir4Cipher)
whenever(fileNameCryptor.decryptFilename(BaseEncoding.base32(), dir4Cipher, "dir3-id".toByteArray())).thenReturn(dir4Name) whenever(fileNameCryptor.decryptFilename(BaseEncoding.base32(), dir4Cipher, "dir3-id".toByteArray())).thenReturn(dir4Name)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(testDir4NameFile), any(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(dir4Cipher.toByteArray(charset("UTF-8"))), out) copyStreamToStream(ByteArrayInputStream(dir4Cipher.toByteArray(charset("UTF-8"))), out)
null null
}.`when`(cloudContentRepository).read(eq(testDir4NameFile), any(), any(), any()) }
val dir4Files: ArrayList<CloudNode> = object : ArrayList<CloudNode>() { val dir4Files: ArrayList<CloudNode> = object : ArrayList<CloudNode>() {
init { init {
@ -242,11 +242,12 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(cloudContentRepository.file(directory5x250, "name.c9s")).thenReturn(testFile5NameFile) whenever(cloudContentRepository.file(directory5x250, "name.c9s")).thenReturn(testFile5NameFile)
whenever(fileNameCryptor.encryptFilename(BaseEncoding.base32(), file5Name, "dir3-id".toByteArray())).thenReturn(file5Cipher) whenever(fileNameCryptor.encryptFilename(BaseEncoding.base32(), file5Name, "dir3-id".toByteArray())).thenReturn(file5Cipher)
whenever(fileNameCryptor.decryptFilename(BaseEncoding.base32(), file5Cipher, "dir3-id".toByteArray())).thenReturn(file5Name) whenever(fileNameCryptor.decryptFilename(BaseEncoding.base32(), file5Cipher, "dir3-id".toByteArray())).thenReturn(file5Name)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(testFile5NameFile), any(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(file5Cipher.toByteArray(charset("UTF-8"))), out) copyStreamToStream(ByteArrayInputStream(file5Cipher.toByteArray(charset("UTF-8"))), out)
null null
}.`when`(cloudContentRepository).read(eq(testFile5NameFile), any(), any(), any()) }
val dir5Files: ArrayList<CloudNode> = object : ArrayList<CloudNode>() { val dir5Files: ArrayList<CloudNode> = object : ArrayList<CloudNode>() {
init { init {
add(testFile5ContentFile) add(testFile5ContentFile)
@ -286,11 +287,11 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(fileHeaderCryptor.decryptHeader(StandardCharsets.UTF_8.encode("hhhhh"))).thenReturn(header) whenever(fileHeaderCryptor.decryptHeader(StandardCharsets.UTF_8.encode("hhhhh"))).thenReturn(header)
whenever(fileContentCryptor.decryptChunk(eq(StandardCharsets.UTF_8.encode("TOPSECRET!")), any(), eq(header), any())) whenever(fileContentCryptor.decryptChunk(eq(StandardCharsets.UTF_8.encode("TOPSECRET!")), any(), eq(header), any()))
.then { invocation: InvocationOnMock? -> StandardCharsets.UTF_8.encode("geheim!!") } .then { invocation: InvocationOnMock? -> StandardCharsets.UTF_8.encode("geheim!!") }
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(cryptoFile1.cloudFile), any(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(file1Content), out) copyStreamToStream(ByteArrayInputStream(file1Content), out)
null null
}.`when`(cloudContentRepository).read(eq(cryptoFile1.cloudFile), any(), any(), any()) }
val outputStream = ByteArrayOutputStream(1000) val outputStream = ByteArrayOutputStream(1000)
inTest.read(cryptoFile1, outputStream, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD) inTest.read(cryptoFile1, outputStream, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD)
@ -318,11 +319,11 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(fileContentCryptor.decryptChunk(eq(StandardCharsets.UTF_8.encode("TOPSECRET!")), any(), eq(header), any())) whenever(fileContentCryptor.decryptChunk(eq(StandardCharsets.UTF_8.encode("TOPSECRET!")), any(), eq(header), any()))
.then { invocation: InvocationOnMock? -> StandardCharsets.UTF_8.encode("geheim!!") } .then { invocation: InvocationOnMock? -> StandardCharsets.UTF_8.encode("geheim!!") }
val cryptoFile15 = CryptoFile(root, file3Name, "/$file3Name", null, testFile3ContentFile) val cryptoFile15 = CryptoFile(root, file3Name, "/$file3Name", null, testFile3ContentFile)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(cryptoFile15.cloudFile), any(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(file1Content), out) copyStreamToStream(ByteArrayInputStream(file1Content), out)
null null
}.`when`(cloudContentRepository).read(eq(cryptoFile15.cloudFile), any(), any(), any()) }
val outputStream = ByteArrayOutputStream(1000) val outputStream = ByteArrayOutputStream(1000)
inTest.read(cryptoFile15, outputStream, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD) inTest.read(cryptoFile15, outputStream, ProgressAware.NO_OP_PROGRESS_AWARE_DOWNLOAD)
@ -593,11 +594,11 @@ internal class CryptoImplVaultFormatPre7Test {
val testDir2DirFile = TestFile(bbFolder, "0dir2", "/d/11/BBBBBBBBBBBBBBBBBBBBBBBBBBBBBB/0dir2", null, null) val testDir2DirFile = TestFile(bbFolder, "0dir2", "/d/11/BBBBBBBBBBBBBBBBBBBBBBBBBBBBBB/0dir2", null, null)
val cryptoFolder2 = CryptoFolder(cryptoFolder1, "Directory 2", "/Directory 1/Directory 2", testDir2DirFile) val cryptoFolder2 = CryptoFolder(cryptoFolder1, "Directory 2", "/Directory 1/Directory 2", testDir2DirFile)
Mockito.doAnswer { invocation: InvocationOnMock -> whenever(cloudContentRepository.read(eq(cryptoFolder2.dirFile!!), anyOrNull(), any(), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val out = invocation.getArgument<OutputStream>(2) val out = invocationOnMock.getArgument<OutputStream>(2)
copyStreamToStream(ByteArrayInputStream(dirId2.toByteArray()), out) copyStreamToStream(ByteArrayInputStream(dirId2.toByteArray()), out)
null null
}.`when`(cloudContentRepository).read(eq(cryptoFolder2.dirFile!!), any(), any(), any()) }
val dir1Items: ArrayList<CloudNode> = object : ArrayList<CloudNode>() { val dir1Items: ArrayList<CloudNode> = object : ArrayList<CloudNode>() {
init { init {
@ -657,6 +658,7 @@ internal class CryptoImplVaultFormatPre7Test {
whenever(cloudContentRepository.folder(d, "33")).thenReturn(ddLvl2Dir) whenever(cloudContentRepository.folder(d, "33")).thenReturn(ddLvl2Dir)
whenever(cloudContentRepository.folder(lvl2Dir, "DDDDDDDDDDDDDDDDDDDDDDDDDDDDDD")).thenReturn(ddFolder) whenever(cloudContentRepository.folder(lvl2Dir, "DDDDDDDDDDDDDDDDDDDDDDDDDDDDDD")).thenReturn(ddFolder)
whenever(dirIdCache.put(eq(cryptoFolder3), any())).thenReturn(DirIdInfo("dir3-id", ddFolder)) whenever(dirIdCache.put(eq(cryptoFolder3), any())).thenReturn(DirIdInfo("dir3-id", ddFolder))
whenever(dirIdCache[cryptoFolder3]).thenReturn(DirIdInfo("dir3-id", ddFolder))
whenever(cloudContentRepository.file(aaFolder, shortenedFileName)).thenReturn(testDir3DirFile) whenever(cloudContentRepository.file(aaFolder, shortenedFileName)).thenReturn(testDir3DirFile)
whenever(cloudContentRepository.file(testDir3NameFile.parent!!, shortenedFileName, 257L)).thenReturn(testDir3NameFile) whenever(cloudContentRepository.file(testDir3NameFile.parent!!, shortenedFileName, 257L)).thenReturn(testDir3NameFile)
whenever<List<*>>(cloudContentRepository.list(ddFolder)).thenReturn(ArrayList<CloudNode>()) whenever<List<*>>(cloudContentRepository.list(ddFolder)).thenReturn(ArrayList<CloudNode>())
@ -767,7 +769,7 @@ internal class CryptoImplVaultFormatPre7Test {
val cryptoMovedFile4 = CryptoFile(cryptoFolder1, file4Name, "/Directory 1/$file4Name", null, testFile4ContentFile) val cryptoMovedFile4 = CryptoFile(cryptoFolder1, file4Name, "/Directory 1/$file4Name", null, testFile4ContentFile)
whenever(cloudContentRepository.move(testFile4ContentFileOld, testFile4ContentFile)).thenReturn(testFile4ContentFile) whenever(cloudContentRepository.move(testFile4ContentFileOld, testFile4ContentFile)).thenReturn(testFile4ContentFile)
whenever(cloudContentRepository.create(testFile4NameFile.parent!!)).thenReturn(testFile4NameFile.parent) whenever(cloudContentRepository.create(testFile4NameFile.parent)).thenReturn(testFile4NameFile.parent)
whenever(cloudContentRepository.write(eq(testFile4NameFile), any(), any(), eq(true), any())).thenAnswer { invocationOnMock: InvocationOnMock -> whenever(cloudContentRepository.write(eq(testFile4NameFile), any(), any(), eq(true), any())).thenAnswer { invocationOnMock: InvocationOnMock ->
val inputStream = invocationOnMock.getArgument<DataSource>(1) val inputStream = invocationOnMock.getArgument<DataSource>(1)
val dirContent = BufferedReader(InputStreamReader(inputStream.open(context)!!, StandardCharsets.UTF_8)).readLine() val dirContent = BufferedReader(InputStreamReader(inputStream.open(context)!!, StandardCharsets.UTF_8)).readLine()