Add backup
This commit is contained in:
parent
94c35b9067
commit
07a619f826
6 changed files with 76 additions and 9 deletions
|
@ -85,6 +85,7 @@
|
||||||
.board-running {
|
.board-running {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
align-content: flex-start;
|
||||||
|
|
||||||
&:empty + .board-running-empty {
|
&:empty + .board-running-empty {
|
||||||
display: block;
|
display: block;
|
||||||
|
|
|
@ -3,6 +3,7 @@ package de.kif.backend
|
||||||
import com.fasterxml.jackson.databind.SerializationFeature
|
import com.fasterxml.jackson.databind.SerializationFeature
|
||||||
import de.kif.backend.route.*
|
import de.kif.backend.route.*
|
||||||
import de.kif.backend.route.api.*
|
import de.kif.backend.route.api.*
|
||||||
|
import de.kif.backend.util.Backup
|
||||||
import de.kif.backend.util.pushService
|
import de.kif.backend.util.pushService
|
||||||
import io.ktor.application.Application
|
import io.ktor.application.Application
|
||||||
import io.ktor.application.call
|
import io.ktor.application.call
|
||||||
|
@ -101,4 +102,6 @@ fun Application.main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info { "Responding at http://${Configuration.Server.host}:${Configuration.Server.port}$prefix/" }
|
logger.info { "Responding at http://${Configuration.Server.host}:${Configuration.Server.port}$prefix/" }
|
||||||
|
|
||||||
|
Backup.startBackupService()
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,6 +43,7 @@ object Configuration {
|
||||||
val uploads by required<String>()
|
val uploads by required<String>()
|
||||||
val database by required<String>()
|
val database by required<String>()
|
||||||
val announcement by required<String>()
|
val announcement by required<String>()
|
||||||
|
val backup by required<String>()
|
||||||
}
|
}
|
||||||
|
|
||||||
object Path {
|
object Path {
|
||||||
|
@ -60,6 +61,9 @@ object Configuration {
|
||||||
|
|
||||||
val announcement by c(PathSpec.announcement)
|
val announcement by c(PathSpec.announcement)
|
||||||
val announcementPath: java.nio.file.Path by lazy { Paths.get(announcement).toAbsolutePath() }
|
val announcementPath: java.nio.file.Path by lazy { Paths.get(announcement).toAbsolutePath() }
|
||||||
|
|
||||||
|
val backup by c(PathSpec.backup)
|
||||||
|
val backupPath: java.nio.file.Path by lazy { Paths.get(backup).toAbsolutePath() }
|
||||||
}
|
}
|
||||||
|
|
||||||
private object ScheduleSpec : ConfigSpec("schedule") {
|
private object ScheduleSpec : ConfigSpec("schedule") {
|
||||||
|
@ -93,6 +97,7 @@ object Configuration {
|
||||||
private object GeneralSpec : ConfigSpec("general") {
|
private object GeneralSpec : ConfigSpec("general") {
|
||||||
val allowedUploadExtensions by required<String>("allowed_upload_extensions")
|
val allowedUploadExtensions by required<String>("allowed_upload_extensions")
|
||||||
val wikiUrl by required<String>("wiki_url")
|
val wikiUrl by required<String>("wiki_url")
|
||||||
|
val backupInterval by required<Long>("backup_interval")
|
||||||
}
|
}
|
||||||
|
|
||||||
object General {
|
object General {
|
||||||
|
@ -101,6 +106,7 @@ object Configuration {
|
||||||
allowedUploadExtensions.split(",").map { it.trim().toLowerCase() }.toSet()
|
allowedUploadExtensions.split(",").map { it.trim().toLowerCase() }.toSet()
|
||||||
}
|
}
|
||||||
val wikiUrl by c(GeneralSpec.wikiUrl)
|
val wikiUrl by c(GeneralSpec.wikiUrl)
|
||||||
|
val backupInterval by c(GeneralSpec.backupInterval)
|
||||||
}
|
}
|
||||||
|
|
||||||
private object TwitterSpec : ConfigSpec("twitter") {
|
private object TwitterSpec : ConfigSpec("twitter") {
|
||||||
|
|
|
@ -83,11 +83,19 @@ object Resources {
|
||||||
Files.createDirectories(Configuration.Path.sessionsPath)
|
Files.createDirectories(Configuration.Path.sessionsPath)
|
||||||
Files.createDirectories(Configuration.Path.uploadsPath)
|
Files.createDirectories(Configuration.Path.uploadsPath)
|
||||||
Files.createDirectories(Configuration.Path.webPath)
|
Files.createDirectories(Configuration.Path.webPath)
|
||||||
|
Files.createDirectories(Configuration.Path.backupPath)
|
||||||
|
|
||||||
|
if (!Files.exists(Configuration.Path.announcementPath)) {
|
||||||
|
Files.createFile(Configuration.Path.announcementPath)
|
||||||
|
}
|
||||||
|
|
||||||
logger.info { "Database path: ${Configuration.Path.databasePath}" }
|
logger.info { "Database path: ${Configuration.Path.databasePath}" }
|
||||||
logger.info { "Sessions path: ${Configuration.Path.sessionsPath}" }
|
logger.info { "Sessions path: ${Configuration.Path.sessionsPath}" }
|
||||||
logger.info { "Uploads path: ${Configuration.Path.uploadsPath}" }
|
logger.info { "Uploads path: ${Configuration.Path.uploadsPath}" }
|
||||||
logger.info { "Web path: ${Configuration.Path.webPath}" }
|
logger.info { "Web path: ${Configuration.Path.webPath}" }
|
||||||
|
logger.info { "Backup path: ${Configuration.Path.backupPath}" }
|
||||||
|
|
||||||
|
logger.info { "Announcement file: ${Configuration.Path.announcementPath}" }
|
||||||
|
|
||||||
logger.info { "Extract web content..." }
|
logger.info { "Extract web content..." }
|
||||||
extractWeb()
|
extractWeb()
|
||||||
|
|
|
@ -1,11 +1,18 @@
|
||||||
package de.kif.backend.util
|
package de.kif.backend.util
|
||||||
|
|
||||||
|
import de.kif.backend.Configuration
|
||||||
import de.kif.backend.database.Connection
|
import de.kif.backend.database.Connection
|
||||||
import de.kif.backend.repository.*
|
import de.kif.backend.repository.*
|
||||||
import de.kif.common.RepositoryType
|
import de.kif.common.RepositoryType
|
||||||
import de.kif.common.Serialization
|
import de.kif.common.Serialization
|
||||||
import de.kif.common.model.*
|
import de.kif.common.model.*
|
||||||
|
import kotlinx.coroutines.runBlocking
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
|
import mu.KotlinLogging
|
||||||
|
import java.lang.Exception
|
||||||
|
import java.text.SimpleDateFormat
|
||||||
|
import java.util.*
|
||||||
|
import kotlin.concurrent.thread
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class Backup(
|
data class Backup(
|
||||||
|
@ -17,6 +24,8 @@ data class Backup(
|
||||||
val workGroups: List<WorkGroup> = emptyList()
|
val workGroups: List<WorkGroup> = emptyList()
|
||||||
) {
|
) {
|
||||||
companion object {
|
companion object {
|
||||||
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
suspend fun backup(vararg repositories: RepositoryType): String {
|
suspend fun backup(vararg repositories: RepositoryType): String {
|
||||||
var backup = Backup()
|
var backup = Backup()
|
||||||
|
|
||||||
|
@ -78,7 +87,9 @@ data class Backup(
|
||||||
var workGroup = it
|
var workGroup = it
|
||||||
val track = workGroup.track
|
val track = workGroup.track
|
||||||
if (track != null) {
|
if (track != null) {
|
||||||
workGroup = workGroup.copy(track = track.copy(id = trackMap.firstOrNull { (i,_) -> i.equalsIgnoreId(track) }?.second ?: run {
|
workGroup = workGroup.copy(track = track.copy(id = trackMap.firstOrNull { (i, _) ->
|
||||||
|
i.equalsIgnoreId(track)
|
||||||
|
}?.second ?: run {
|
||||||
println("Cannot import work group, due to missing track")
|
println("Cannot import work group, due to missing track")
|
||||||
return@mapNotNull null
|
return@mapNotNull null
|
||||||
}))
|
}))
|
||||||
|
@ -94,14 +105,16 @@ data class Backup(
|
||||||
newSchedules.forEach {
|
newSchedules.forEach {
|
||||||
ScheduleRepository.create(
|
ScheduleRepository.create(
|
||||||
it.copy(
|
it.copy(
|
||||||
room = it.room.copy(id = roomMap.firstOrNull { (i,_) -> i.equalsIgnoreId(it.room) }?.second ?: run {
|
room = it.room.copy(
|
||||||
println("Cannot import schedule, due to missing room")
|
id = roomMap.firstOrNull { (i, _) -> i.equalsIgnoreId(it.room) }?.second ?: run {
|
||||||
return@forEach
|
println("Cannot import schedule, due to missing room")
|
||||||
}),
|
return@forEach
|
||||||
workGroup = it.workGroup.copy(id = workGroupMap.firstOrNull { (i,_) -> i.equalsIgnoreId(it.workGroup) }?.second ?: run {
|
}),
|
||||||
println("Cannot import schedule, due to missing work group")
|
workGroup = it.workGroup.copy(
|
||||||
return@forEach
|
id = workGroupMap.firstOrNull { (i, _) -> i.equalsIgnoreId(it.workGroup) }?.second ?: run {
|
||||||
})
|
println("Cannot import schedule, due to missing work group")
|
||||||
|
return@forEach
|
||||||
|
})
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -114,5 +127,39 @@ data class Backup(
|
||||||
|
|
||||||
import(data)
|
import(data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun startBackupService() {
|
||||||
|
val backupPath = Configuration.Path.backupPath.toFile()
|
||||||
|
val backupInterval = Configuration.General.backupInterval
|
||||||
|
|
||||||
|
val formatter = SimpleDateFormat("yyyyMMdd'T'HHmmss")
|
||||||
|
|
||||||
|
thread(
|
||||||
|
start = true,
|
||||||
|
isDaemon = true,
|
||||||
|
name = "backup-service"
|
||||||
|
) {
|
||||||
|
while (true) {
|
||||||
|
try {
|
||||||
|
Thread.sleep(backupInterval)
|
||||||
|
|
||||||
|
if (!backupPath.exists()) {
|
||||||
|
backupPath.mkdirs()
|
||||||
|
}
|
||||||
|
|
||||||
|
val date = formatter.format(Date())
|
||||||
|
|
||||||
|
val backupFile = backupPath.resolve("backup_$date.json")
|
||||||
|
|
||||||
|
val backup = runBlocking {
|
||||||
|
backup(*RepositoryType.values())
|
||||||
|
}
|
||||||
|
backupFile.writeText(backup)
|
||||||
|
} catch (e: Exception) {
|
||||||
|
logger.error(e) { "Cannot create backup" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ sessions = "data/sessions"
|
||||||
uploads = "data/uploads"
|
uploads = "data/uploads"
|
||||||
database = "data/portal.db"
|
database = "data/portal.db"
|
||||||
announcement = "data/announcement.txt"
|
announcement = "data/announcement.txt"
|
||||||
|
backup = "data/backup"
|
||||||
|
|
||||||
[schedule]
|
[schedule]
|
||||||
reference = "1970-01-01"
|
reference = "1970-01-01"
|
||||||
|
@ -27,6 +28,7 @@ sign_key = "d1 20 23 8c 01 f8 f0 0d 9d 7c ff 68 21 97 75 31 38 3f fb 91 20 3a 8d
|
||||||
[general]
|
[general]
|
||||||
allowed_upload_extensions = "png, jpg, jpeg"
|
allowed_upload_extensions = "png, jpg, jpeg"
|
||||||
wiki_url = ""
|
wiki_url = ""
|
||||||
|
backup_interval = 3600000
|
||||||
|
|
||||||
[twitter]
|
[twitter]
|
||||||
timeline = ""
|
timeline = ""
|
||||||
|
|
Loading…
Add table
Reference in a new issue