Add registry receiving function
This commit is contained in:
@ -10,6 +10,7 @@ import (
|
|||||||
"github.com/Tnze/go-mc/net"
|
"github.com/Tnze/go-mc/net"
|
||||||
pk "github.com/Tnze/go-mc/net/packet"
|
pk "github.com/Tnze/go-mc/net/packet"
|
||||||
"github.com/Tnze/go-mc/net/queue"
|
"github.com/Tnze/go-mc/net/queue"
|
||||||
|
"github.com/Tnze/go-mc/registry"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Client is used to access Minecraft server
|
// Client is used to access Minecraft server
|
||||||
@ -20,7 +21,7 @@ type Client struct {
|
|||||||
// These are filled when login process
|
// These are filled when login process
|
||||||
Name string
|
Name string
|
||||||
UUID uuid.UUID
|
UUID uuid.UUID
|
||||||
ConfigData
|
Registries registry.NetworkCodec
|
||||||
|
|
||||||
// Ingame packet handlers
|
// Ingame packet handlers
|
||||||
Events Events
|
Events Events
|
||||||
|
@ -6,6 +6,7 @@ import (
|
|||||||
|
|
||||||
"github.com/Tnze/go-mc/chat"
|
"github.com/Tnze/go-mc/chat"
|
||||||
"github.com/Tnze/go-mc/data/packetid"
|
"github.com/Tnze/go-mc/data/packetid"
|
||||||
|
"github.com/Tnze/go-mc/nbt"
|
||||||
"github.com/Tnze/go-mc/net"
|
"github.com/Tnze/go-mc/net"
|
||||||
pk "github.com/Tnze/go-mc/net/packet"
|
pk "github.com/Tnze/go-mc/net/packet"
|
||||||
"github.com/Tnze/go-mc/registry"
|
"github.com/Tnze/go-mc/registry"
|
||||||
@ -32,10 +33,6 @@ type ResourcePack struct {
|
|||||||
PromptMessage *chat.Message // Optional
|
PromptMessage *chat.Message // Optional
|
||||||
}
|
}
|
||||||
|
|
||||||
type ConfigData struct {
|
|
||||||
Registries registry.NetworkCodec
|
|
||||||
}
|
|
||||||
|
|
||||||
type ConfigErr struct {
|
type ConfigErr struct {
|
||||||
Stage string
|
Stage string
|
||||||
Err error
|
Err error
|
||||||
@ -143,10 +140,50 @@ func (c *Client) joinConfiguration(conn *net.Conn) error {
|
|||||||
// TODO
|
// TODO
|
||||||
|
|
||||||
case packetid.ClientboundConfigRegistryData:
|
case packetid.ClientboundConfigRegistryData:
|
||||||
|
const ErrStage = "registry"
|
||||||
// err := p.Scan(pk.NBT(&c.ConfigData.Registries))
|
// err := p.Scan(pk.NBT(&c.ConfigData.Registries))
|
||||||
// if err != nil {
|
// if err != nil {
|
||||||
// return ConfigErr{"registry data", err}
|
// return ConfigErr{"registry data", err}
|
||||||
// }
|
// }
|
||||||
|
var registryID pk.Identifier
|
||||||
|
var length pk.VarInt
|
||||||
|
|
||||||
|
r := bytes.NewReader(p.Data)
|
||||||
|
_, err := registryID.ReadFrom(r)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = length.ReadFrom(r)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := 0; i < int(length); i++ {
|
||||||
|
var entryId pk.Identifier
|
||||||
|
var hasData pk.Boolean
|
||||||
|
var data nbt.RawMessage
|
||||||
|
_, err = entryId.ReadFrom(r)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = hasData.ReadFrom(r)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
|
||||||
|
if hasData {
|
||||||
|
_, err = pk.NBT(&data).ReadFrom(r)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
err = registry.InsertNBTDataIntoRegistry(&c.Registries, string(registryID), string(entryId), data)
|
||||||
|
if err != nil {
|
||||||
|
return ConfigErr{ErrStage, err}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
case packetid.ClientboundConfigResourcePackPop:
|
case packetid.ClientboundConfigResourcePackPop:
|
||||||
var id pk.Option[pk.UUID, *pk.UUID]
|
var id pk.Option[pk.UUID, *pk.UUID]
|
||||||
|
@ -82,15 +82,15 @@ func (m *Manager) handlePlayerChat(packet pk.Packet) error {
|
|||||||
|
|
||||||
unpackedMsg, err := body.Unpack(&m.SignatureCache)
|
unpackedMsg, err := body.Unpack(&m.SignatureCache)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return InvalidChatPacket
|
return InvalidChatPacket{err}
|
||||||
}
|
}
|
||||||
senderInfo, ok := m.pl.PlayerInfos[uuid.UUID(sender)]
|
senderInfo, ok := m.pl.PlayerInfos[uuid.UUID(sender)]
|
||||||
if !ok {
|
if !ok {
|
||||||
return InvalidChatPacket
|
return InvalidChatPacket{ErrUnknownPlayer}
|
||||||
}
|
}
|
||||||
ct := m.c.Registries.ChatType.FindByID(chatType.ID)
|
ct := m.c.Registries.ChatType.FindByID(chatType.ID)
|
||||||
if ct == nil {
|
if ct == nil {
|
||||||
return InvalidChatPacket
|
return InvalidChatPacket{ErrUnknwonChatType}
|
||||||
}
|
}
|
||||||
|
|
||||||
var message sign.Message
|
var message sign.Message
|
||||||
@ -115,7 +115,7 @@ func (m *Manager) handlePlayerChat(packet pk.Packet) error {
|
|||||||
var validated bool
|
var validated bool
|
||||||
if senderInfo.ChatSession != nil {
|
if senderInfo.ChatSession != nil {
|
||||||
if !senderInfo.ChatSession.VerifyAndUpdate(&message) {
|
if !senderInfo.ChatSession.VerifyAndUpdate(&message) {
|
||||||
return ValidationFailed
|
return ErrValidationFailed
|
||||||
}
|
}
|
||||||
validated = true
|
validated = true
|
||||||
// store signature into signatureCache
|
// store signature into signatureCache
|
||||||
@ -143,7 +143,7 @@ func (m *Manager) handleDisguisedChat(packet pk.Packet) error {
|
|||||||
|
|
||||||
ct := m.c.Registries.ChatType.FindByID(chatType.ID)
|
ct := m.c.Registries.ChatType.FindByID(chatType.ID)
|
||||||
if ct == nil {
|
if ct == nil {
|
||||||
return InvalidChatPacket
|
return InvalidChatPacket{ErrUnknwonChatType}
|
||||||
}
|
}
|
||||||
msg := chatType.Decorate(message, &ct.Chat)
|
msg := chatType.Decorate(message, &ct.Chat)
|
||||||
|
|
||||||
@ -199,7 +199,23 @@ func (m *Manager) SendCommand(command string) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type InvalidChatPacket struct {
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i InvalidChatPacket) Error() string {
|
||||||
|
if i.err == nil {
|
||||||
|
return "invalid chat packet"
|
||||||
|
}
|
||||||
|
return "invalid chat packet: " + i.err.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i InvalidChatPacket) Unwrap() error {
|
||||||
|
return i.err
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
InvalidChatPacket = errors.New("invalid chat packet")
|
ErrUnknownPlayer = errors.New("unknown player")
|
||||||
ValidationFailed error = bot.DisconnectErr(chat.TranslateMsg("multiplayer.disconnect.chat_validation_failed"))
|
ErrUnknwonChatType = errors.New("unknown chat type")
|
||||||
|
ErrValidationFailed error = bot.DisconnectErr(chat.TranslateMsg("multiplayer.disconnect.chat_validation_failed"))
|
||||||
)
|
)
|
||||||
|
@ -41,7 +41,7 @@ func (w *World) onPlayerSpawn(pk.Packet) error {
|
|||||||
|
|
||||||
func (w *World) handleLevelChunkWithLightPacket(packet pk.Packet) error {
|
func (w *World) handleLevelChunkWithLightPacket(packet pk.Packet) error {
|
||||||
var pos level.ChunkPos
|
var pos level.ChunkPos
|
||||||
_, currentDimType := w.c.ConfigData.Registries.DimensionType.Find(w.p.DimensionType)
|
_, currentDimType := w.c.Registries.DimensionType.Find(w.p.DimensionType)
|
||||||
if currentDimType == nil {
|
if currentDimType == nil {
|
||||||
return errors.New("dimension type " + w.p.DimensionType + " not found")
|
return errors.New("dimension type " + w.p.DimensionType + " not found")
|
||||||
}
|
}
|
||||||
|
@ -73,3 +73,13 @@ func (m RawMessage) Unmarshal(v any) error {
|
|||||||
}
|
}
|
||||||
return d.unmarshal(val, m.Type)
|
return d.unmarshal(val, m.Type)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m RawMessage) UnmarshalDisallowUnknownField(v any) error {
|
||||||
|
d := NewDecoder(bytes.NewReader(m.Data))
|
||||||
|
d.DisallowUnknownFields()
|
||||||
|
val := reflect.ValueOf(v)
|
||||||
|
if val.Kind() != reflect.Ptr {
|
||||||
|
return errors.New("nbt: non-pointer passed to UnmarshalNBT")
|
||||||
|
}
|
||||||
|
return d.unmarshal(val, m.Type)
|
||||||
|
}
|
||||||
|
@ -1,17 +1,26 @@
|
|||||||
package registry
|
package registry
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
"github.com/Tnze/go-mc/chat"
|
"github.com/Tnze/go-mc/chat"
|
||||||
"github.com/Tnze/go-mc/nbt"
|
"github.com/Tnze/go-mc/nbt"
|
||||||
)
|
)
|
||||||
|
|
||||||
type NetworkCodec struct {
|
type NetworkCodec struct {
|
||||||
ChatType Registry[ChatType] `nbt:"minecraft:chat_type"`
|
ChatType Registry[ChatType] `registry:"minecraft:chat_type"`
|
||||||
DamageType Registry[DamageType] `nbt:"minecraft:damage_type"`
|
DamageType Registry[DamageType] `registry:"minecraft:damage_type"`
|
||||||
DimensionType Registry[Dimension] `nbt:"minecraft:dimension_type"`
|
DimensionType Registry[Dimension] `registry:"minecraft:dimension_type"`
|
||||||
TrimMaterial Registry[nbt.RawMessage] `nbt:"minecraft:trim_material"`
|
TrimMaterial Registry[nbt.RawMessage] `registry:"minecraft:trim_material"`
|
||||||
TrimPattern Registry[nbt.RawMessage] `nbt:"minecraft:trim_pattern"`
|
TrimPattern Registry[nbt.RawMessage] `registry:"minecraft:trim_pattern"`
|
||||||
WorldGenBiome Registry[nbt.RawMessage] `nbt:"minecraft:worldgen/biome"`
|
WorldGenBiome Registry[nbt.RawMessage] `registry:"minecraft:worldgen/biome"`
|
||||||
|
Wolfvariant Registry[nbt.RawMessage] `registry:"minecraft:wolf_variant"`
|
||||||
|
PaintingVariant Registry[nbt.RawMessage] `registry:"minecraft:painting_variant"`
|
||||||
|
BannerPattern Registry[nbt.RawMessage] `registry:"minecraft:banner_pattern"`
|
||||||
|
Enchantment Registry[nbt.RawMessage] `registry:"minecraft:enchantment"`
|
||||||
|
JukeboxSong Registry[nbt.RawMessage] `registry:"minecraft:jukebox_song"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChatType struct {
|
type ChatType struct {
|
||||||
@ -48,3 +57,42 @@ type Dimension struct {
|
|||||||
MonsterSpawnLightLevel nbt.RawMessage `nbt:"monster_spawn_light_level"` // Tag_Int or {type:"minecraft:uniform", value:{min_inclusive: Tag_Int, max_inclusive: Tag_Int}}
|
MonsterSpawnLightLevel nbt.RawMessage `nbt:"monster_spawn_light_level"` // Tag_Int or {type:"minecraft:uniform", value:{min_inclusive: Tag_Int, max_inclusive: Tag_Int}}
|
||||||
MonsterSpawnBlockLightLimit int32 `nbt:"monster_spawn_block_light_limit"`
|
MonsterSpawnBlockLightLimit int32 `nbt:"monster_spawn_block_light_limit"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InsertNBTDataIntoRegistry insert data (entry, data) into the registry.
|
||||||
|
// The codec should be a pointer of a struct. And the registry should be a field of the codec struct.
|
||||||
|
//
|
||||||
|
// This function is a temporary solution while the registry system isn't implemented well.
|
||||||
|
func InsertNBTDataIntoRegistry(codec any, registry, entry string, data nbt.RawMessage) error {
|
||||||
|
codecVal := reflect.ValueOf(codec)
|
||||||
|
if codecVal.Kind() != reflect.Pointer {
|
||||||
|
return errors.New("codec is not a pointer")
|
||||||
|
}
|
||||||
|
|
||||||
|
codecVal = codecVal.Elem()
|
||||||
|
if codecVal.Kind() != reflect.Struct {
|
||||||
|
return errors.New("codec is not a pointer of struct")
|
||||||
|
}
|
||||||
|
|
||||||
|
codecTyp := codecVal.Type()
|
||||||
|
|
||||||
|
numField := codecVal.NumField()
|
||||||
|
for i := 0; i < numField; i++ {
|
||||||
|
registryID, ok := codecTyp.Field(i).Tag.Lookup("registry")
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if registryID == registry {
|
||||||
|
fieldVal := codecVal.Field(i).Addr()
|
||||||
|
args := []reflect.Value{reflect.ValueOf(entry), reflect.ValueOf(data)}
|
||||||
|
err := fieldVal.MethodByName("InsertNBT").Call(args)[0]
|
||||||
|
if !err.IsNil() {
|
||||||
|
return err.Interface().(error)
|
||||||
|
}
|
||||||
|
if registry == "minecraft:chat_type" {
|
||||||
|
fmt.Println(fieldVal.Interface())
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errors.New("registry " + registry + " not found in the codec")
|
||||||
|
}
|
||||||
|
@ -1,12 +1,16 @@
|
|||||||
package registry
|
package registry
|
||||||
|
|
||||||
|
import "github.com/Tnze/go-mc/nbt"
|
||||||
|
|
||||||
type Registry[E any] struct {
|
type Registry[E any] struct {
|
||||||
Type string `nbt:"type"`
|
Type string `nbt:"type"`
|
||||||
Value []struct {
|
Value []Entry[E] `nbt:"value"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Entry[E any] struct {
|
||||||
Name string `nbt:"name"`
|
Name string `nbt:"name"`
|
||||||
ID int32 `nbt:"id"`
|
ID int32 `nbt:"id"`
|
||||||
Element E `nbt:"element"`
|
Element E `nbt:"element"`
|
||||||
} `nbt:"value"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Registry[E]) Find(name string) (int32, *E) {
|
func (r *Registry[E]) Find(name string) (int32, *E) {
|
||||||
@ -29,3 +33,18 @@ func (r *Registry[E]) FindByID(id int32) *E {
|
|||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *Registry[E]) Insert(name string, data E) {
|
||||||
|
r.Value = append(r.Value, Entry[E]{Name: name, Element: data})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Registry[E]) InsertNBT(name string, data nbt.RawMessage) error {
|
||||||
|
entry := Entry[E]{Name: name, ID: int32(len(r.Value))}
|
||||||
|
if data.Type != 0 {
|
||||||
|
if err := data.UnmarshalDisallowUnknownField(&entry.Element); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
r.Value = append(r.Value, entry)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user