aboutsummaryrefslogtreecommitdiff
path: root/src/main/kotlin/FileController.kt
diff options
context:
space:
mode:
authorMarvin Borner2019-04-16 23:52:59 +0200
committerMarvin Borner2019-04-16 23:52:59 +0200
commitcdd6fa1981ebf8d96e4849167805c471451e5d6d (patch)
tree80d9ac1670d6f08a5ec0bd540fd9d96a8fa2be6f /src/main/kotlin/FileController.kt
parentf84c2ada032efee0b9d9eb03c63b10458f135dcd (diff)
Abstracted functions to classes
Co-authored-by: LarsVomMars <lars@kroenner.eu>
Diffstat (limited to 'src/main/kotlin/FileController.kt')
-rw-r--r--src/main/kotlin/FileController.kt165
1 files changed, 165 insertions, 0 deletions
diff --git a/src/main/kotlin/FileController.kt b/src/main/kotlin/FileController.kt
new file mode 100644
index 0000000..08f0763
--- /dev/null
+++ b/src/main/kotlin/FileController.kt
@@ -0,0 +1,165 @@
+package space.anity
+
+import io.javalin.*
+import io.javalin.core.util.*
+import io.javalin.rendering.template.*
+import java.io.*
+import java.nio.charset.*
+import java.nio.file.*
+import java.text.*
+import java.util.*
+
+class FileController {
+ /**
+ * Crawls the requested file and either renders the directory view or the file view
+ */
+ fun crawl(ctx: Context) {
+ try {
+ val usersFileHome = "$fileHome/${userHandler.getVerifiedUserId(ctx)}"
+ File(usersFileHome).mkdirs()
+ when {
+ File("$usersFileHome/${ctx.splats()[0]}").isDirectory -> {
+ val files = ArrayList<Array<String>>()
+ Files.list(Paths.get("$usersFileHome/${ctx.splats()[0]}/")).forEach {
+ 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)
+ 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(),
+ 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(
+ "files.rocker.html", TemplateUtil.model(
+ "files", files,
+ "path", ctx.splats()[0]
+ )
+ )
+ }
+ isHumanReadable(File("$usersFileHome/${ctx.splats()[0]}")) ->
+ ctx.render(
+ "fileview.rocker.html", TemplateUtil.model(
+ "content", Files.readAllLines(
+ Paths.get("$usersFileHome/${ctx.splats()[0]}"),
+ Charsets.UTF_8
+ ).joinToString(separator = "\n"),
+ "filename", File("$usersFileHome/${ctx.splats()[0]}").name,
+ "extension", File("$usersFileHome/${ctx.splats()[0]}").extension
+ )
+ )
+ else -> ctx.result(FileInputStream(File("$usersFileHome/${ctx.splats()[0]}")))
+ }
+ } catch (_: Exception) {
+ throw NotFoundResponse("Error: File or directory does not exist.")
+ }
+ }
+
+ /**
+ * Gets directory size recursively
+ */
+ private fun getDirectorySize(directory: File): Long {
+ var length: Long = 0
+ for (file in directory.listFiles()!!) {
+ length += if (file.isFile) file.length()
+ else getDirectorySize(file)
+ }
+ return length
+ }
+
+ /**
+ * Checks whether the file is binary or human-readable (text)
+ */
+ private fun isHumanReadable(file: File): Boolean {
+ val input = FileInputStream(file)
+ var size = input.available()
+ if (size > 1000) size = 1000
+ val data = ByteArray(size)
+ input.read(data)
+ input.close()
+ val text = String(data, Charset.forName("ISO-8859-1"))
+ val replacedText = text.replace(
+ ("[a-zA-Z0-9ßöäü\\.\\*!\"§\\$\\%&/()=\\?@~'#:,;\\+><\\|\\[\\]\\{\\}\\^°²³\\\\ \\n\\r\\t_\\-`´âêîôÂÊÔÎáéíóàèìòÁÉÍÓÀÈÌÒ©‰¢£¥€±¿»«¼½¾™ª]").toRegex(),
+ ""
+ )
+ val d = (text.length - replacedText.length).toDouble() / text.length.toDouble()
+ return d > 0.95
+ }
+
+ private fun humanReadableBytes(bytes: Long): String {
+ val unit = 1024
+ if (bytes < unit) return "$bytes B"
+ val exp = (Math.log(bytes.toDouble()) / Math.log(unit.toDouble())).toInt()
+ val pre = "KMGTPE"[exp - 1] + "i"
+ return String.format("%.1f %sB", bytes / Math.pow(unit.toDouble(), exp.toDouble()), pre)
+ }
+
+ /**
+ * Saves multipart media data into requested directory
+ */
+ 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))
+ }
+ }
+
+ /**
+ * Deletes the requested file
+ */
+ fun delete(ctx: Context) {
+ val userId = userHandler.getVerifiedUserId(ctx)
+ if (userId > 0) {
+ val path = ctx.splats()[0]
+ File("$fileHome/$userId/$path").delete()
+ databaseController.deleteFile(path, userId)
+ }
+ }
+
+ /**
+ * Shares the requested file via the accessId
+ */
+ fun share(ctx: Context) {
+ val userId = userHandler.getVerifiedUserId(ctx)
+ if (userId > 0) {
+ val accessId = databaseController.getAccessId(ctx.splats()[0], userId)
+ ctx.result("${ctx.host()}/shared?id=$accessId")
+ }
+ }
+
+ /**
+ * Renders the shared file
+ */
+ 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))) {
+ 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
+ )
+ )
+ } else ctx.result(FileInputStream(File(sharedFileLocation))) // TODO: other file types
+ } else {
+ log.info("Unknown file!")
+ }
+ }
+
+}