Use synchronous=NORMAL in sqlite since we are using WAL

This commit is contained in:
DBotThePony 2024-05-02 17:54:48 +07:00
parent 782fb29dbf
commit b5c9b1f35a
Signed by: DBot
GPG Key ID: DCC23B5715498507
6 changed files with 55 additions and 61 deletions

View File

@ -71,6 +71,7 @@ sealed class StarboundServer(val root: File) : BlockableEventLoop("Server thread
init {
database.createStatement().use {
it.execute("PRAGMA journal_mode=WAL")
it.execute("PRAGMA synchronous=NORMAL")
it.execute("CREATE TABLE IF NOT EXISTS `metadata` (`key` VARCHAR NOT NULL PRIMARY KEY, `value` BLOB NOT NULL)")
it.execute("CREATE TABLE IF NOT EXISTS `universe_flags` (`flag` VARCHAR NOT NULL PRIMARY KEY)")
it.execute("CREATE TABLE IF NOT EXISTS `client_context` (`uuid` VARCHAR NOT NULL PRIMARY KEY, `data` BLOB NOT NULL)")

View File

@ -39,6 +39,7 @@ import java.sql.DriverManager
import java.sql.PreparedStatement
import java.util.concurrent.CompletableFuture
import java.util.concurrent.CopyOnWriteArrayList
import java.util.concurrent.Executor
import java.util.concurrent.TimeUnit
import java.util.concurrent.locks.ReentrantLock
import java.util.function.Function
@ -51,6 +52,8 @@ sealed class LegacyWorldStorage() : WorldStorage() {
protected abstract fun load(at: ByteKey): CompletableFuture<KOptional<ByteArray>>
protected abstract fun write(at: ByteKey, value: ByteArray)
protected abstract val executor: Executor
override fun loadCells(pos: ChunkPos): CompletableFuture<KOptional<Pair<Object2DArray<out AbstractCell>, ChunkState>>> {
val chunkX = pos.x
val chunkY = pos.y
@ -112,62 +115,48 @@ sealed class LegacyWorldStorage() : WorldStorage() {
}
override fun saveEntities(pos: ChunkPos, data: Collection<AbstractEntity>, now: Boolean): Boolean {
if (now) {
executor.execute {
val chunkX = pos.x
val chunkY = pos.y
val key = ByteKey(2, (chunkX shr 8).toByte(), chunkX.toByte(), (chunkY shr 8).toByte(), chunkY.toByte())
write(key, writeEntities(data))
} else {
Starbound.EXECUTOR.execute {
val chunkX = pos.x
val chunkY = pos.y
val key = ByteKey(2, (chunkX shr 8).toByte(), chunkX.toByte(), (chunkY shr 8).toByte(), chunkY.toByte())
write(key, writeEntities(data))
}
}
return true
}
private fun saveCells0(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState) {
val buff = FastByteArrayOutputStream()
val stream = DataOutputStream(BufferedOutputStream(DeflaterOutputStream(buff)))
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState): Boolean {
executor.execute {
val buff = FastByteArrayOutputStream()
val stream = DataOutputStream(BufferedOutputStream(DeflaterOutputStream(buff)))
stream.writeVarInt(when (state) {
ChunkState.FRESH -> 0
ChunkState.EMPTY -> 0
ChunkState.TERRAIN -> 1
ChunkState.MICRO_DUNGEONS -> 2
ChunkState.CAVE_LIQUID -> 3
ChunkState.FULL -> 4
})
stream.writeVarInt(
when (state) {
ChunkState.FRESH -> 0
ChunkState.EMPTY -> 0
ChunkState.TERRAIN -> 1
ChunkState.MICRO_DUNGEONS -> 2
ChunkState.CAVE_LIQUID -> 3
ChunkState.FULL -> 4
}
)
stream.writeVarInt(418)
stream.writeVarInt(418)
for (y in 0 until CHUNK_SIZE) {
for (x in 0 until CHUNK_SIZE) {
val cell = data.getOrNull(x, y) ?: AbstractCell.NULL
cell.writeLegacy(stream)
for (y in 0 until CHUNK_SIZE) {
for (x in 0 until CHUNK_SIZE) {
val cell = data.getOrNull(x, y) ?: AbstractCell.NULL
cell.writeLegacy(stream)
}
}
}
val chunkX = pos.x
val chunkY = pos.y
val key = ByteKey(1, (chunkX shr 8).toByte(), chunkX.toByte(), (chunkY shr 8).toByte(), chunkY.toByte())
val chunkX = pos.x
val chunkY = pos.y
val key = ByteKey(1, (chunkX shr 8).toByte(), chunkX.toByte(), (chunkY shr 8).toByte(), chunkY.toByte())
stream.close()
write(key, buff.array.copyOf(buff.length))
}
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState, now: Boolean): Boolean {
if (now) {
saveCells0(pos, data, state)
} else {
Starbound.EXECUTOR.execute {
saveCells0(pos, data, state)
}
stream.close()
write(key, buff.array.copyOf(buff.length))
}
return true
@ -188,6 +177,8 @@ sealed class LegacyWorldStorage() : WorldStorage() {
}
class Memory(private val get: (ByteKey) -> ByteArray?, private val set: (ByteKey, ByteArray) -> Unit) : LegacyWorldStorage() {
override val executor: Executor = Executor { it.run() }
override fun load(at: ByteKey): CompletableFuture<KOptional<ByteArray>> {
return CompletableFuture.completedFuture(KOptional.ofNullable(get(at)))
}
@ -200,10 +191,10 @@ sealed class LegacyWorldStorage() : WorldStorage() {
}
class DB5(private val database: BTreeDB5) : LegacyWorldStorage() {
private val carrier = CarriedExecutor(Starbound.IO_EXECUTOR)
override val executor = CarriedExecutor(Starbound.IO_EXECUTOR)
override fun load(at: ByteKey): CompletableFuture<KOptional<ByteArray>> {
return CompletableFuture.supplyAsync(Supplier { database.read(at) }, carrier)
return CompletableFuture.supplyAsync(Supplier { database.read(at) }, executor)
}
override fun write(at: ByteKey, value: ByteArray) {
@ -211,13 +202,13 @@ sealed class LegacyWorldStorage() : WorldStorage() {
}
override fun close() {
carrier.execute { database.close() }
carrier.wait(300L, TimeUnit.SECONDS)
executor.execute { database.close() }
executor.wait(300L, TimeUnit.SECONDS)
}
}
class SQL(path: File) : LegacyWorldStorage() {
private val carrier = CarriedExecutor(Starbound.IO_EXECUTOR)
override val executor = CarriedExecutor(Starbound.IO_EXECUTOR)
private val connection = DriverManager.getConnection("jdbc:sqlite:${path.canonicalPath.replace('\\', '/')}")
private val cleaner: Cleaner.Cleanable
@ -231,10 +222,11 @@ sealed class LegacyWorldStorage() : WorldStorage() {
connection.createStatement().use {
it.execute("PRAGMA journal_mode=WAL")
it.execute("PRAGMA synchronous=NORMAL")
it.execute("""CREATE TABLE IF NOT EXISTS `data` (
|`key` BLOB NOT NULL PRIMARY KEY,
|`value` BLOB NOT NULL
it.execute("""CREATE TABLE IF NOT EXISTS "data" (
|"key" BLOB NOT NULL PRIMARY KEY,
|"value" BLOB NOT NULL
|)""".trimMargin())
}
}
@ -256,11 +248,11 @@ sealed class LegacyWorldStorage() : WorldStorage() {
KOptional()
}
}
}, carrier)
}, executor)
}
override fun write(at: ByteKey, value: ByteArray) {
carrier.execute {
executor.execute {
writer.setBytes(1, at.toByteArray())
writer.setBytes(2, value)
writer.execute()
@ -269,8 +261,8 @@ sealed class LegacyWorldStorage() : WorldStorage() {
override fun close() {
// carrier.execute { connection.commit() }
carrier.execute { cleaner.clean() }
carrier.wait(300L, TimeUnit.SECONDS)
executor.execute { cleaner.clean() }
executor.wait(300L, TimeUnit.SECONDS)
}
}

View File

@ -27,8 +27,8 @@ class NativeWorldStorage() : WorldStorage() {
return super.saveEntities(pos, data, now)
}
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState, now: Boolean): Boolean {
return super.saveCells(pos, data, state, now)
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState): Boolean {
return super.saveCells(pos, data, state)
}
override fun saveMetadata(data: Metadata): Boolean {

View File

@ -533,7 +533,7 @@ class ServerChunk(world: ServerWorld, pos: ChunkPos) : Chunk<ServerWorld, Server
}
}
private fun writeToStorage(now: Boolean = false): Collection<AbstractEntity> {
private fun writeToStorage(): Collection<AbstractEntity> {
if (!cells.isInitialized() || state <= ChunkState.EMPTY)
return emptyList()
@ -542,8 +542,8 @@ class ServerChunk(world: ServerWorld, pos: ChunkPos) : Chunk<ServerWorld, Server
aabbd,
filter = Predicate { !it.isRemote && aabbd.isInside(it.position) })
world.storage.saveCells(pos, copyCells(), state, now)
world.storage.saveEntities(pos, unloadable.filter { it.isPersistent }, now)
world.storage.saveCells(pos, copyCells(), state)
world.storage.saveEntities(pos, unloadable.filter { it.isPersistent })
return unloadable
}
@ -567,7 +567,7 @@ class ServerChunk(world: ServerWorld, pos: ChunkPos) : Chunk<ServerWorld, Server
world.chunkMap.remove(pos)
flushTask.cancel(false)
writeToStorage(true).forEach {
writeToStorage().forEach {
it.remove(AbstractEntity.RemovalReason.UNLOADED)
}
}

View File

@ -97,6 +97,7 @@ class ServerUniverse(folder: File? = null) : Universe(), Closeable {
database.createStatement().use {
it.execute("PRAGMA journal_mode=WAL")
it.execute("PRAGMA synchronous=NORMAL")
it.execute("""
CREATE TABLE IF NOT EXISTS `chunk` (

View File

@ -83,7 +83,7 @@ abstract class WorldStorage : Closeable {
return false
}
open fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState, now: Boolean = false): Boolean {
open fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState): Boolean {
return false
}
@ -152,8 +152,8 @@ abstract class WorldStorage : Closeable {
return children.any { it.saveEntities(pos, data, now) }
}
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState, now: Boolean): Boolean {
return children.any { it.saveCells(pos, data, state, now) }
override fun saveCells(pos: ChunkPos, data: Object2DArray<out AbstractCell>, state: ChunkState): Boolean {
return children.any { it.saveCells(pos, data, state) }
}
override fun saveMetadata(data: Metadata): Boolean {