@ -1,5 +1,11 @@
package org.openrs2.game.net.login
package org.openrs2.game.net.login
import com.github.michaelbull.result.Err
import com.github.michaelbull.result.Ok
import com.github.michaelbull.result.Result
import com.github.michaelbull.result.and
import com.github.michaelbull.result.coroutines.binding.binding
import com.github.michaelbull.result.getErrorOr
import io.netty.buffer.Unpooled
import io.netty.buffer.Unpooled
import io.netty.channel.ChannelFutureListener
import io.netty.channel.ChannelFutureListener
import io.netty.channel.ChannelHandlerContext
import io.netty.channel.ChannelHandlerContext
@ -10,50 +16,88 @@ import io.netty.handler.codec.http.HttpRequestDecoder
import io.netty.handler.codec.http.HttpResponseEncoder
import io.netty.handler.codec.http.HttpResponseEncoder
import io.netty.handler.codec.string.StringDecoder
import io.netty.handler.codec.string.StringDecoder
import io.netty.handler.timeout.IdleStateEvent
import io.netty.handler.timeout.IdleStateEvent
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.asCoroutineDispatcher
import kotlinx.coroutines.cancel
import kotlinx.coroutines.launch
import org.openrs2.buffer.copiedBuffer
import org.openrs2.buffer.copiedBuffer
import org.openrs2.conf.CountryCode
import org.openrs2.crypto.secureRandom
import org.openrs2.game.cluster.Cluster
import org.openrs2.game.cluster.Cluster
import org.openrs2.game.net.crossdomain.CrossDomainChannelHandler
import org.openrs2.game.net.crossdomain.CrossDomainChannelHandler
import org.openrs2.game.net.http.Http
import org.openrs2.game.net.http.Http
import org.openrs2.game.net.jaggrab.JaggrabChannelHandler
import org.openrs2.game.net.jaggrab.JaggrabChannelHandler
import org.openrs2.game.net.js5.Js5ChannelHandler
import org.openrs2.game.net.js5.Js5ChannelHandler
import org.openrs2.game.store.PlayerStore
import org.openrs2.protocol.Protocol
import org.openrs2.protocol.Protocol
import org.openrs2.protocol.Rs2Decoder
import org.openrs2.protocol.Rs2Decoder
import org.openrs2.protocol.Rs2Encoder
import org.openrs2.protocol.Rs2Encoder
import org.openrs2.protocol.create.downstream.CreateDownstream
import org.openrs2.protocol.create.downstream.CreateResponse
import org.openrs2.protocol.jaggrab.upstream.JaggrabRequestDecoder
import org.openrs2.protocol.jaggrab.upstream.JaggrabRequestDecoder
import org.openrs2.protocol.js5.downstream.Js5LoginResponse
import org.openrs2.protocol.js5.downstream.Js5LoginResponse
import org.openrs2.protocol.js5.downstream.Js5RemoteDownstream
import org.openrs2.protocol.js5.downstream.Js5RemoteDownstream
import org.openrs2.protocol.js5.downstream.Js5ResponseEncoder
import org.openrs2.protocol.js5.downstream.Js5ResponseEncoder
import org.openrs2.protocol.js5.downstream.XorDecoder
import org.openrs2.protocol.js5.downstream.XorDecoder
import org.openrs2.protocol.js5.upstream.Js5RequestDecoder
import org.openrs2.protocol.js5.upstream.Js5RequestDecoder
import org.openrs2.protocol.login.downstream.LoginResponse
import org.openrs2.protocol.login.upstream.LoginRequest
import org.openrs2.protocol.login.upstream.LoginRequest
import org.openrs2.protocol.world.downstream.WorldListDownstream
import org.openrs2.protocol.world.downstream.WorldListDownstream
import org.openrs2.protocol.world.downstream.WorldListResponse
import org.openrs2.protocol.world.downstream.WorldListResponse
import java.time.DateTimeException
import java.time.LocalDate
import javax.inject.Inject
import javax.inject.Inject
import javax.inject.Provider
import javax.inject.Provider
public class LoginChannelHandler @Inject constructor (
public class LoginChannelHandler @Inject constructor (
private val cluster : Cluster ,
private val cluster : Cluster ,
private val store : PlayerStore ,
private val js5HandlerProvider : Provider < Js5ChannelHandler > ,
private val js5HandlerProvider : Provider < Js5ChannelHandler > ,
private val jaggrabHandler : JaggrabChannelHandler ,
private val jaggrabHandler : JaggrabChannelHandler ,
@CreateDownstream
private val createDownstreamProtocol : Protocol ,
@Js5RemoteDownstream
@Js5RemoteDownstream
private val js5RemoteDownstreamProtocol : Protocol ,
private val js5RemoteDownstreamProtocol : Protocol ,
@WorldListDownstream
@WorldListDownstream
private val worldListDownstreamProtocol : Protocol
private val worldListDownstreamProtocol : Protocol
) : SimpleChannelInboundHandler < LoginRequest > ( LoginRequest :: class . java ) {
) : SimpleChannelInboundHandler < LoginRequest > ( LoginRequest :: class . java ) {
private lateinit var scope : CoroutineScope
private var usernameHash = 0
private var serverKey = 0L
override fun handlerAdded ( ctx : ChannelHandlerContext ) {
scope = CoroutineScope ( ctx . executor ( ) . asCoroutineDispatcher ( ) )
}
override fun handlerRemoved ( ctx : ChannelHandlerContext ) {
scope . cancel ( )
}
override fun channelActive ( ctx : ChannelHandlerContext ) {
override fun channelActive ( ctx : ChannelHandlerContext ) {
ctx . read ( )
ctx . read ( )
}
}
override fun channelRead0 ( ctx : ChannelHandlerContext , msg : LoginRequest ) {
override fun channelRead0 ( ctx : ChannelHandlerContext , msg : LoginRequest ) {
when ( msg ) {
when ( msg ) {
is LoginRequest . InitGameConnection -> handleInitGameConnection ( ctx , msg )
is LoginRequest . InitJs5RemoteConnection -> handleInitJs5RemoteConnection ( ctx , msg )
is LoginRequest . InitJs5RemoteConnection -> handleInitJs5RemoteConnection ( ctx , msg )
is LoginRequest . InitJaggrabConnection -> handleInitJaggrabConnection ( ctx )
is LoginRequest . InitJaggrabConnection -> handleInitJaggrabConnection ( ctx )
is LoginRequest . CreateCheckDateOfBirthCountry -> handleCreateCheckDateOfBirthCountry ( ctx , msg )
is LoginRequest . CreateCheckName -> handleCreateCheckName ( ctx , msg )
is LoginRequest . CreateAccount -> handleCreateAccount ( ctx , msg )
is LoginRequest . RequestWorldList -> handleRequestWorldList ( ctx , msg )
is LoginRequest . RequestWorldList -> handleRequestWorldList ( ctx , msg )
is LoginRequest . CheckWorldSuitability -> handleCheckWorldSuitability ( ctx , msg )
is LoginRequest . InitCrossDomainConnection -> handleInitCrossDomainConnection ( ctx )
is LoginRequest . InitCrossDomainConnection -> handleInitCrossDomainConnection ( ctx )
else -> Unit
else -> Unit
}
}
}
}
private fun handleInitGameConnection ( ctx : ChannelHandlerContext , msg : LoginRequest . InitGameConnection ) {
usernameHash = msg . usernameHash
serverKey = secureRandom . nextLong ( )
ctx . write ( LoginResponse . ExchangeSessionKey ( serverKey ) , ctx . voidPromise ( ) )
}
private fun handleInitJs5RemoteConnection ( ctx : ChannelHandlerContext , msg : LoginRequest . InitJs5RemoteConnection ) {
private fun handleInitJs5RemoteConnection ( ctx : ChannelHandlerContext , msg : LoginRequest . InitJs5RemoteConnection ) {
val encoder = ctx . pipeline ( ) . get ( Rs2Encoder :: class . java )
val encoder = ctx . pipeline ( ) . get ( Rs2Encoder :: class . java )
encoder . protocol = js5RemoteDownstreamProtocol
encoder . protocol = js5RemoteDownstreamProtocol
@ -91,6 +135,88 @@ public class LoginChannelHandler @Inject constructor(
ctx . pipeline ( ) . remove ( this )
ctx . pipeline ( ) . remove ( this )
}
}
private fun validateDateOfBirth ( year : Int , month : Int , day : Int ) : Result < LocalDate , CreateResponse > {
val date = try {
LocalDate . of ( year , month + 1 , day )
} catch ( ex : DateTimeException ) {
return Err ( CreateResponse . DateOfBirthInvalid )
}
val now = LocalDate . now ( )
if ( date . isAfter ( now ) ) {
return Err ( CreateResponse . DateOfBirthFuture )
} else if ( date . year == now . year ) {
return Err ( CreateResponse . DateOfBirthThisYear )
} else if ( date . year == ( now . year - 1 ) ) {
return Err ( CreateResponse . DateOfBirthLastYear )
}
return Ok ( date )
}
private fun validateCountry ( id : Int ) : Result < CountryCode , CreateResponse > {
// TODO
return Ok ( CountryCode . GB )
}
private fun handleCreateCheckDateOfBirthCountry (
ctx : ChannelHandlerContext ,
msg : LoginRequest . CreateCheckDateOfBirthCountry
) {
val encoder = ctx . pipeline ( ) . get ( Rs2Encoder :: class . java )
encoder . protocol = createDownstreamProtocol
val response = validateDateOfBirth ( msg . year , msg . month , msg . day )
. and ( validateCountry ( msg . country ) )
. getErrorOr ( CreateResponse . Ok )
ctx . write ( response ) . addListener ( ChannelFutureListener . CLOSE )
}
private fun handleCreateCheckName ( ctx : ChannelHandlerContext , msg : LoginRequest . CreateCheckName ) {
val encoder = ctx . pipeline ( ) . get ( Rs2Encoder :: class . java )
encoder . protocol = createDownstreamProtocol
scope . launch {
val response = store . checkName ( msg . username )
. getErrorOr ( CreateResponse . Ok )
ctx . writeAndFlush ( response ) . addListener ( ChannelFutureListener . CLOSE )
}
}
private fun handleCreateAccount ( ctx : ChannelHandlerContext , msg : LoginRequest . CreateAccount ) {
val encoder = ctx . pipeline ( ) . get ( Rs2Encoder :: class . java )
encoder . protocol = createDownstreamProtocol
if ( msg . build != BUILD ) {
ctx . write ( CreateResponse . ClientOutOfDate ) . addListener ( ChannelFutureListener . CLOSE )
return
}
scope . launch {
val response = binding {
val dateOfBirth = validateDateOfBirth ( msg . year , msg . month , msg . day ) . bind ( )
val country = validateCountry ( msg . country ) . bind ( )
store . create (
msg . gameNewsletters ,
msg . otherNewsletters ,
msg . shareDetailsWithBusinessPartners ,
msg . username ,
msg . password ,
msg . affiliate ,
dateOfBirth ,
country ,
msg . email ,
) . bind ( )
} . getErrorOr ( CreateResponse . Ok )
ctx . writeAndFlush ( response ) . addListener ( ChannelFutureListener . CLOSE )
}
}
private fun handleRequestWorldList ( ctx : ChannelHandlerContext , msg : LoginRequest . RequestWorldList ) {
private fun handleRequestWorldList ( ctx : ChannelHandlerContext , msg : LoginRequest . RequestWorldList ) {
val ( worlds , players ) = cluster . getWorldList ( )
val ( worlds , players ) = cluster . getWorldList ( )
@ -116,6 +242,14 @@ public class LoginChannelHandler @Inject constructor(
ctx . write ( WorldListResponse ( worldList , players ) ) . addListener ( ChannelFutureListener . CLOSE )
ctx . write ( WorldListResponse ( worldList , players ) ) . addListener ( ChannelFutureListener . CLOSE )
}
}
private fun handleCheckWorldSuitability ( ctx : ChannelHandlerContext , msg : LoginRequest . CheckWorldSuitability ) {
// TODO
val ( worlds , _ ) = cluster . getWorldList ( )
val id = worlds . firstKey ( )
ctx . write ( LoginResponse . SwitchWorld ( id ) ) . addListener ( ChannelFutureListener . CLOSE )
}
private fun handleInitCrossDomainConnection ( ctx : ChannelHandlerContext ) {
private fun handleInitCrossDomainConnection ( ctx : ChannelHandlerContext ) {
ctx . pipeline ( ) . addLast (
ctx . pipeline ( ) . addLast (
HttpRequestDecoder ( ) ,
HttpRequestDecoder ( ) ,