diff options
Diffstat (limited to 'src/main/kotlin/DatabaseController.kt')
-rw-r--r-- | src/main/kotlin/DatabaseController.kt | 511 |
1 files changed, 0 insertions, 511 deletions
diff --git a/src/main/kotlin/DatabaseController.kt b/src/main/kotlin/DatabaseController.kt deleted file mode 100644 index 5316bb0..0000000 --- a/src/main/kotlin/DatabaseController.kt +++ /dev/null @@ -1,511 +0,0 @@ -package space.anity - -import at.favre.lib.crypto.bcrypt.* -import io.javalin.* -import io.javalin.rendering.template.TemplateUtil.model -import org.jetbrains.exposed.sql.* -import org.jetbrains.exposed.sql.transactions.* -import org.joda.time.* -import org.slf4j.* -import java.sql.* - -class DatabaseController { - private val dbFileLocation = - if (System.getProperty("os.name") != "Linux" || debug) "main.db" else "/usr/share/kloud/main.db" - val db: Database = Database.connect("jdbc:sqlite:$dbFileLocation", "org.sqlite.JDBC") - private val log = LoggerFactory.getLogger(this.javaClass.name) - - /** - * Database table indexing the file locations - */ - object FileLocation : Table() { - val id = integer("id").autoIncrement().primaryKey() - val path = text("path") - 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) - } - - /** - * Database table indexing the users with their regarding passwords - */ - object UserData : Table() { - val id = integer("id").autoIncrement().primaryKey() - val username = varchar("username", 24).uniqueIndex() - val password = varchar("password", 64) - val verification = varchar("verification", 64).uniqueIndex() - val darkTheme = bool("darkTheme").default(false) - } - - /** - * Database table indexing the users with their regarding role (multi line per user) - */ - object UserRoles : Table() { - val id = integer("id").autoIncrement().primaryKey() - val userId = integer("userId").references(UserData.id) - val roleId = integer("role").references(RolesData.id) - } - - /** - * Database table indexing the soon-to-be registered users by username - */ - object UserRegistration : Table() { - val id = integer("id").autoIncrement().primaryKey() - val username = varchar("username", 24).uniqueIndex() - val token = varchar("token", 64).uniqueIndex() - } - - /** - * Database table declaring available roles - */ - object RolesData : Table() { - val id = integer("id").autoIncrement().primaryKey() - val role = varchar("roles", 16) - } - - /** - * Database table indexing the login attempts of an ip in combination with the timestamp - */ - object LoginAttempts : Table() { - val id = integer("id").autoIncrement().primaryKey() - val ip = varchar("ip", 16) - val timestamp = datetime("timestamp") - } - - /** - * Database table storing general data/states - */ - object General : Table() { - val id = integer("id").autoIncrement().primaryKey() - val initialUse = bool("initialUse").default(true) - val isSetup = bool("isSetup").default(false) - } - - init { - // Create connection - TransactionManager.manager.defaultIsolationLevel = Connection.TRANSACTION_SERIALIZABLE - - // Add tables - transaction { - SchemaUtils.createMissingTablesAndColumns( - FileLocation, - UserData, - UserRoles, - UserRegistration, - RolesData, - LoginAttempts, - General - ) - } - } - - /** - * Creates the user in the database using username, password and the role - */ - fun createUser(usernameString: String, passwordString: String, roleString: String): Boolean { - return transaction { - try { - val usersId = UserData.insert { - it[username] = usernameString - it[password] = BCrypt.withDefaults().hashToString(12, passwordString.toCharArray()) - it[verification] = generateRandomString() - }[UserData.id] - - UserRoles.insert { roles -> - roles[userId] = usersId!! - roles[roleId] = RolesData.select { RolesData.role eq roleString }.map { it[RolesData.id] }[0] - } - true - } catch (_: Exception) { - log.warn("User already exists!") - false - } - } - } - - /** - * Checks whether the user is allowed to register - */ - fun isUserRegistrationValid(usernameString: String, tokenString: String): Boolean { - return transaction { - try { - if (UserData.select { UserData.username eq usernameString }.empty() && - UserRegistration.select { UserRegistration.token eq tokenString }.map { it[UserRegistration.token] }[0] == tokenString - ) { - usernameString == UserRegistration.select { UserRegistration.username eq usernameString }.map { it[UserRegistration.username] }[0] - } else false - } catch (err: Exception) { - log.error(err.toString()) - false - } - } - } - - /** - * Adds a user to the registration table - */ - fun indexUserRegistration(ctx: Context) { - val usernameString = ctx.formParam("username", "").toString() - - if (usernameString.matches("[a-zA-Z0-9]+".toRegex()) && usernameString.length > 3) { - val tokenString = generateRandomString() - var error = false - - transaction { - try { - UserRegistration.insert { - it[username] = usernameString - it[token] = tokenString - } - } catch (err: Exception) { - log.error(err.toString()) - error = true - } - } - - if (error) ctx.render("admin.rocker.html", model("message", "User already exists!", "ctx", ctx)) - else ctx.render( - "admin.rocker.html", model( - "message", "http://${ctx.host()}/user/register?username=$usernameString&token=$tokenString", - "ctx", ctx - ) - ) - } else ctx.render("admin.rocker.html", model("message", "Please only use alphabetical characters!", "ctx", ctx)) - } - - /** - * Removes the registration index of [usernameString] - */ - fun removeRegistrationIndex(usernameString: String) { - transaction { - UserRegistration.deleteWhere { UserRegistration.username eq usernameString } - } - } - - /** - * Tests whether the password [passwordString] of the user [usernameString] is correct - */ - fun checkUser(usernameString: String, passwordString: String): Boolean { - return transaction { - try { - val passwordHash = - UserData.select { UserData.username eq usernameString }.map { it[UserData.password] }[0] - BCrypt.verifyer().verify(passwordString.toCharArray(), passwordHash).verified - } catch (err: Exception) { - log.error(err.toString()) - false - } - } - } - - /** - * Returns the corresponding username using [userId] - */ - fun getUsername(userId: Int): String { - return transaction { - try { - UserData.select { UserData.id eq userId }.map { it[UserData.username] }[0] - } catch (err: Exception) { - log.error(err.toString()) - "" - } - } - } - - /** - * Returns the corresponding username using [verificationId] - */ - fun getUserIdByVerificationId(verificationId: String): Int { - return transaction { - try { - UserData.select { UserData.verification eq verificationId }.map { it[UserData.id] }[0] - } catch (err: Exception) { - -1 - } - } - } - - /** - * Returns true when user uses dark theme - */ - fun isDarkTheme(userId: Int): Boolean { - return transaction { - try { - UserData.select { UserData.id eq userId }.map { it[UserData.darkTheme] }[0] - } catch (err: Exception) { - false - } - } - } - - /** - * Toggles the dark theme - */ - fun toggleDarkTheme(userId: Int) { - return transaction { - try { - UserData.update({ (UserData.id eq userId) }) { - it[darkTheme] = !isDarkTheme(userId) - } - } catch (err: Exception) { - log.error(err.toString()) - } - } - } - - /** - * Returns the corresponding verification id using [usernameString] - */ - fun getVerificationId(usernameString: String): String { - return transaction { - try { - UserData.select { UserData.username eq usernameString }.map { it[UserData.verification] }[0] - } catch (err: Exception) { - log.error(err.toString()) - "" - } - } - } - - /** - * Returns the corresponding userId using [usernameString] - */ - fun getUserId(usernameString: String): Int { - return transaction { - try { - UserData.select { UserData.username eq usernameString }.map { it[UserData.id] }[0] - } catch (err: Exception) { - log.error(err.toString()) - -1 - } - } - } - - /** - * Returns the corresponding role using [userId] - */ - fun getRoles(userId: Int): List<Roles> { - return transaction { - try { - val userRoleId = UserRoles.select { UserRoles.userId eq userId }.map { it[UserRoles.roleId] }[0] - - val userRoles = mutableListOf<Roles>() - RolesData.select { RolesData.id eq userRoleId }.map { it[RolesData.role] }.forEach { - when (Roles.valueOf(it)) { - Roles.GUEST -> { - userRoles.add(Roles.GUEST) - } - Roles.USER -> { - userRoles.add(Roles.USER) - } - Roles.ADMIN -> { - userRoles.add(Roles.GUEST) - userRoles.add(Roles.USER) - userRoles.add(Roles.ADMIN) - } - } - } - userRoles - } catch (err: Exception) { - listOf(Roles.GUEST) - } - } - } - - /** - * Adds the uploaded file to the database - */ - fun addFile(fileLocation: String, usersId: Int, isDirectoryBool: Boolean = false): Boolean { - return transaction { - try { - if (FileLocation.select { (FileLocation.path eq fileLocation) and (FileLocation.userId eq usersId) }.empty()) { - FileLocation.insert { - it[path] = fileLocation - it[userId] = usersId - it[accessId] = generateRandomString() - it[isDirectory] = isDirectoryBool - } - true - } else { - if (!isDirectoryBool && debug) log.warn("File already exists!") - false - } - } catch (err: Exception) { - error(err) - } - } - } - - /** - * Removes the file from the database - */ - fun deleteFile(fileLocation: String, userId: Int) { - transaction { - try { - // TODO: Think of new solution for directory deleting (instead of wildcards) - FileLocation.deleteWhere { (FileLocation.path like "$fileLocation%") and (FileLocation.userId eq userId) } - } catch (err: Exception) { - log.error(err.toString()) - log.warn("File does not exist!") - } - } - } - - /** - * Returns the accessId of the given file - */ - fun getAccessId(fileLocation: String, userId: Int): String { - return transaction { - try { - FileLocation.update({ (FileLocation.userId eq userId) and (FileLocation.path like "$fileLocation%") }) { - it[isShared] = true - } - FileLocation.select { (FileLocation.path eq fileLocation) and (FileLocation.userId eq userId) }.map { it[FileLocation.accessId] }[0] - } catch (err: Exception) { - log.error(err.toString()) - "" - } - } - } - - /** - * Returns accessId of file in directory - */ - fun getAccessIdOfDirectory(filename: String, accessId: String): String { - return transaction { - try { - val fileData = - FileLocation.select { - FileLocation.accessId eq accessId - }.map { it[FileLocation.path] to it[FileLocation.userId] to it[FileLocation.isShared] }[0] - if (fileData.second) - FileLocation.select { - (FileLocation.path eq "${fileData.first.first}${filename.substring(1)}") and (FileLocation.userId eq fileData.first.second) - }.map { it[FileLocation.accessId] }[0] - else "" - } catch (err: Exception) { - log.error(err.toString()) - "" - } - } - } - - /** - * Gets the shared file via [accessId] - */ - fun getSharedFile(accessId: String): ReturnFileData { - return transaction { - try { - 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 - ReturnFileData(-1, "", false) - } catch (err: Exception) { - log.error(err.toString()) - log.warn("File does not exist!") - ReturnFileData(-1, "", false) - } - } - } - - /** - * Checks whether the site has been set up - */ - fun isSetup(): Boolean { - return transaction { - try { - General.selectAll().map { it[General.isSetup] }[0] - } catch (err: Exception) { - log.error(err.toString()) - false - } - } - } - - /** - * Toggles the setup state - */ - fun toggleSetup() { - transaction { - General.update({ General.initialUse eq false }) { - it[isSetup] = true - } - } - } - - /** - * Adds an login attempt to the database - */ - fun loginAttempt(dateTime: DateTime, requestIp: String) { - transaction { - LoginAttempts.insert { - it[timestamp] = dateTime - it[ip] = requestIp - } - } - } - - /** - * Gets all login attempts of [requestIp] - */ - fun getLoginAttempts(requestIp: String): List<Pair<DateTime, String>> { - return transaction { - LoginAttempts.select { LoginAttempts.ip eq requestIp } - .map { it[LoginAttempts.timestamp] to it[LoginAttempts.ip] } - } - } - - /** - * Initializes the database - */ - fun initDatabase() { - val initialUseRow = transaction { General.selectAll().map { it[General.initialUse] } } - if (initialUseRow.isEmpty() || initialUseRow[0]) { - transaction { - RolesData.insert { - it[role] = "ADMIN" - } - RolesData.insert { - it[role] = "USER" - } - RolesData.insert { - it[role] = "GUEST" - } - - UserRoles.insert { - it[userId] = 1 - it[roleId] = 1 - } - - General.insert { - it[initialUse] = false - } - } - } else { - log.warn("Already initialized Database.") - } - } - - /** - * Generates a random string with [length] characters - */ - private fun generateRandomString(length: Int = 64): String { - val allowedChars = "ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz0123456789" - return (1..length) - .map { allowedChars.random() } - .joinToString("") - } -} - -data class ReturnFileData( - val userId: Int, - val fileLocation: String, - val isDirectory: Boolean -) |