1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
|
package space.anity
import at.favre.lib.crypto.bcrypt.*
import org.jetbrains.exposed.sql.*
import org.jetbrains.exposed.sql.transactions.*
import java.sql.*
import java.util.logging.*
class DatabaseController(dbFileLocation: String = "main.db") {
val db: Database = Database.connect("jdbc:sqlite:$dbFileLocation", "org.sqlite.JDBC")
private val log = Logger.getLogger(this.javaClass.name)
/**
* Database table indexing the file locations
*/
object FileLocation : Table() {
val id = integer("id").autoIncrement().primaryKey()
val location = text("location").uniqueIndex()
val username = varchar("username", 24)
}
/**
* 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)
}
/**
* 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 declaring available roles
*/
object RolesData : Table() {
val id = integer("id").autoIncrement().primaryKey()
val role = varchar("roles", 16)
}
/**
* Database table storing general data/states
*/
object General : Table() {
val id = integer("id").autoIncrement().primaryKey()
val initialUse = integer("initialUse").default(1).primaryKey()
}
init {
// Create connection
TransactionManager.manager.defaultIsolationLevel = Connection.TRANSACTION_SERIALIZABLE
// Add tables
transaction {
SchemaUtils.createMissingTablesAndColumns(FileLocation, UserData, UserRoles, RolesData, General)
}
}
/**
* Creates the user in the database using username, password and the role
*/
fun createUser(usernameString: String, passwordString: String, roleString: String) {
transaction {
try {
val usersId = UserData.insert {
it[username] = usernameString
it[password] = BCrypt.withDefaults().hashToString(12, passwordString.toCharArray())
}[UserData.id]
UserRoles.insert { roles ->
roles[userId] = usersId!!
roles[roleId] = RolesData.select { RolesData.role eq roleString }.map { it[RolesData.id] }[0]
}
} catch (_: org.jetbrains.exposed.exceptions.ExposedSQLException) {
log.warning("User already exists!")
}
}
}
/**
* Tests whether the password [passwordString] of the user [usernameString] is correct
*/
fun checkUser(usernameString: String, passwordString: String): Boolean {
return transaction {
val passwordHash = UserData.select { UserData.username eq usernameString }.map { it[UserData.password] }[0]
BCrypt.verifyer().verify(passwordString.toCharArray(), passwordHash).verified
}
}
/**
* Returns the corresponding role using [usernameString]
*/
fun getRole(usernameString: String): Roles {
return transaction {
val userId = UserData.select { UserData.username eq usernameString }.map { it[UserData.id] }[0]
val userRoleId = UserRoles.select { UserRoles.userId eq userId }.map { it[UserRoles.roleId] }[0]
val userRole = RolesData.select { RolesData.id eq userRoleId }.map { it[RolesData.role] }[0]
if (userRole == "ADMIN") Roles.ADMIN else Roles.USER
}
}
/**
* Adds the uploaded file to the database
*/
fun addFile(fileLocation: String, usernameString: String) {
transaction {
try {
FileLocation.insert {
it[location] = fileLocation
it[username] = usernameString
}
} catch (_: org.jetbrains.exposed.exceptions.ExposedSQLException) {
log.warning("File already exists!")
}
}
}
/**
* Initializes the database
*/
fun initDatabase() {
val initialUseRow = transaction { General.selectAll().map { it[General.initialUse] } }
if (initialUseRow.isEmpty() || initialUseRow[0] == 1) {
transaction {
RolesData.insert {
it[role] = "ADMIN"
}
RolesData.insert {
it[role] = "USER"
}
RolesData.insert {
it[role] = "GUEST"
}
databaseController.createUser("melvin", "supersecure", "ADMIN")
UserRoles.insert {
it[userId] = 1
it[roleId] = 1
}
General.insert {
it[initialUse] = 0
}
}
} else {
log.info("Already initialized Database.")
}
}
}
|