2021-01-24 05:00:26 -08:00
|
|
|
package main
|
2021-01-14 04:40:31 -08:00
|
|
|
|
2021-01-16 12:48:21 -08:00
|
|
|
import (
|
|
|
|
"crypto/subtle"
|
|
|
|
"encoding/binary"
|
|
|
|
"log"
|
2021-01-18 13:08:22 -08:00
|
|
|
|
|
|
|
"github.com/HimbeerserverDE/srp"
|
|
|
|
"github.com/anon55555/mt/rudp"
|
2021-01-16 12:48:21 -08:00
|
|
|
)
|
2021-01-14 04:40:31 -08:00
|
|
|
|
|
|
|
const (
|
2021-01-14 08:09:06 -08:00
|
|
|
ToClientHello = 0x02
|
|
|
|
ToClientAuthAccept = 0x03
|
|
|
|
ToClientAcceptSudoMode = 0x04
|
|
|
|
ToClientDenySudoMode = 0x05
|
|
|
|
ToClientAccessDenied = 0x0A
|
|
|
|
ToClientBlockdata = 0x20
|
|
|
|
ToClientAddNode = 0x21
|
|
|
|
ToClientRemoveNode = 0x22
|
|
|
|
ToClientInventory = 0x27
|
|
|
|
ToClientTimeOfDay = 0x29
|
|
|
|
ToClientCsmRestrictionFlags = 0x2A
|
|
|
|
ToClientPlayerSpeed = 0x2B
|
|
|
|
ToClientMediaPush = 0x2C
|
|
|
|
ToClientChatMessage = 0x2F
|
2021-01-14 04:40:31 -08:00
|
|
|
ToClientActiveObjectRemoveAdd = 0x31
|
2021-01-14 08:09:06 -08:00
|
|
|
ToClientActiveObjectMessages = 0x32
|
|
|
|
ToClientHp = 0x33
|
|
|
|
ToClientMovePlayer = 0x34
|
|
|
|
ToClientFov = 0x36
|
|
|
|
ToClientDeathscreen = 0x37
|
|
|
|
ToClientMedia = 0x38
|
|
|
|
ToClientTooldef = 0x39
|
|
|
|
ToClientNodedef = 0x3A
|
|
|
|
ToClientCraftitemdef = 0x3B
|
|
|
|
ToClientAnnounceMedia = 0x3C
|
|
|
|
ToClientItemdef = 0x3D
|
|
|
|
ToClientPlaySound = 0x3F
|
|
|
|
ToClientStopSound = 0x40
|
|
|
|
ToClientPrivileges = 0x41
|
|
|
|
ToClientInventoryFormspec = 0x42
|
|
|
|
ToClientDetachedInventory = 0x43
|
|
|
|
ToClientShowFormspec = 0x44
|
|
|
|
ToClientMovement = 0x45
|
|
|
|
ToClientSpawnParticle = 0x46
|
|
|
|
ToClientAddParticlespawner = 0x47
|
|
|
|
ToClientHudAdd = 0x49
|
|
|
|
ToClientHudRm = 0x4A
|
|
|
|
ToClientHudChange = 0x4B
|
|
|
|
ToClientHudSetFlags = 0x4C
|
|
|
|
ToClientHudSetParam = 0x4D
|
|
|
|
ToClientBreath = 0x4E
|
|
|
|
ToClientSetSky = 0x4F
|
2021-01-14 04:40:31 -08:00
|
|
|
ToClientOverrideDayNightRatio = 0x50
|
|
|
|
ToClientLocalPlayerAnimations = 0x51
|
2021-01-14 08:09:06 -08:00
|
|
|
ToClientEyeOffset = 0x52
|
2021-01-14 04:40:31 -08:00
|
|
|
ToClientDeleteParticlespawner = 0x53
|
2021-01-14 08:09:06 -08:00
|
|
|
ToClientCloudParams = 0x54
|
|
|
|
ToClientFadeSound = 0x55
|
|
|
|
ToClientUpdatePlayerList = 0x56
|
2021-01-23 08:01:22 -08:00
|
|
|
ToClientModChannelMsg = 0x57
|
|
|
|
ToClientModChannelSignal = 0x58
|
2021-01-14 08:09:06 -08:00
|
|
|
ToClientNodeMetaChanged = 0x59
|
|
|
|
ToClientSetSun = 0x5A
|
|
|
|
ToClientSetMoon = 0x5B
|
|
|
|
ToClientSetStars = 0x5C
|
|
|
|
ToClientSrpBytesSB = 0x60
|
|
|
|
ToClientFormspecPrepend = 0x61
|
|
|
|
ToClientMinimapModes = 0x62
|
2021-01-14 04:40:31 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2021-01-14 08:09:06 -08:00
|
|
|
ToServerInit = 0x02
|
|
|
|
ToServerInit2 = 0x11
|
2021-01-23 08:01:22 -08:00
|
|
|
ToServerModChannelJoin = 0x17
|
|
|
|
ToServerModChannelLeave = 0x18
|
|
|
|
ToServerModChannelMsg = 0x19
|
2021-01-14 08:09:06 -08:00
|
|
|
ToServerPlayerPos = 0x23
|
|
|
|
ToServerGotblocks = 0x24
|
|
|
|
ToServerDeletedblocks = 0x25
|
2021-01-14 04:40:31 -08:00
|
|
|
ToServerInventoryAction = 0x31
|
2021-01-14 08:09:06 -08:00
|
|
|
ToServerChatMessage = 0x32
|
|
|
|
ToServerDamage = 0x35
|
|
|
|
ToServerPlayerItem = 0x37
|
|
|
|
ToServerRespawn = 0x38
|
|
|
|
ToServerInteract = 0x39
|
|
|
|
ToServerRemovedSounds = 0x3A
|
|
|
|
ToServerNodeMetaFields = 0x3B
|
2021-01-14 04:40:31 -08:00
|
|
|
ToServerInventoryFields = 0x3C
|
2021-01-14 08:09:06 -08:00
|
|
|
ToServerRequestMedia = 0x40
|
|
|
|
ToServerClientReady = 0x43
|
|
|
|
ToServerFirstSrp = 0x50
|
|
|
|
ToServerSrpBytesA = 0x51
|
|
|
|
ToServerSrpBytesM = 0x52
|
2021-01-14 04:40:31 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
AccessDeniedWrongPassword = iota
|
|
|
|
AccessDeniedUnexpectedData
|
|
|
|
AccessDeniedSingleplayer
|
|
|
|
AccessDeniedWrongVersion
|
|
|
|
AccessDeniedWrongCharsInName
|
|
|
|
AccessDeniedWrongName
|
|
|
|
AccessDeniedTooManyUsers
|
|
|
|
AccessDeniedEmptyPassword
|
|
|
|
AccessDeniedAlreadyConnected
|
|
|
|
AccessDeniedServerFail
|
|
|
|
AccessDeniedCustomString
|
|
|
|
AccessDeniedShutdown
|
|
|
|
AccessDeniedCrash
|
|
|
|
)
|
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
func processPktCommand(src, dst *Peer, pkt *rudp.Pkt) bool {
|
2021-01-14 08:09:06 -08:00
|
|
|
if src.IsSrv() {
|
2021-01-14 04:40:31 -08:00
|
|
|
switch cmd := binary.BigEndian.Uint16(pkt.Data[0:2]); cmd {
|
2021-01-14 08:09:06 -08:00
|
|
|
case ToClientActiveObjectRemoveAdd:
|
|
|
|
pkt.Data = processAoRmAdd(dst, pkt.Data)
|
|
|
|
return false
|
2021-01-17 02:08:34 -08:00
|
|
|
case ToClientChatMessage:
|
|
|
|
namelen := binary.BigEndian.Uint16(pkt.Data[4:6])
|
|
|
|
msglen := binary.BigEndian.Uint16(pkt.Data[6+namelen : 8+namelen])
|
|
|
|
msg := pkt.Data[8+namelen:]
|
|
|
|
|
|
|
|
data := make([]byte, 4+msglen*2)
|
|
|
|
data[0] = uint8(0x00)
|
|
|
|
data[1] = uint8(ToServerChatMessage)
|
|
|
|
binary.BigEndian.PutUint16(data[2:4], uint16(msglen))
|
|
|
|
copy(data[4:], msg)
|
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
return processServerChatMessage(dst, rudp.Pkt{Data: data, ChNo: pkt.ChNo})
|
2021-01-23 08:01:22 -08:00
|
|
|
case ToClientModChannelSignal:
|
|
|
|
chlen := binary.BigEndian.Uint16(pkt.Data[3:5])
|
2021-01-24 05:00:26 -08:00
|
|
|
ch := string(pkt.Data[5 : 5+chlen])
|
2021-01-23 08:01:22 -08:00
|
|
|
if ch == rpcCh {
|
|
|
|
switch sig := pkt.Data[2]; sig {
|
|
|
|
case ModChSigJoinOk:
|
|
|
|
src.useRpc = true
|
|
|
|
case ModChSigSetState:
|
|
|
|
state := pkt.Data[5+chlen]
|
|
|
|
if state == ModChStateRO {
|
|
|
|
src.useRpc = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case ToClientModChannelMsg:
|
2021-01-24 05:18:29 -08:00
|
|
|
return processRpc(src, *pkt)
|
2021-01-14 04:40:31 -08:00
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch cmd := binary.BigEndian.Uint16(pkt.Data[0:2]); cmd {
|
|
|
|
case ToServerChatMessage:
|
2021-01-14 08:09:06 -08:00
|
|
|
return processChatMessage(src, *pkt)
|
2021-01-16 12:48:21 -08:00
|
|
|
case ToServerFirstSrp:
|
|
|
|
if src.sudoMode {
|
|
|
|
src.sudoMode = false
|
|
|
|
|
|
|
|
// This is a password change, save verifier and salt
|
|
|
|
lenS := binary.BigEndian.Uint16(pkt.Data[2:4])
|
|
|
|
s := pkt.Data[4 : 4+lenS]
|
|
|
|
|
|
|
|
lenV := binary.BigEndian.Uint16(pkt.Data[4+lenS : 6+lenS])
|
|
|
|
v := pkt.Data[6+lenS : 6+lenS+lenV]
|
|
|
|
|
|
|
|
pwd := encodeVerifierAndSalt(s, v)
|
|
|
|
|
|
|
|
db, err := initAuthDB()
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2021-01-30 03:24:37 -08:00
|
|
|
err = modAuthItem(db, src.Username(), pwd)
|
2021-01-16 12:48:21 -08:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
db.Close()
|
|
|
|
} else {
|
2021-01-30 03:24:37 -08:00
|
|
|
log.Print("User " + src.Username() + " at " + src.Addr().String() + " did not enter sudo mode before attempting to change the password")
|
2021-01-16 12:48:21 -08:00
|
|
|
}
|
|
|
|
return true
|
|
|
|
case ToServerSrpBytesA:
|
|
|
|
if !src.sudoMode {
|
|
|
|
lenA := binary.BigEndian.Uint16(pkt.Data[2:4])
|
|
|
|
A := pkt.Data[4 : 4+lenA]
|
|
|
|
|
|
|
|
db, err := initAuthDB()
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2021-01-30 03:24:37 -08:00
|
|
|
pwd, err := readAuthItem(db, src.Username())
|
2021-01-16 12:48:21 -08:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
db.Close()
|
|
|
|
|
|
|
|
s, v, err := decodeVerifierAndSalt(pwd)
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
B, _, K, err := srp.Handshake(A, v)
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
src.srp_s = s
|
|
|
|
src.srp_A = A
|
|
|
|
src.srp_B = B
|
|
|
|
src.srp_K = K
|
|
|
|
|
|
|
|
// Send SRP_BYTES_S_B
|
|
|
|
data := make([]byte, 6+len(s)+len(B))
|
|
|
|
data[0] = uint8(0x00)
|
|
|
|
data[1] = uint8(ToClientSrpBytesSB)
|
|
|
|
binary.BigEndian.PutUint16(data[2:4], uint16(len(s)))
|
|
|
|
copy(data[4:4+len(s)], s)
|
|
|
|
binary.BigEndian.PutUint16(data[4+len(s):6+len(s)], uint16(len(B)))
|
|
|
|
copy(data[6+len(s):6+len(s)+len(B)], B)
|
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
ack, err := src.Send(rudp.Pkt{Data: data})
|
2021-01-16 12:48:21 -08:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
<-ack
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
case ToServerSrpBytesM:
|
|
|
|
if !src.sudoMode {
|
|
|
|
lenM := binary.BigEndian.Uint16(pkt.Data[2:4])
|
|
|
|
M := pkt.Data[4 : 4+lenM]
|
|
|
|
|
2021-01-30 03:24:37 -08:00
|
|
|
M2 := srp.CalculateM([]byte(src.Username()), src.srp_s, src.srp_A, src.srp_B, src.srp_K)
|
2021-01-16 12:48:21 -08:00
|
|
|
|
|
|
|
if subtle.ConstantTimeCompare(M, M2) == 1 {
|
|
|
|
// Password is correct
|
|
|
|
// Enter sudo mode
|
|
|
|
src.sudoMode = true
|
|
|
|
|
|
|
|
// Send ACCEPT_SUDO_MODE
|
2021-01-19 11:37:35 -08:00
|
|
|
data := []byte{0, ToClientAcceptSudoMode}
|
2021-01-16 12:48:21 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
ack, err := src.Send(rudp.Pkt{Data: data})
|
2021-01-16 12:48:21 -08:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
<-ack
|
|
|
|
} else {
|
|
|
|
// Client supplied wrong password
|
2021-01-30 03:24:37 -08:00
|
|
|
log.Print("User " + src.Username() + " at " + src.Addr().String() + " supplied wrong password for sudo mode")
|
2021-01-16 12:48:21 -08:00
|
|
|
|
|
|
|
// Send DENY_SUDO_MODE
|
2021-01-19 11:37:35 -08:00
|
|
|
data := []byte{0, ToClientDenySudoMode}
|
2021-01-16 12:48:21 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
ack, err := src.Send(rudp.Pkt{Data: data})
|
2021-01-16 12:48:21 -08:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
<-ack
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
2021-01-24 05:18:29 -08:00
|
|
|
case ToServerModChannelJoin:
|
|
|
|
ch := string(pkt.Data[4:])
|
|
|
|
if ch == rpcCh {
|
|
|
|
data := make([]byte, 5+len(rpcCh))
|
|
|
|
data[0] = uint8(0x00)
|
|
|
|
data[1] = uint8(ToClientModChannelSignal)
|
|
|
|
data[2] = uint8(ModChSigJoinFail)
|
|
|
|
binary.BigEndian.PutUint16(data[3:5], uint16(len(rpcCh)))
|
|
|
|
copy(data[5:], []byte(rpcCh))
|
|
|
|
|
|
|
|
ack, err := src.Send(rudp.Pkt{Data: data})
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
}
|
|
|
|
<-ack
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
src.modChs[ch] = true
|
|
|
|
return false
|
|
|
|
case ToServerModChannelLeave:
|
|
|
|
ch := string(pkt.Data[4:])
|
|
|
|
if ch == rpcCh {
|
|
|
|
data := make([]byte, 5+len(rpcCh))
|
|
|
|
data[0] = uint8(0x00)
|
|
|
|
data[1] = uint8(ToClientModChannelSignal)
|
|
|
|
data[2] = uint8(ModChSigLeaveFail)
|
|
|
|
binary.BigEndian.PutUint16(data[3:5], uint16(len(rpcCh)))
|
|
|
|
copy(data[5:], []byte(rpcCh))
|
|
|
|
|
|
|
|
ack, err := src.Send(rudp.Pkt{Data: data})
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
}
|
|
|
|
<-ack
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
src.modChs[ch] = false
|
|
|
|
return false
|
|
|
|
case ToServerModChannelMsg:
|
|
|
|
chlen := binary.BigEndian.Uint16(pkt.Data[2:4])
|
2021-01-24 06:01:17 -08:00
|
|
|
ch := string(pkt.Data[4 : 4+chlen])
|
2021-01-24 05:18:29 -08:00
|
|
|
if ch == rpcCh {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2021-01-14 04:40:31 -08:00
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2021-01-16 12:48:21 -08:00
|
|
|
return false
|
2021-01-14 04:40:31 -08:00
|
|
|
}
|