Fix race condition in event loop shutdown when using awaitTermination()
This commit is contained in:
parent
195de2d160
commit
9797202af2
@ -298,7 +298,6 @@ class ServerConnection(val server: StarboundServer, type: ConnectionType) : Conn
|
||||
run {
|
||||
val action = ship.location.orbitalAction(world)
|
||||
currentOrbitalWarpAction = action
|
||||
orbitalWarpAction = action
|
||||
|
||||
for (client in shipWorld.clients) {
|
||||
client.client.orbitalWarpAction = action
|
||||
|
@ -7,6 +7,7 @@ import com.google.gson.JsonObject
|
||||
import com.google.gson.JsonPrimitive
|
||||
import it.unimi.dsi.fastutil.io.FastByteArrayInputStream
|
||||
import it.unimi.dsi.fastutil.io.FastByteArrayOutputStream
|
||||
import it.unimi.dsi.fastutil.objects.ObjectArrayList
|
||||
import it.unimi.dsi.fastutil.objects.ObjectArraySet
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.SupervisorJob
|
||||
@ -164,24 +165,45 @@ sealed class StarboundServer(val root: File) : BlockableEventLoop("Server thread
|
||||
}
|
||||
|
||||
private suspend fun loadCelestialWorld(location: WorldID.Celestial): ServerWorld {
|
||||
val file = File(universeFolder, location.pos.toString().replace(':', '_') + ".db")
|
||||
val fileName = location.pos.toString().replace(':', '_') + ".db"
|
||||
val file = File(universeFolder, fileName)
|
||||
val firstTime = !file.exists()
|
||||
val storage = LegacyWorldStorage.SQL(file)
|
||||
|
||||
val world = if (firstTime) {
|
||||
LOGGER.info("Creating celestial world $location")
|
||||
ServerWorld.create(this, WorldTemplate.create(location.pos, universe), storage, location)
|
||||
} else {
|
||||
LOGGER.info("Loading celestial world $location")
|
||||
ServerWorld.load(this, storage, location).await()
|
||||
val world = try {
|
||||
if (firstTime) {
|
||||
LOGGER.info("Creating celestial world $location")
|
||||
ServerWorld.create(this, WorldTemplate.create(location.pos, universe), storage, location)
|
||||
} else {
|
||||
LOGGER.info("Loading celestial world $location")
|
||||
ServerWorld.load(this, storage, location).await()
|
||||
}
|
||||
} catch (err: Throwable) {
|
||||
storage.close()
|
||||
|
||||
if (firstTime) {
|
||||
file.delete()
|
||||
throw err
|
||||
} else {
|
||||
LOGGER.fatal("Exception loading celestial world at $location, recreating!")
|
||||
|
||||
var i = 0
|
||||
while (!file.renameTo(File(universeFolder, "$fileName-fail$i")) && ++i < 1000) {}
|
||||
|
||||
ServerWorld.create(this, WorldTemplate.create(location.pos, universe), LegacyWorldStorage.SQL(file), location)
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
world.sky.referenceClock = universeClock
|
||||
world.eventLoop.start()
|
||||
world.prepare(firstTime).await()
|
||||
|
||||
if (firstTime) {
|
||||
world.saveMetadata()
|
||||
}
|
||||
} catch (err: Throwable) {
|
||||
LOGGER.fatal("Exception while creating celestial world at $location!", err)
|
||||
LOGGER.fatal("Exception while initializing celestial world at $location!", err)
|
||||
world.eventLoop.shutdown()
|
||||
throw err
|
||||
}
|
||||
@ -390,23 +412,21 @@ sealed class StarboundServer(val root: File) : BlockableEventLoop("Server thread
|
||||
scope.cancel("Server shutting down")
|
||||
channels.close()
|
||||
|
||||
worlds.values.forEach {
|
||||
if (it.isDone && !it.isCompletedExceptionally) {
|
||||
it.get().eventLoop.shutdown()
|
||||
}
|
||||
val worldSlice = ObjectArrayList(worlds.values)
|
||||
|
||||
worldSlice.forEach {
|
||||
it.thenAccept { it.eventLoop.shutdown() }
|
||||
}
|
||||
|
||||
worlds.values.forEach {
|
||||
if (it.isDone && !it.isCompletedExceptionally) {
|
||||
it.get().eventLoop.awaitTermination(60L, TimeUnit.SECONDS)
|
||||
worldSlice.forEach {
|
||||
it.thenAccept {
|
||||
it.eventLoop.awaitTermination(60L, TimeUnit.SECONDS)
|
||||
|
||||
if (!it.get().eventLoop.isTerminated) {
|
||||
LOGGER.warn("World ${it.get()} did not shutdown in 60 seconds, forcing termination. This might leave world in inconsistent state!")
|
||||
it.get().eventLoop.shutdownNow()
|
||||
if (!it.eventLoop.isTerminated) {
|
||||
LOGGER.warn("World $it did not shutdown in 60 seconds, forcing termination. This might leave world in inconsistent state!")
|
||||
it.eventLoop.shutdownNow()
|
||||
}
|
||||
}
|
||||
|
||||
it.cancel(true)
|
||||
}
|
||||
|
||||
database.commit()
|
||||
|
@ -197,8 +197,16 @@ sealed class LegacyWorldStorage() : WorldStorage() {
|
||||
class SQL(path: File) : LegacyWorldStorage() {
|
||||
private val carrier = CarriedExecutor(Starbound.IO_EXECUTOR)
|
||||
private val connection = DriverManager.getConnection("jdbc:sqlite:${path.canonicalPath.replace('\\', '/')}")
|
||||
private val cleaner: Cleaner.Cleanable
|
||||
|
||||
init {
|
||||
val connection = connection
|
||||
|
||||
cleaner = Starbound.CLEANER.register(this) {
|
||||
/*connection.commit();*/
|
||||
connection.close()
|
||||
}
|
||||
|
||||
connection.autoCommit = false
|
||||
|
||||
connection.createStatement().use {
|
||||
@ -243,7 +251,7 @@ sealed class LegacyWorldStorage() : WorldStorage() {
|
||||
|
||||
override fun close() {
|
||||
carrier.execute { connection.commit() }
|
||||
carrier.execute { connection.close() }
|
||||
carrier.execute { cleaner.clean() }
|
||||
carrier.wait(300L, TimeUnit.SECONDS)
|
||||
}
|
||||
}
|
||||
|
@ -120,6 +120,21 @@ class ServerWorld private constructor(
|
||||
}
|
||||
}
|
||||
|
||||
fun saveMetadata() {
|
||||
val metadata = MetadataJson(
|
||||
playerStart = playerSpawnPosition,
|
||||
respawnInWorld = respawnInWorld,
|
||||
adjustPlayerStart = adjustPlayerSpawn,
|
||||
worldTemplate = if (storage is LegacyWorldStorage) Starbound.legacyJson { template.toJson() } else template.toJson(),
|
||||
centralStructure = centralStructure,
|
||||
protectedDungeonIds = protectedDungeonIDs,
|
||||
worldProperties = copyProperties(),
|
||||
spawningEnabled = true
|
||||
)
|
||||
|
||||
storage.saveMetadata(WorldStorage.Metadata(geometry, VersionRegistry.make("WorldMetadata", Starbound.gson.toJsonTree(metadata))))
|
||||
}
|
||||
|
||||
override val eventLoop = object : BlockableEventLoop("Server World $worldID") {
|
||||
init {
|
||||
isDaemon = true
|
||||
@ -148,18 +163,7 @@ class ServerWorld private constructor(
|
||||
it.client.enqueueWarp(WarpAlias.Return)
|
||||
}
|
||||
|
||||
val metadata = MetadataJson(
|
||||
playerStart = playerSpawnPosition,
|
||||
respawnInWorld = respawnInWorld,
|
||||
adjustPlayerStart = adjustPlayerSpawn,
|
||||
worldTemplate = if (storage is LegacyWorldStorage) Starbound.legacyJson { template.toJson() } else template.toJson(),
|
||||
centralStructure = centralStructure,
|
||||
protectedDungeonIds = protectedDungeonIDs,
|
||||
worldProperties = copyProperties(),
|
||||
spawningEnabled = true
|
||||
)
|
||||
|
||||
storage.saveMetadata(WorldStorage.Metadata(geometry, VersionRegistry.make("WorldMetadata", Starbound.gson.toJsonTree(metadata))))
|
||||
saveMetadata()
|
||||
storage.close()
|
||||
}
|
||||
}
|
||||
@ -644,9 +648,11 @@ class ServerWorld private constructor(
|
||||
world.protectedDungeonIDs.addAll(meta.protectedDungeonIds)
|
||||
world
|
||||
}
|
||||
}.exceptionally {
|
||||
LOGGER.error("Error while instancing world $worldID", it)
|
||||
null
|
||||
}.also {
|
||||
it.exceptionally {
|
||||
LOGGER.error("Error while instancing world $worldID", it)
|
||||
null
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -160,20 +160,27 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
|
||||
if (isShutdown && isRunning) {
|
||||
while (eventLoopIteration()) {}
|
||||
isRunning = false
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
|
||||
try {
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
} catch (err: Throwable) {
|
||||
LOGGER.fatal("Exception shutting down $name")
|
||||
return
|
||||
} finally {
|
||||
isRunning = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
LOGGER.info("Thread ${this.name} stopped gracefully")
|
||||
LOGGER.info("Thread $name stopped gracefully")
|
||||
}
|
||||
|
||||
final override fun execute(command: Runnable) {
|
||||
if (currentThread() === this) {
|
||||
command.run()
|
||||
} else {
|
||||
if (!isRunning)
|
||||
if (isShutdown)
|
||||
throw RejectedExecutionException("EventLoop is shutting down")
|
||||
|
||||
val future = CompletableFuture<Unit>()
|
||||
@ -196,7 +203,7 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
return CompletableFuture.failedFuture(err)
|
||||
}
|
||||
} else {
|
||||
if (!isRunning)
|
||||
if (isShutdown)
|
||||
throw RejectedExecutionException("EventLoop is shutting down")
|
||||
|
||||
val future = CompletableFuture<T>()
|
||||
@ -235,7 +242,7 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
return CompletableFuture.failedFuture<Unit>(err)
|
||||
}
|
||||
} else {
|
||||
if (!isRunning)
|
||||
if (isShutdown)
|
||||
throw RejectedExecutionException("EventLoop is shutting down")
|
||||
|
||||
val future = CompletableFuture<Unit>()
|
||||
@ -260,7 +267,7 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
return CompletableFuture.failedFuture(err)
|
||||
}
|
||||
} else {
|
||||
if (!isRunning)
|
||||
if (isShutdown)
|
||||
throw RejectedExecutionException("EventLoop is shutting down")
|
||||
|
||||
val future = CompletableFuture<T>()
|
||||
@ -297,6 +304,8 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
|
||||
final override fun shutdown() {
|
||||
if (!isShutdown) {
|
||||
LOGGER.info("$name shutdown initiated")
|
||||
|
||||
isShutdown = true
|
||||
|
||||
if (currentThread() === this || state == State.NEW) {
|
||||
@ -312,9 +321,14 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
}
|
||||
}
|
||||
|
||||
isRunning = false
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
try {
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
} catch (err: Throwable) {
|
||||
LOGGER.fatal("Exception shutting down $name", err)
|
||||
} finally {
|
||||
isRunning = false
|
||||
}
|
||||
} else {
|
||||
// wake up thread
|
||||
LockSupport.unpark(this)
|
||||
@ -351,9 +365,14 @@ open class BlockableEventLoop(name: String) : Thread(name), ScheduledExecutorSer
|
||||
}
|
||||
}
|
||||
|
||||
isRunning = false
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
try {
|
||||
scope.cancel(CancellationException("EventLoop shut down"))
|
||||
performShutdown()
|
||||
} catch (err: Throwable) {
|
||||
LOGGER.fatal("Exception shutting down $name")
|
||||
} finally {
|
||||
isRunning = false
|
||||
}
|
||||
} else {
|
||||
// wake up thread
|
||||
LockSupport.unpark(this)
|
||||
|
Loading…
Reference in New Issue
Block a user