aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarvin Borner2019-04-19 01:08:21 +0200
committerMarvin Borner2019-04-19 01:08:21 +0200
commitfe97c12731193211caf8652a6fd40e02d0d429b4 (patch)
treec6cb62b0d625c94b15a6c888693599c6fbef8568
parent0e7c4ef365d21e1e283c814124dba318a7772603 (diff)
Added sharing of directories
Co-authored-by: LarsVomMars <lars@kroenner.eu>
-rw-r--r--src/main/kotlin/DatabaseController.kt36
-rw-r--r--src/main/kotlin/FileController.kt87
-rw-r--r--src/main/resources/js/files.js5
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