diff options
author | Marvin Borner | 2019-04-19 01:08:21 +0200 |
---|---|---|
committer | Marvin Borner | 2019-04-19 01:08:21 +0200 |
commit | fe97c12731193211caf8652a6fd40e02d0d429b4 (patch) | |
tree | c6cb62b0d625c94b15a6c888693599c6fbef8568 | |
parent | 0e7c4ef365d21e1e283c814124dba318a7772603 (diff) |
Added sharing of directories
Co-authored-by: LarsVomMars <lars@kroenner.eu>
-rw-r--r-- | src/main/kotlin/DatabaseController.kt | 36 | ||||
-rw-r--r-- | src/main/kotlin/FileController.kt | 87 | ||||
-rw-r--r-- | src/main/resources/js/files.js | 5 |
3 files changed, 94 insertions, 34 deletions
diff --git a/src/main/kotlin/DatabaseController.kt b/src/main/kotlin/DatabaseController.kt index 51179e8..77f2e70 100644 --- a/src/main/kotlin/DatabaseController.kt +++ b/src/main/kotlin/DatabaseController.kt @@ -16,7 +16,8 @@ class DatabaseController(dbFileLocation: String = "main.db") { */ object FileLocation : Table() { val id = integer("id").autoIncrement().primaryKey() - val path = text("path").uniqueIndex() + val path = text("path").uniqueIndex() // TODO: Don't use uniqueIndex() or double-check this with userId + val isDirectory = bool("isDirectory").default(false) val userId = integer("userId").references(UserData.id) val accessId = varchar("accessId", 64).uniqueIndex() val isShared = bool("isShared").default(false) @@ -210,20 +211,24 @@ class DatabaseController(dbFileLocation: String = "main.db") { /** * Adds the uploaded file to the database */ - fun addFile(fileLocation: String, usersId: Int) { - transaction { + fun addFile(fileLocation: String, usersId: Int, isDirectoryBool: Boolean = false): Boolean { + return transaction { try { FileLocation.insert { it[path] = fileLocation it[userId] = usersId it[accessId] = generateRandomString() + it[isDirectory] = isDirectoryBool } + true } catch (err: org.jetbrains.exposed.exceptions.ExposedSQLException) { log.warning("File already exists!") + false } } } + /** * Removes the file from the database */ @@ -256,16 +261,23 @@ class DatabaseController(dbFileLocation: String = "main.db") { /** * Gets the shared file via [accessId] */ - fun getSharedFile(accessId: String): Pair<Int, String> { + fun getSharedFile(accessId: String): ReturnFileData { return transaction { try { - if (FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.isShared] }[0]) - FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.userId] to it[FileLocation.path] }[0] + if (FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.isShared] }[0]) { + val userId = + FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.userId] }[0] + val fileLocation = + FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.path] }[0] + val isDir = + FileLocation.select { FileLocation.accessId eq accessId }.map { it[FileLocation.isDirectory] }[0] + ReturnFileData(userId, fileLocation, isDir) + } else - Pair(-1, "") + ReturnFileData(-1, "", false) } catch (_: org.jetbrains.exposed.exceptions.ExposedSQLException) { log.warning("File does not exist!") - Pair(-1, "") + ReturnFileData(-1, "", false) } } } @@ -289,7 +301,7 @@ class DatabaseController(dbFileLocation: String = "main.db") { fun toggleSetup() { transaction { General.update({ General.initialUse eq false }) { - it[General.isSetup] = true + it[isSetup] = true } } } @@ -357,3 +369,9 @@ class DatabaseController(dbFileLocation: String = "main.db") { .joinToString("") } } + +data class ReturnFileData( + val userId: Int, + val fileLocation: String, + val isDirectory: Boolean +) // TODO: Think about using dataclass diff --git a/src/main/kotlin/FileController.kt b/src/main/kotlin/FileController.kt index e8526cf..d8f7dfb 100644 --- a/src/main/kotlin/FileController.kt +++ b/src/main/kotlin/FileController.kt @@ -3,11 +3,11 @@ package space.anity import io.javalin.* import io.javalin.core.util.* import io.javalin.rendering.template.* +import io.javalin.rendering.template.TemplateUtil.model import java.io.* import java.nio.charset.* import java.nio.file.* import java.text.* -import java.util.* class FileController { /** @@ -21,7 +21,7 @@ class FileController { File("$usersFileHome/${ctx.splats()[0]}").isDirectory -> { val files = ArrayList<Array<String>>() Files.list(Paths.get("$usersFileHome/${ctx.splats()[0]}/")).forEach { - val fileName = it.toString() + val filename = it.toString() .drop(usersFileHome.length + (if (ctx.splats()[0].isNotEmpty()) ctx.splats()[0].length + 2 else 1)) val filePath = "$usersFileHome${it.toString().drop(usersFileHome.length)}" val file = File(filePath) @@ -29,7 +29,7 @@ class FileController { files.add( // TODO: Clean up file array responses arrayOf( - if (file.isDirectory) "$fileName/" else fileName, + if (file.isDirectory) "$filename/" else filename, humanReadableBytes(fileSize), SimpleDateFormat("MM/dd/yyyy HH:mm:ss").format(file.lastModified()).toString(), if (file.isDirectory) "true" else isHumanReadable(file).toString(), @@ -113,11 +113,18 @@ class FileController { fun upload(ctx: Context) { ctx.uploadedFiles("file").forEach { (_, content, name, _) -> val path = "${ctx.splats()[0]}/$name" - FileUtil.streamToFile( - content, - "$fileHome/${userHandler.getVerifiedUserId(ctx)}/$path" - ) - databaseController.addFile(path, userHandler.getVerifiedUserId(ctx)) + val uid = userHandler.getVerifiedUserId(ctx) + var addPath = "" + path.split("/").forEach { + addPath += "$it/" + if (!path.endsWith(it)) databaseController.addFile(addPath, uid, true) + } + if (databaseController.addFile(path, uid)) { + FileUtil.streamToFile( + content, + "$fileHome/$uid/$path" + ) + } } } @@ -144,10 +151,12 @@ class FileController { /** * Shares the requested file via the accessId */ - fun share(ctx: Context) { // TODO: Add support for directory sharing + fun share(ctx: Context) { val userId = userHandler.getVerifiedUserId(ctx) + val shareType = ctx.queryParam("type").toString() if (userId > 0) { - val path = if (ctx.splats()[0].startsWith("/")) ctx.splats()[0] else "/${ctx.splats()[0]}" + val path = + "${(if (ctx.splats()[0].startsWith("/")) ctx.splats()[0] else "/${ctx.splats()[0]}")}${if (shareType == "dir") "/" else ""}" val accessId = databaseController.getAccessId(path, userId) ctx.result("${ctx.host()}/shared?id=$accessId") } @@ -159,22 +168,54 @@ class FileController { fun renderShared(ctx: Context) { val accessId = ctx.queryParam("id").toString() val sharedFileData = databaseController.getSharedFile(accessId) - if (sharedFileData.first > 0 && sharedFileData.second.isNotEmpty()) { - val sharedFileLocation = "$fileHome/${sharedFileData.first}/${sharedFileData.second}" - if (isHumanReadable(File(sharedFileLocation))) { + if (sharedFileData.userId > 0 && sharedFileData.fileLocation.isNotEmpty()) { + val sharedFileLocation = "$fileHome/${sharedFileData.userId}/${sharedFileData.fileLocation}" + if (!sharedFileData.isDirectory) { + if (isHumanReadable(File(sharedFileLocation))) { + ctx.render( + "fileview.rocker.html", model( + "content", Files.readAllLines( + Paths.get(sharedFileLocation), + Charsets.UTF_8 + ).joinToString(separator = "\n"), + "filename", File(sharedFileLocation).name, + "extension", File(sharedFileLocation).extension + ) + ) + } else { + ctx.contentType(Files.probeContentType(Paths.get(sharedFileLocation))) + ctx.result(FileInputStream(File(sharedFileLocation))) + } + } else { + // TODO: Add support for accessing files in shared directories + // TODO: Combine the two file-crawling-render functions + val files = ArrayList<Array<String>>() + Files.list(Paths.get(sharedFileLocation)).forEach { + val filename = it.toString() + .drop(sharedFileLocation.length - 1) + val filePath = "$sharedFileLocation$filename" + val file = File(filePath) + val fileSize = if (file.isDirectory) getDirectorySize(file) else file.length() + files.add( + // TODO: Clean up file array responses + arrayOf( + if (file.isDirectory) "$filename/" else filename, + humanReadableBytes(fileSize), + SimpleDateFormat("MM/dd/yyyy HH:mm:ss").format(file.lastModified()).toString(), + if (file.isDirectory) "true" else isHumanReadable(file).toString(), + file.isDirectory.toString(), + fileSize.toString(), // unformatted file size + file.lastModified().toString() // unformatted last modified date + ) + ) + } + //files.sortWith(String.CASE_INSENSITIVE_ORDER) // TODO: Reimplement file array sorting in backend ctx.render( - "fileview.rocker.html", TemplateUtil.model( - "content", Files.readAllLines( - Paths.get(sharedFileLocation), - Charsets.UTF_8 - ).joinToString(separator = "\n"), - "filename", File(sharedFileLocation).name, - "extension", File(sharedFileLocation).extension + "files.rocker.html", TemplateUtil.model( + "files", files, + "path", sharedFileData.fileLocation ) ) - } else { - ctx.contentType(Files.probeContentType(Paths.get(sharedFileLocation))) - ctx.result(FileInputStream(File(sharedFileLocation))) } } else { log.info("Unknown file!") diff --git a/src/main/resources/js/files.js b/src/main/resources/js/files.js index 523df0d..6016ba0 100644 --- a/src/main/resources/js/files.js +++ b/src/main/resources/js/files.js @@ -66,7 +66,7 @@ drop.addEventListener('drop', e => { iterateFiles(item); } else { formData.append("file", file); - request.open("POST", `/upload/${path}`); + request.open("POST", `/upload/${path}`, true); request.send(formData); } } @@ -167,8 +167,9 @@ function setListeners() { const request = new XMLHttpRequest(); const parent = e.target.closest("tr"); const fileName = parent.getAttribute("data-href") || parent.getAttribute("data-path"); + const type = fileName.endsWith('/') ? 'dir' : 'file'; - request.open("POST", `/share/${path}/${fileName}`); + request.open("POST", `/share/${path}/${fileName}?type=${type}`, true); request.onload = () => { if (request.readyState === 4) { if (request.status === 200) { // TODO: fix clipboard in Firefox |