mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Added AccountStore.
This commit is contained in:
parent
010be59701
commit
4db38f3457
2
buildSrc/gradle.properties
Normal file
2
buildSrc/gradle.properties
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
kotlin.code.style=official
|
||||||
|
kotlin.mpp.stability.nowarn=true
|
@ -13,7 +13,9 @@ val log4jVersion: String by project.extra
|
|||||||
tasks.withType<KotlinCompile>().configureEach {
|
tasks.withType<KotlinCompile>().configureEach {
|
||||||
kotlinOptions {
|
kotlinOptions {
|
||||||
jvmTarget = "11"
|
jvmTarget = "11"
|
||||||
freeCompilerArgs = freeCompilerArgs + EXPERIMENTAL_ANNOTATION_COMPILER_ARGS
|
freeCompilerArgs = freeCompilerArgs +
|
||||||
|
EXPERIMENTAL_ANNOTATION_COMPILER_ARGS +
|
||||||
|
"-Xjvm-default=all"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,6 +5,7 @@ import kotlinx.serialization.ExperimentalSerializationApi
|
|||||||
import kotlinx.serialization.hocon.Hocon
|
import kotlinx.serialization.hocon.Hocon
|
||||||
import kotlinx.serialization.hocon.decodeFromConfig
|
import kotlinx.serialization.hocon.decodeFromConfig
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
|
import world.phantasmal.psoserv.data.AccountStore
|
||||||
import world.phantasmal.psoserv.encryption.BbCipher
|
import world.phantasmal.psoserv.encryption.BbCipher
|
||||||
import world.phantasmal.psoserv.encryption.Cipher
|
import world.phantasmal.psoserv.encryption.Cipher
|
||||||
import world.phantasmal.psoserv.encryption.PcCipher
|
import world.phantasmal.psoserv.encryption.PcCipher
|
||||||
@ -71,7 +72,8 @@ fun main(args: Array<String>) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Initialize and start the server.
|
// Initialize and start the server.
|
||||||
val servers = initialize(config)
|
val accountStore = AccountStore(LOGGER)
|
||||||
|
val servers = initialize(config, accountStore)
|
||||||
|
|
||||||
if (servers.isEmpty()) {
|
if (servers.isEmpty()) {
|
||||||
LOGGER.info { "No servers configured, stopping." }
|
LOGGER.info { "No servers configured, stopping." }
|
||||||
@ -85,7 +87,7 @@ fun main(args: Array<String>) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun initialize(config: Config): List<Server> {
|
private fun initialize(config: Config, accountStore: AccountStore): List<Server> {
|
||||||
val address = config.address?.let(::inet4Address) ?: DEFAULT_ADDRESS
|
val address = config.address?.let(::inet4Address) ?: DEFAULT_ADDRESS
|
||||||
|
|
||||||
LOGGER.info { "Binding to $address." }
|
LOGGER.info { "Binding to $address." }
|
||||||
@ -96,18 +98,24 @@ private fun initialize(config: Config): List<Server> {
|
|||||||
val blocks: Map<String, BlockInfo> = run {
|
val blocks: Map<String, BlockInfo> = run {
|
||||||
var blockI = 1
|
var blockI = 1
|
||||||
var blockPort = DEFAULT_FIRST_SHIP_PORT + config.ships.size
|
var blockPort = DEFAULT_FIRST_SHIP_PORT + config.ships.size
|
||||||
|
val blocks = mutableMapOf<String, BlockInfo>()
|
||||||
|
|
||||||
config.blocks.associate { blockCfg ->
|
for (blockCfg in config.blocks) {
|
||||||
val block = BlockInfo(
|
val block = BlockInfo(
|
||||||
name = validateName("Block", blockCfg.name) ?: "block_$blockI",
|
name = validateName("Block", blockCfg.name) ?: "block_$blockI",
|
||||||
uiName = blockCfg.uiName ?: "BLOCK${blockI.toString(2).padStart(2, '0')}",
|
uiName = blockCfg.uiName ?: "BLOCK${blockI.toString(2).padStart(2, '0')}",
|
||||||
bindPair = Inet4Pair(address, blockCfg.port ?: blockPort++),
|
bindPair = Inet4Pair(address, blockCfg.port ?: blockPort++),
|
||||||
)
|
)
|
||||||
blockI++
|
blockI++
|
||||||
Pair(block.name, block)
|
|
||||||
|
require(blocks.put(block.name, block) == null) {
|
||||||
|
"""Duplicate block with name ${block.name}."""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
blocks
|
||||||
|
}
|
||||||
|
|
||||||
val ships: List<ShipInfo> = run {
|
val ships: List<ShipInfo> = run {
|
||||||
var shipI = 1
|
var shipI = 1
|
||||||
var shipPort = DEFAULT_FIRST_SHIP_PORT
|
var shipPort = DEFAULT_FIRST_SHIP_PORT
|
||||||
@ -118,7 +126,7 @@ private fun initialize(config: Config): List<Server> {
|
|||||||
uiName = shipCfg.uiName ?: "Ship $shipI",
|
uiName = shipCfg.uiName ?: "Ship $shipI",
|
||||||
bindPair = Inet4Pair(address, shipCfg.port ?: shipPort++),
|
bindPair = Inet4Pair(address, shipCfg.port ?: shipPort++),
|
||||||
blocks = shipCfg.blocks.map {
|
blocks = shipCfg.blocks.map {
|
||||||
blocks[it] ?: error("""No block with name "$it".""")
|
blocks[it] ?: error("""No block with name $it.""")
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
shipI++
|
shipI++
|
||||||
@ -164,12 +172,13 @@ private fun initialize(config: Config): List<Server> {
|
|||||||
LOGGER.info { "Configuring account server to bind to port ${bindPair.port}." }
|
LOGGER.info { "Configuring account server to bind to port ${bindPair.port}." }
|
||||||
LOGGER.info {
|
LOGGER.info {
|
||||||
"Account server will redirect to ${ships.size} ship servers: ${
|
"Account server will redirect to ${ships.size} ship servers: ${
|
||||||
ships.joinToString { """"${it.name}" (port ${it.bindPair.port})""" }
|
ships.joinToString { """${it.name} (port ${it.bindPair.port})""" }
|
||||||
}."
|
}."
|
||||||
}
|
}
|
||||||
|
|
||||||
servers.add(
|
servers.add(
|
||||||
AccountServer(
|
AccountServer(
|
||||||
|
accountStore,
|
||||||
bindPair,
|
bindPair,
|
||||||
ships,
|
ships,
|
||||||
)
|
)
|
||||||
@ -198,9 +207,10 @@ private fun initialize(config: Config): List<Server> {
|
|||||||
|
|
||||||
servers.add(
|
servers.add(
|
||||||
BlockServer(
|
BlockServer(
|
||||||
|
accountStore,
|
||||||
block.name,
|
block.name,
|
||||||
block.bindPair,
|
block.bindPair,
|
||||||
blockNo = index + 1,
|
blockId = index + 1,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,47 @@
|
|||||||
|
package world.phantasmal.psoserv.data
|
||||||
|
|
||||||
|
class Account(
|
||||||
|
val id: Long,
|
||||||
|
val username: String,
|
||||||
|
val guildCardNo: Int,
|
||||||
|
val teamId: Int,
|
||||||
|
val characters: List<Character>,
|
||||||
|
) {
|
||||||
|
init {
|
||||||
|
require(username.length <= 16)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class PlayingAccount(
|
||||||
|
val account: Account,
|
||||||
|
val char: Character,
|
||||||
|
val blockId: Int,
|
||||||
|
)
|
||||||
|
|
||||||
|
class Character(
|
||||||
|
val id: Long,
|
||||||
|
val accountId: Long,
|
||||||
|
val name: String,
|
||||||
|
val sectionId: SectionId,
|
||||||
|
val exp: Int,
|
||||||
|
val level: Int,
|
||||||
|
) {
|
||||||
|
init {
|
||||||
|
require(name.length <= 16)
|
||||||
|
require(exp >= 0)
|
||||||
|
require(level in 1..200)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum class SectionId {
|
||||||
|
Viridia,
|
||||||
|
Greenill,
|
||||||
|
Skyly,
|
||||||
|
Bluefull,
|
||||||
|
Purplenum,
|
||||||
|
Pinkal,
|
||||||
|
Redria,
|
||||||
|
Oran,
|
||||||
|
Yellowboze,
|
||||||
|
Whitill,
|
||||||
|
}
|
@ -0,0 +1,111 @@
|
|||||||
|
package world.phantasmal.psoserv.data
|
||||||
|
|
||||||
|
import mu.KLogger
|
||||||
|
|
||||||
|
class AccountStore(private val logger: KLogger) {
|
||||||
|
private var nextId: Long = 1L
|
||||||
|
private var nextGuildCardNo: Int = 1
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps usernames to accounts. Accounts are created on the fly.
|
||||||
|
*/
|
||||||
|
private val accounts = mutableMapOf<String, AccountData>()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logged in accounts must always be logged out with [logOut].
|
||||||
|
*/
|
||||||
|
fun logIn(username: String, password: String): LogInResult =
|
||||||
|
synchronized(this) {
|
||||||
|
val data = accounts.getOrPut(username) {
|
||||||
|
val accountId = nextId++
|
||||||
|
AccountData(
|
||||||
|
account = Account(
|
||||||
|
id = accountId,
|
||||||
|
username = username,
|
||||||
|
guildCardNo = nextGuildCardNo++,
|
||||||
|
teamId = 1337,
|
||||||
|
characters = listOf(
|
||||||
|
Character(
|
||||||
|
id = nextId++,
|
||||||
|
accountId = accountId,
|
||||||
|
name = "${username.take(14)} 1",
|
||||||
|
sectionId = SectionId.Viridia,
|
||||||
|
exp = 1_000_000,
|
||||||
|
level = 200,
|
||||||
|
)
|
||||||
|
),
|
||||||
|
),
|
||||||
|
playing = null,
|
||||||
|
password = password,
|
||||||
|
loggedIn = false,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (password != data.password) {
|
||||||
|
LogInResult.BadPassword
|
||||||
|
} else if (data.loggedIn) {
|
||||||
|
LogInResult.AlreadyLoggedIn
|
||||||
|
} else {
|
||||||
|
data.loggedIn = true
|
||||||
|
LogInResult.Ok(data.account)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun logOut(accountId: Long) {
|
||||||
|
synchronized(this) {
|
||||||
|
val data = accounts.values.find { it.account.id == accountId }
|
||||||
|
|
||||||
|
if (data == null) {
|
||||||
|
logger.warn {
|
||||||
|
"Trying to log out nonexistent account $accountId."
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!data.loggedIn) {
|
||||||
|
logger.warn {
|
||||||
|
"""Trying to log out account ${data.account.id} "${data.account.username}" while it wasn't logged in."""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
data.playing = null
|
||||||
|
data.loggedIn = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getAccountById(accountId: Long): Account? =
|
||||||
|
synchronized(this) {
|
||||||
|
accounts.values.find { it.account.id == accountId }?.account
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setAccountPlaying(accountId: Long, char: Character, blockId: Int): Account {
|
||||||
|
synchronized(this) {
|
||||||
|
val data = accounts.values.first { it.account.id == accountId }
|
||||||
|
data.playing = PlayingAccount(data.account, char, blockId)
|
||||||
|
return data.account
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getAccountsByBlock(blockId: Int): List<PlayingAccount> =
|
||||||
|
synchronized(this) {
|
||||||
|
accounts.values
|
||||||
|
.filter { it.loggedIn && it.playing?.blockId == blockId }
|
||||||
|
.mapNotNull { it.playing }
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class LogInResult {
|
||||||
|
class Ok(val account: Account) : LogInResult()
|
||||||
|
object BadPassword : LogInResult()
|
||||||
|
object AlreadyLoggedIn : LogInResult()
|
||||||
|
}
|
||||||
|
|
||||||
|
private class AccountData(
|
||||||
|
var account: Account,
|
||||||
|
var playing: PlayingAccount?,
|
||||||
|
val password: String,
|
||||||
|
var loggedIn: Boolean,
|
||||||
|
) {
|
||||||
|
init {
|
||||||
|
require(password.length <= 16)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -180,8 +180,9 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
lobbyNo: UByte,
|
lobbyNo: UByte,
|
||||||
blockNo: UShort,
|
blockNo: UShort,
|
||||||
event: UShort,
|
event: UShort,
|
||||||
|
players: List<LobbyPlayer>,
|
||||||
) : this(
|
) : this(
|
||||||
buf(0x0067, 12 + 1312, flags = 1) { // TODO: Set flags to player count.
|
buf(0x0067, 12 + players.size * 1312, flags = players.size) {
|
||||||
writeUByte(clientId)
|
writeUByte(clientId)
|
||||||
writeUByte(leaderId)
|
writeUByte(leaderId)
|
||||||
writeByte(if (disableUdp) 1 else 0)
|
writeByte(if (disableUdp) 1 else 0)
|
||||||
@ -189,7 +190,16 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
writeUShort(blockNo)
|
writeUShort(blockNo)
|
||||||
writeUShort(event)
|
writeUShort(event)
|
||||||
writeInt(0) // Unused.
|
writeInt(0) // Unused.
|
||||||
repeat(328) { writeInt(0) }
|
|
||||||
|
for (player in players) {
|
||||||
|
writeInt(player.playerTag)
|
||||||
|
writeInt(player.guildCardNo)
|
||||||
|
repeat(5) { writeInt(0) } // Unknown.
|
||||||
|
writeInt(player.clientId)
|
||||||
|
writeStringUtf16(player.charName, 32)
|
||||||
|
writeInt(0) // Unknown.
|
||||||
|
repeat(311) { writeInt(0) }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -223,10 +233,10 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
|
|
||||||
// 0x0093
|
// 0x0093
|
||||||
class Authenticate(buffer: Buffer) : BbMessage(buffer) {
|
class Authenticate(buffer: Buffer) : BbMessage(buffer) {
|
||||||
val guildCard: Int get() = int(4)
|
val guildCardNo: Int get() = int(4)
|
||||||
val version: Short get() = short(8)
|
val version: Short get() = short(8)
|
||||||
val teamId: Int get() = int(16)
|
val teamId: Int get() = int(16)
|
||||||
val userName: String get() = stringAscii(offset = 20, maxByteLength = 16)
|
val username: String get() = stringAscii(offset = 20, maxByteLength = 16)
|
||||||
val password: String get() = stringAscii(offset = 68, maxByteLength = 16)
|
val password: String get() = stringAscii(offset = 68, maxByteLength = 16)
|
||||||
val magic: Int get() = int(132) // Should be 0xDEADBEEF
|
val magic: Int get() = int(132) // Should be 0xDEADBEEF
|
||||||
val charSlot: Int get() = byte(136).toInt()
|
val charSlot: Int get() = byte(136).toInt()
|
||||||
@ -301,17 +311,19 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 0x00E0
|
// 0x00E0
|
||||||
class GetAccount(buffer: Buffer) : BbMessage(buffer)
|
class GetAccount(buffer: Buffer) : BbMessage(buffer) {
|
||||||
|
constructor() : this(buf(0x00E1))
|
||||||
|
}
|
||||||
|
|
||||||
// 0x00E2
|
// 0x00E2
|
||||||
class Account(buffer: Buffer) : BbMessage(buffer) {
|
class Account(buffer: Buffer) : BbMessage(buffer) {
|
||||||
constructor(guildCard: Int, teamId: Int) : this(
|
constructor(guildCardNo: Int, teamId: Int) : this(
|
||||||
buf(0x00E2, 2804) {
|
buf(0x00E2, 2804) {
|
||||||
// 276 Bytes of unknown data.
|
// 276 Bytes of unknown data.
|
||||||
repeat(69) { writeInt(0) }
|
repeat(69) { writeInt(0) }
|
||||||
writeByteArray(DEFAULT_KEYBOARD_CONFIG)
|
writeByteArray(DEFAULT_KEYBOARD_CONFIG)
|
||||||
writeByteArray(DEFAULT_GAMEPAD_CONFIG)
|
writeByteArray(DEFAULT_GAMEPAD_CONFIG)
|
||||||
writeInt(guildCard)
|
writeInt(guildCardNo)
|
||||||
writeInt(teamId)
|
writeInt(teamId)
|
||||||
// 2092 Bytes of team data.
|
// 2092 Bytes of team data.
|
||||||
repeat(523) { writeInt(0) }
|
repeat(523) { writeInt(0) }
|
||||||
@ -383,12 +395,26 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
|
|
||||||
// 0x00E6
|
// 0x00E6
|
||||||
class AuthData(buffer: Buffer) : BbMessage(buffer) {
|
class AuthData(buffer: Buffer) : BbMessage(buffer) {
|
||||||
|
var status: AuthStatus
|
||||||
|
get() = when (val value = int(0)) {
|
||||||
|
0 -> AuthStatus.Success
|
||||||
|
1 -> AuthStatus.Error
|
||||||
|
8 -> AuthStatus.Nonexistent
|
||||||
|
else -> AuthStatus.Unknown(value)
|
||||||
|
}
|
||||||
|
set(status) = setInt(0, when (status) {
|
||||||
|
AuthStatus.Success -> 0
|
||||||
|
AuthStatus.Error -> 1
|
||||||
|
AuthStatus.Nonexistent -> 8
|
||||||
|
is AuthStatus.Unknown -> status.value
|
||||||
|
})
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
status: AuthStatus,
|
status: AuthStatus,
|
||||||
guildCard: Int,
|
guildCardNo: Int,
|
||||||
teamId: Int,
|
teamId: Int,
|
||||||
slot: Int,
|
charSlot: Int,
|
||||||
selected: Boolean,
|
charSelected: Boolean,
|
||||||
) : this(
|
) : this(
|
||||||
buf(0x00E6, 60) {
|
buf(0x00E6, 60) {
|
||||||
writeInt(
|
writeInt(
|
||||||
@ -396,27 +422,31 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
AuthStatus.Success -> 0
|
AuthStatus.Success -> 0
|
||||||
AuthStatus.Error -> 1
|
AuthStatus.Error -> 1
|
||||||
AuthStatus.Nonexistent -> 8
|
AuthStatus.Nonexistent -> 8
|
||||||
|
is AuthStatus.Unknown -> status.value
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
writeInt(0x10000)
|
writeInt(0x10000)
|
||||||
writeInt(guildCard)
|
writeInt(guildCardNo)
|
||||||
writeInt(teamId)
|
writeInt(teamId)
|
||||||
writeInt(
|
writeInt(
|
||||||
if (status == AuthStatus.Success) (0xDEADBEEF).toInt() else 0
|
if (status == AuthStatus.Success) (0xDEADBEEF).toInt() else 0
|
||||||
)
|
)
|
||||||
writeByte(slot.toByte())
|
writeByte(charSlot.toByte())
|
||||||
writeByte(if (selected) 1 else 0)
|
writeByte(if (charSelected) 1 else 0)
|
||||||
// 34 Bytes of unknown data.
|
// 34 Bytes of unknown data.
|
||||||
writeShort(0)
|
writeShort(0)
|
||||||
repeat(8) { writeInt(0) }
|
repeat(8) { writeInt(0) }
|
||||||
writeInt(0x102)
|
writeInt(0x102)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
override fun toString(): String =
|
||||||
|
messageString("status" to status)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 0x00E7
|
// 0x00E7
|
||||||
class FullCharacterData(buffer: Buffer) : BbMessage(buffer) {
|
class FullCharacterData(buffer: Buffer) : BbMessage(buffer) {
|
||||||
constructor(char: PsoCharData) : this(
|
constructor(char: PsoCharData, name: String, sectionId: Byte, charClass: Byte) : this(
|
||||||
buf(0x00E7, 14744) {
|
buf(0x00E7, 14744) {
|
||||||
repeat(211) { writeInt(0) }
|
repeat(211) { writeInt(0) }
|
||||||
repeat(3) { writeShort(0) } // ATP/MST/EVP
|
repeat(3) { writeShort(0) } // ATP/MST/EVP
|
||||||
@ -426,7 +456,15 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
repeat(2) { writeInt(0) } // Unknown.
|
repeat(2) { writeInt(0) } // Unknown.
|
||||||
writeInt(char.level)
|
writeInt(char.level)
|
||||||
writeInt(char.exp)
|
writeInt(char.exp)
|
||||||
repeat(2835) { writeInt(0) }
|
repeat(92) { writeInt(0) } // Rest of char.
|
||||||
|
repeat(1275) { writeInt(0) }
|
||||||
|
writeStringUtf16(name, byteLength = 48)
|
||||||
|
repeat(8) { writeInt(0) } // Team name.
|
||||||
|
repeat(44) { writeInt(0) } // Guild card description.
|
||||||
|
writeShort(0) // Reserved.
|
||||||
|
writeByte(sectionId)
|
||||||
|
writeByte(charClass)
|
||||||
|
repeat(1403) { writeInt(0) }
|
||||||
writeByteArray(DEFAULT_KEYBOARD_CONFIG)
|
writeByteArray(DEFAULT_KEYBOARD_CONFIG)
|
||||||
writeByteArray(DEFAULT_GAMEPAD_CONFIG)
|
writeByteArray(DEFAULT_GAMEPAD_CONFIG)
|
||||||
repeat(527) { writeInt(0) }
|
repeat(527) { writeInt(0) }
|
||||||
@ -523,8 +561,15 @@ sealed class BbMessage(override val buffer: Buffer) : AbstractMessage(BB_HEADER_
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enum class AuthStatus {
|
sealed class AuthStatus {
|
||||||
Success, Error, Nonexistent
|
object Success : AuthStatus()
|
||||||
|
object Error : AuthStatus()
|
||||||
|
object Nonexistent : AuthStatus()
|
||||||
|
class Unknown(val value: Int) : AuthStatus() {
|
||||||
|
override fun toString(): String = "Unknown[$value]"
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun toString(): String = this::class.simpleName!!
|
||||||
}
|
}
|
||||||
|
|
||||||
class PsoCharacter(
|
class PsoCharacter(
|
||||||
@ -561,7 +606,7 @@ class GuildCardEntry(
|
|||||||
)
|
)
|
||||||
|
|
||||||
class GuildCard(
|
class GuildCard(
|
||||||
val entries: List<GuildCardEntry>
|
val entries: List<GuildCardEntry>,
|
||||||
)
|
)
|
||||||
|
|
||||||
class FileListEntry(
|
class FileListEntry(
|
||||||
@ -610,3 +655,10 @@ class PsoCharData(
|
|||||||
val level: Int,
|
val level: Int,
|
||||||
val exp: Int,
|
val exp: Int,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
class LobbyPlayer(
|
||||||
|
val playerTag: Int,
|
||||||
|
val guildCardNo: Int,
|
||||||
|
val clientId: Int,
|
||||||
|
val charName: String,
|
||||||
|
)
|
||||||
|
@ -80,6 +80,10 @@ abstract class AbstractMessage(override val headerSize: Int) : Message {
|
|||||||
buffer.setShort(headerSize + offset, value)
|
buffer.setShort(headerSize + offset, value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected fun setInt(offset: Int, value: Int) {
|
||||||
|
buffer.setInt(headerSize + offset, value)
|
||||||
|
}
|
||||||
|
|
||||||
protected fun setByteArray(offset: Int, array: ByteArray) {
|
protected fun setByteArray(offset: Int, array: ByteArray) {
|
||||||
for ((index, byte) in array.withIndex()) {
|
for ((index, byte) in array.withIndex()) {
|
||||||
setByte(offset + index, byte)
|
setByte(offset + index, byte)
|
||||||
|
@ -4,12 +4,15 @@ import world.phantasmal.core.math.clamp
|
|||||||
import world.phantasmal.psolib.Endianness
|
import world.phantasmal.psolib.Endianness
|
||||||
import world.phantasmal.psolib.buffer.Buffer
|
import world.phantasmal.psolib.buffer.Buffer
|
||||||
import world.phantasmal.psolib.cursor.cursor
|
import world.phantasmal.psolib.cursor.cursor
|
||||||
|
import world.phantasmal.psoserv.data.AccountStore
|
||||||
|
import world.phantasmal.psoserv.data.AccountStore.LogInResult
|
||||||
import world.phantasmal.psoserv.encryption.BbCipher
|
import world.phantasmal.psoserv.encryption.BbCipher
|
||||||
import world.phantasmal.psoserv.encryption.Cipher
|
import world.phantasmal.psoserv.encryption.Cipher
|
||||||
import world.phantasmal.psoserv.messages.*
|
import world.phantasmal.psoserv.messages.*
|
||||||
import world.phantasmal.psoserv.utils.crc32Checksum
|
import world.phantasmal.psoserv.utils.crc32Checksum
|
||||||
|
|
||||||
class AccountServer(
|
class AccountServer(
|
||||||
|
private val accountStore: AccountStore,
|
||||||
bindPair: Inet4Pair,
|
bindPair: Inet4Pair,
|
||||||
private val ships: List<ShipInfo>,
|
private val ships: List<ShipInfo>,
|
||||||
) : GameServer<BbMessage>("account", bindPair) {
|
) : GameServer<BbMessage>("account", bindPair) {
|
||||||
@ -23,11 +26,10 @@ class AccountServer(
|
|||||||
serverCipher: Cipher,
|
serverCipher: Cipher,
|
||||||
clientCipher: Cipher,
|
clientCipher: Cipher,
|
||||||
): ClientReceiver<BbMessage> = object : ClientReceiver<BbMessage> {
|
): ClientReceiver<BbMessage> = object : ClientReceiver<BbMessage> {
|
||||||
|
private var accountId: Long? = null
|
||||||
private val guildCardBuffer = Buffer.withSize(54672)
|
private val guildCardBuffer = Buffer.withSize(54672)
|
||||||
private var fileChunkNo = 0
|
private var fileChunkNo = 0
|
||||||
private var guildCard: Int = -1
|
private var charSlot: Int = 0
|
||||||
private var teamId: Int = -1
|
|
||||||
private var slot: Int = 0
|
|
||||||
private var charSelected: Boolean = false
|
private var charSelected: Boolean = false
|
||||||
|
|
||||||
init {
|
init {
|
||||||
@ -43,74 +45,105 @@ class AccountServer(
|
|||||||
|
|
||||||
override fun process(message: BbMessage): Boolean = when (message) {
|
override fun process(message: BbMessage): Boolean = when (message) {
|
||||||
is BbMessage.Authenticate -> {
|
is BbMessage.Authenticate -> {
|
||||||
// TODO: Actual authentication.
|
when (
|
||||||
guildCard = message.guildCard
|
val result = accountStore.logIn(
|
||||||
teamId = message.teamId
|
message.username,
|
||||||
|
message.password,
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
is LogInResult.Ok -> {
|
||||||
|
val account = result.account
|
||||||
|
this.accountId = account.id
|
||||||
|
|
||||||
|
charSlot = message.charSlot
|
||||||
|
charSelected = message.charSelected
|
||||||
|
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.AuthData(
|
BbMessage.AuthData(
|
||||||
AuthStatus.Success,
|
AuthStatus.Success,
|
||||||
guildCard,
|
account.guildCardNo,
|
||||||
teamId,
|
account.teamId,
|
||||||
slot,
|
charSlot,
|
||||||
charSelected,
|
charSelected,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
// When the player has selected a character, we send him the list of ships to
|
// When the player has selected a character, we send him the list of ships
|
||||||
// choose from.
|
// to choose from.
|
||||||
if (message.charSelected) {
|
if (charSelected) {
|
||||||
ctx.send(BbMessage.ShipList(ships.map { it.uiName }))
|
ctx.send(BbMessage.ShipList(ships.map { it.uiName }))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
LogInResult.BadPassword -> {
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.AuthData(
|
||||||
|
AuthStatus.Nonexistent,
|
||||||
|
message.guildCardNo,
|
||||||
|
message.teamId,
|
||||||
|
message.charSlot,
|
||||||
|
message.charSelected,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
LogInResult.AlreadyLoggedIn -> {
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.AuthData(
|
||||||
|
AuthStatus.Error,
|
||||||
|
message.guildCardNo,
|
||||||
|
message.teamId,
|
||||||
|
message.charSlot,
|
||||||
|
message.charSelected,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
is BbMessage.GetAccount -> {
|
is BbMessage.GetAccount -> {
|
||||||
// TODO: Send correct guild card number and team ID.
|
accountId?.let(accountStore::getAccountById)?.let {
|
||||||
ctx.send(BbMessage.Account(0, 0))
|
ctx.send(BbMessage.Account(it.guildCardNo, it.teamId))
|
||||||
|
}
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
is BbMessage.CharSelect -> {
|
is BbMessage.CharSelect -> {
|
||||||
|
val account = accountId?.let(accountStore::getAccountById)
|
||||||
|
|
||||||
|
if (account != null && message.slot in account.characters.indices) {
|
||||||
if (message.selected) {
|
if (message.selected) {
|
||||||
// Player has chosen a character.
|
// Player has chosen a character.
|
||||||
// TODO: Verify slot.
|
charSlot = message.slot
|
||||||
if (slot in 0..3) {
|
|
||||||
slot = message.slot
|
|
||||||
charSelected = true
|
charSelected = true
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.AuthData(
|
BbMessage.AuthData(
|
||||||
AuthStatus.Success,
|
AuthStatus.Success,
|
||||||
guildCard,
|
account.guildCardNo,
|
||||||
teamId,
|
account.teamId,
|
||||||
slot,
|
charSlot,
|
||||||
charSelected,
|
charSelected,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.CharSelectAck(slot, CharSelectStatus.Select)
|
BbMessage.CharSelectAck(charSlot, CharSelectStatus.Select)
|
||||||
)
|
)
|
||||||
} else {
|
|
||||||
ctx.send(
|
|
||||||
BbMessage.CharSelectAck(slot, CharSelectStatus.Nonexistent)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Player is previewing characters.
|
// Player is previewing characters.
|
||||||
// TODO: Look up character data.
|
val char = account.characters[message.slot]
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.Char(
|
BbMessage.Char(
|
||||||
PsoCharacter(
|
PsoCharacter(
|
||||||
slot = message.slot,
|
slot = message.slot,
|
||||||
exp = 0,
|
exp = char.exp,
|
||||||
level = 0,
|
level = char.level - 1,
|
||||||
guildCardString = "",
|
guildCardString = "",
|
||||||
nameColor = 0,
|
nameColor = 0,
|
||||||
model = 0,
|
model = 0,
|
||||||
nameColorChecksum = 0,
|
nameColorChecksum = 0,
|
||||||
sectionId = message.slot,
|
sectionId = char.sectionId.ordinal,
|
||||||
characterClass = message.slot,
|
characterClass = 0,
|
||||||
costume = 0,
|
costume = 0,
|
||||||
skin = 0,
|
skin = 0,
|
||||||
face = 0,
|
face = 0,
|
||||||
@ -121,12 +154,17 @@ class AccountServer(
|
|||||||
hairBlue = 0,
|
hairBlue = 0,
|
||||||
propX = 0.5,
|
propX = 0.5,
|
||||||
propY = 0.5,
|
propY = 0.5,
|
||||||
name = "Phantasmal ${message.slot}",
|
name = char.name,
|
||||||
playTime = 0,
|
playTime = 0,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.CharSelectAck(message.slot, CharSelectStatus.Nonexistent)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
@ -205,17 +243,34 @@ class AccountServer(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Disconnect.
|
// Log out and disconnect.
|
||||||
|
logOut()
|
||||||
false
|
false
|
||||||
} else {
|
} else {
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
is BbMessage.Disconnect -> false
|
is BbMessage.Disconnect -> {
|
||||||
|
// Log out and disconnect.
|
||||||
|
logOut()
|
||||||
|
false
|
||||||
|
}
|
||||||
|
|
||||||
else -> ctx.unexpectedMessage(message)
|
else -> ctx.unexpectedMessage(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun connectionClosed() {
|
||||||
|
logOut()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun logOut() {
|
||||||
|
try {
|
||||||
|
accountId?.let(accountStore::logOut)
|
||||||
|
} finally {
|
||||||
|
accountId = null
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
@ -35,14 +35,15 @@ class AuthServer(
|
|||||||
|
|
||||||
override fun process(message: BbMessage): Boolean = when (message) {
|
override fun process(message: BbMessage): Boolean = when (message) {
|
||||||
is BbMessage.Authenticate -> {
|
is BbMessage.Authenticate -> {
|
||||||
// TODO: Actual authentication.
|
// Don't actually authenticate, since we're simply redirecting the player to the
|
||||||
|
// account server.
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.AuthData(
|
BbMessage.AuthData(
|
||||||
AuthStatus.Success,
|
AuthStatus.Success,
|
||||||
message.guildCard,
|
message.guildCardNo,
|
||||||
message.teamId,
|
message.teamId,
|
||||||
slot = 0,
|
charSlot = 0,
|
||||||
selected = false,
|
charSelected = false,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
ctx.send(
|
ctx.send(
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
package world.phantasmal.psoserv.servers
|
package world.phantasmal.psoserv.servers
|
||||||
|
|
||||||
|
import world.phantasmal.psoserv.data.AccountStore
|
||||||
|
import world.phantasmal.psoserv.data.AccountStore.LogInResult
|
||||||
import world.phantasmal.psoserv.encryption.BbCipher
|
import world.phantasmal.psoserv.encryption.BbCipher
|
||||||
import world.phantasmal.psoserv.encryption.Cipher
|
import world.phantasmal.psoserv.encryption.Cipher
|
||||||
import world.phantasmal.psoserv.messages.AuthStatus
|
import world.phantasmal.psoserv.messages.*
|
||||||
import world.phantasmal.psoserv.messages.BbMessage
|
|
||||||
import world.phantasmal.psoserv.messages.BbMessageDescriptor
|
|
||||||
import world.phantasmal.psoserv.messages.PsoCharData
|
|
||||||
|
|
||||||
class BlockServer(
|
class BlockServer(
|
||||||
|
private val accountStore: AccountStore,
|
||||||
name: String,
|
name: String,
|
||||||
bindPair: Inet4Pair,
|
bindPair: Inet4Pair,
|
||||||
private val blockNo: Int,
|
private val blockId: Int,
|
||||||
) : GameServer<BbMessage>(name, bindPair) {
|
) : GameServer<BbMessage>(name, bindPair) {
|
||||||
|
|
||||||
override val messageDescriptor = BbMessageDescriptor
|
override val messageDescriptor = BbMessageDescriptor
|
||||||
@ -22,6 +22,8 @@ class BlockServer(
|
|||||||
serverCipher: Cipher,
|
serverCipher: Cipher,
|
||||||
clientCipher: Cipher,
|
clientCipher: Cipher,
|
||||||
): ClientReceiver<BbMessage> = object : ClientReceiver<BbMessage> {
|
): ClientReceiver<BbMessage> = object : ClientReceiver<BbMessage> {
|
||||||
|
private var accountId: Long? = null
|
||||||
|
|
||||||
init {
|
init {
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.InitEncryption(
|
BbMessage.InitEncryption(
|
||||||
@ -35,27 +37,78 @@ class BlockServer(
|
|||||||
|
|
||||||
override fun process(message: BbMessage): Boolean = when (message) {
|
override fun process(message: BbMessage): Boolean = when (message) {
|
||||||
is BbMessage.Authenticate -> {
|
is BbMessage.Authenticate -> {
|
||||||
// TODO: Actual authentication.
|
when (
|
||||||
|
val result = accountStore.logIn(message.username, message.password)
|
||||||
|
) {
|
||||||
|
is LogInResult.Ok -> {
|
||||||
|
accountId = result.account.id
|
||||||
|
val char = result.account.characters.getOrNull(message.charSlot)
|
||||||
|
|
||||||
|
if (char == null) {
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.AuthData(
|
BbMessage.AuthData(
|
||||||
AuthStatus.Success,
|
AuthStatus.Nonexistent,
|
||||||
message.guildCard,
|
message.guildCardNo,
|
||||||
message.teamId,
|
message.teamId,
|
||||||
message.charSlot,
|
message.charSlot,
|
||||||
message.charSelected,
|
message.charSelected,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
} else {
|
||||||
|
val account = accountStore.setAccountPlaying(
|
||||||
|
result.account.id,
|
||||||
|
char,
|
||||||
|
blockId,
|
||||||
|
)
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.AuthData(
|
||||||
|
AuthStatus.Success,
|
||||||
|
account.guildCardNo,
|
||||||
|
account.teamId,
|
||||||
|
message.charSlot,
|
||||||
|
message.charSelected,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
ctx.send(BbMessage.LobbyList())
|
ctx.send(BbMessage.LobbyList())
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.FullCharacterData(
|
BbMessage.FullCharacterData(
|
||||||
PsoCharData(
|
PsoCharData(
|
||||||
hp = 20,
|
hp = 0,
|
||||||
level = 0,
|
level = char.level - 1,
|
||||||
exp = 0,
|
exp = char.exp,
|
||||||
)
|
),
|
||||||
|
char.name,
|
||||||
|
char.sectionId.ordinal.toByte(),
|
||||||
|
charClass = 0,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
ctx.send(BbMessage.GetCharData())
|
ctx.send(BbMessage.GetCharData())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LogInResult.BadPassword -> {
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.AuthData(
|
||||||
|
AuthStatus.Nonexistent,
|
||||||
|
message.guildCardNo,
|
||||||
|
message.teamId,
|
||||||
|
message.charSlot,
|
||||||
|
message.charSelected,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
LogInResult.AlreadyLoggedIn -> {
|
||||||
|
ctx.send(
|
||||||
|
BbMessage.AuthData(
|
||||||
|
AuthStatus.Error,
|
||||||
|
message.guildCardNo,
|
||||||
|
message.teamId,
|
||||||
|
message.charSlot,
|
||||||
|
message.charSelected,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
@ -67,15 +120,41 @@ class BlockServer(
|
|||||||
leaderId = 0u,
|
leaderId = 0u,
|
||||||
disableUdp = true,
|
disableUdp = true,
|
||||||
lobbyNo = 0u,
|
lobbyNo = 0u,
|
||||||
blockNo = blockNo.toUShort(),
|
blockNo = blockId.toUShort(),
|
||||||
event = 0u,
|
event = 0u,
|
||||||
|
players = accountStore.getAccountsByBlock(blockId).map {
|
||||||
|
LobbyPlayer(
|
||||||
|
playerTag = 0,
|
||||||
|
guildCardNo = it.account.guildCardNo,
|
||||||
|
clientId = 0,
|
||||||
|
charName = it.char.name,
|
||||||
|
)
|
||||||
|
}
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
is BbMessage.Disconnect -> {
|
||||||
|
// Log out and disconnect.
|
||||||
|
logOut()
|
||||||
|
false
|
||||||
|
}
|
||||||
|
|
||||||
else -> ctx.unexpectedMessage(message)
|
else -> ctx.unexpectedMessage(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun connectionClosed() {
|
||||||
|
logOut()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun logOut() {
|
||||||
|
try {
|
||||||
|
accountId?.let(accountStore::logOut)
|
||||||
|
} finally {
|
||||||
|
accountId = null
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -33,6 +33,7 @@ abstract class GameServer<MessageType : Message>(
|
|||||||
|
|
||||||
protected interface ClientReceiver<MessageType : Message> {
|
protected interface ClientReceiver<MessageType : Message> {
|
||||||
fun process(message: MessageType): Boolean
|
fun process(message: MessageType): Boolean
|
||||||
|
fun connectionClosed() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected class ClientContext<MessageType : Message>(
|
protected class ClientContext<MessageType : Message>(
|
||||||
@ -74,5 +75,9 @@ abstract class GameServer<MessageType : Message>(
|
|||||||
// Close the connection.
|
// Close the connection.
|
||||||
ProcessResult.Done
|
ProcessResult.Done
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun socketClosed() {
|
||||||
|
receiver.connectionClosed()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -36,11 +36,12 @@ class ShipServer(
|
|||||||
|
|
||||||
override fun process(message: BbMessage): Boolean = when (message) {
|
override fun process(message: BbMessage): Boolean = when (message) {
|
||||||
is BbMessage.Authenticate -> {
|
is BbMessage.Authenticate -> {
|
||||||
// TODO: Actual authentication.
|
// Don't actually authenticate, since we're simply letting the player choose a block
|
||||||
|
// and then redirecting him to the corresponding block server.
|
||||||
ctx.send(
|
ctx.send(
|
||||||
BbMessage.AuthData(
|
BbMessage.AuthData(
|
||||||
AuthStatus.Success,
|
AuthStatus.Success,
|
||||||
message.guildCard,
|
message.guildCardNo,
|
||||||
message.teamId,
|
message.teamId,
|
||||||
message.charSlot,
|
message.charSlot,
|
||||||
message.charSelected,
|
message.charSelected,
|
||||||
|
@ -48,6 +48,7 @@ abstract class SocketHandler<MessageType : Message>(
|
|||||||
|
|
||||||
if (readSize == -1) {
|
if (readSize == -1) {
|
||||||
// Close the connection if no more bytes available.
|
// Close the connection if no more bytes available.
|
||||||
|
logger.debug { "$name ($sockName) end of stream." }
|
||||||
break@readLoop
|
break@readLoop
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user