mirror of
https://github.com/Mezeporta/Erupe.git
synced 2026-03-27 10:03:06 +01:00
chore(merge): merge main into develop
Backports two softlock fixes and playtime regression fix from main. Also removes handleMsgMhfEnterTournamentQuest from the nil-stub test list — it's a real DB-backed handler and has its own dedicated test.
This commit is contained in:
@@ -7,6 +7,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
<<<<<<< HEAD
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- French (`fr`) and Spanish (`es`) server language translations. Set `"Language": "fr"` or `"Language": "es"` in `config.json` to activate.
|
- French (`fr`) and Spanish (`es`) server language translations. Set `"Language": "fr"` or `"Language": "es"` in `config.json` to activate.
|
||||||
@@ -30,6 +31,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixed backup recovery panic: `recoverFromBackups` now rejects decompressed backup data smaller than the minimum save layout size, preventing a slice-bounds panic when nullcomp passes through garbage bytes as "already decompressed" data ([#182](https://github.com/Mezeporta/Erupe/pull/182)).
|
- Fixed backup recovery panic: `recoverFromBackups` now rejects decompressed backup data smaller than the minimum save layout size, preventing a slice-bounds panic when nullcomp passes through garbage bytes as "already decompressed" data ([#182](https://github.com/Mezeporta/Erupe/pull/182)).
|
||||||
|
- Fixed playtime regression across sessions: `updateSaveDataWithStruct` now writes the accumulated playtime back into the binary save blob, preventing each reconnect from loading a stale in-game counter and rolling back progress.
|
||||||
|
- Fixed player softlock when buying items at the forge: `MSG_CA_EXCHANGE_ITEM` `Parse()` was returning `NOT IMPLEMENTED`, causing the dispatch loop to drop the packet without sending an ACK. Now parses the `AckHandle` and responds with `doAckBufFail` so the client's error branch exits cleanly.
|
||||||
|
- Fixed player softlock on N-points (Hunting Road) interactions: same root cause for `MSG_MHF_USE_UD_SHOP_COIN` — `Parse()` now reads the `AckHandle` and responds with `doAckBufFail`.
|
||||||
|
|
||||||
## [9.3.1] - 2026-03-23
|
## [9.3.1] - 2026-03-23
|
||||||
|
|
||||||
|
|||||||
@@ -1594,7 +1594,6 @@ func TestBatchParseNotImplemented(t *testing.T) {
|
|||||||
&MsgSysDispObject{}, &MsgSysHideObject{},
|
&MsgSysDispObject{}, &MsgSysHideObject{},
|
||||||
&MsgMhfServerCommand{}, &MsgMhfSetLoginwindow{}, &MsgMhfShutClient{},
|
&MsgMhfServerCommand{}, &MsgMhfSetLoginwindow{}, &MsgMhfShutClient{},
|
||||||
&MsgMhfUpdateGuildcard{},
|
&MsgMhfUpdateGuildcard{},
|
||||||
&MsgCaExchangeItem{},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, pkt := range packets {
|
for _, pkt := range packets {
|
||||||
|
|||||||
@@ -9,7 +9,9 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// MsgCaExchangeItem represents the MSG_CA_EXCHANGE_ITEM
|
// MsgCaExchangeItem represents the MSG_CA_EXCHANGE_ITEM
|
||||||
type MsgCaExchangeItem struct{}
|
type MsgCaExchangeItem struct {
|
||||||
|
AckHandle uint32 // TODO: complete reverse-engineering of request fields
|
||||||
|
}
|
||||||
|
|
||||||
// Opcode returns the ID associated with this packet type.
|
// Opcode returns the ID associated with this packet type.
|
||||||
func (m *MsgCaExchangeItem) Opcode() network.PacketID {
|
func (m *MsgCaExchangeItem) Opcode() network.PacketID {
|
||||||
@@ -18,7 +20,8 @@ func (m *MsgCaExchangeItem) Opcode() network.PacketID {
|
|||||||
|
|
||||||
// Parse parses the packet from binary
|
// Parse parses the packet from binary
|
||||||
func (m *MsgCaExchangeItem) Parse(bf *byteframe.ByteFrame, ctx *clientctx.ClientContext) error {
|
func (m *MsgCaExchangeItem) Parse(bf *byteframe.ByteFrame, ctx *clientctx.ClientContext) error {
|
||||||
return errors.New("NOT IMPLEMENTED")
|
m.AckHandle = bf.ReadUint32()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build builds a binary packet from the current data.
|
// Build builds a binary packet from the current data.
|
||||||
|
|||||||
@@ -497,6 +497,8 @@ func TestAckHandlePacketsParse(t *testing.T) {
|
|||||||
{"MsgMhfGetKijuInfo", network.MSG_MHF_GET_KIJU_INFO},
|
{"MsgMhfGetKijuInfo", network.MSG_MHF_GET_KIJU_INFO},
|
||||||
{"MsgMhfGetExtraInfo", network.MSG_MHF_GET_EXTRA_INFO},
|
{"MsgMhfGetExtraInfo", network.MSG_MHF_GET_EXTRA_INFO},
|
||||||
{"MsgMhfGetCogInfo", network.MSG_MHF_GET_COG_INFO},
|
{"MsgMhfGetCogInfo", network.MSG_MHF_GET_COG_INFO},
|
||||||
|
{"MsgCaExchangeItem", network.MSG_CA_EXCHANGE_ITEM},
|
||||||
|
{"MsgMhfUseUdShopCoin", network.MSG_MHF_USE_UD_SHOP_COIN},
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx := &clientctx.ClientContext{RealClientMode: cfg.ZZ}
|
ctx := &clientctx.ClientContext{RealClientMode: cfg.ZZ}
|
||||||
|
|||||||
@@ -9,7 +9,9 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// MsgMhfUseUdShopCoin represents the MSG_MHF_USE_UD_SHOP_COIN
|
// MsgMhfUseUdShopCoin represents the MSG_MHF_USE_UD_SHOP_COIN
|
||||||
type MsgMhfUseUdShopCoin struct{}
|
type MsgMhfUseUdShopCoin struct {
|
||||||
|
AckHandle uint32 // TODO: complete reverse-engineering of request fields
|
||||||
|
}
|
||||||
|
|
||||||
// Opcode returns the ID associated with this packet type.
|
// Opcode returns the ID associated with this packet type.
|
||||||
func (m *MsgMhfUseUdShopCoin) Opcode() network.PacketID {
|
func (m *MsgMhfUseUdShopCoin) Opcode() network.PacketID {
|
||||||
@@ -18,7 +20,8 @@ func (m *MsgMhfUseUdShopCoin) Opcode() network.PacketID {
|
|||||||
|
|
||||||
// Parse parses the packet from binary
|
// Parse parses the packet from binary
|
||||||
func (m *MsgMhfUseUdShopCoin) Parse(bf *byteframe.ByteFrame, ctx *clientctx.ClientContext) error {
|
func (m *MsgMhfUseUdShopCoin) Parse(bf *byteframe.ByteFrame, ctx *clientctx.ClientContext) error {
|
||||||
return errors.New("NOT IMPLEMENTED")
|
m.AckHandle = bf.ReadUint32()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build builds a binary packet from the current data.
|
// Build builds a binary packet from the current data.
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ func TestParseSmallNotImplemented(t *testing.T) {
|
|||||||
{"MsgMhfSetUdTacticsFollower", &MsgMhfSetUdTacticsFollower{}},
|
{"MsgMhfSetUdTacticsFollower", &MsgMhfSetUdTacticsFollower{}},
|
||||||
{"MsgMhfStampcardPrize", &MsgMhfStampcardPrize{}},
|
{"MsgMhfStampcardPrize", &MsgMhfStampcardPrize{}},
|
||||||
{"MsgMhfUpdateForceGuildRank", &MsgMhfUpdateForceGuildRank{}},
|
{"MsgMhfUpdateForceGuildRank", &MsgMhfUpdateForceGuildRank{}},
|
||||||
{"MsgMhfUseUdShopCoin", &MsgMhfUseUdShopCoin{}},
|
|
||||||
|
|
||||||
// SYS packets - NOT IMPLEMENTED
|
// SYS packets - NOT IMPLEMENTED
|
||||||
{"MsgSysAuthData", &MsgSysAuthData{}},
|
{"MsgSysAuthData", &MsgSysAuthData{}},
|
||||||
|
|||||||
@@ -243,7 +243,13 @@ func handleMsgMhfGetUdShopCoin(s *Session, p mhfpacket.MHFPacket) {
|
|||||||
doAckSimpleSucceed(s, pkt.AckHandle, bf.Data())
|
doAckSimpleSucceed(s, pkt.AckHandle, bf.Data())
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMsgMhfUseUdShopCoin(s *Session, p mhfpacket.MHFPacket) {} // stub: unimplemented
|
func handleMsgMhfUseUdShopCoin(s *Session, p mhfpacket.MHFPacket) {
|
||||||
|
pkt := p.(*mhfpacket.MsgMhfUseUdShopCoin)
|
||||||
|
// TODO: full response format is not yet reverse-engineered.
|
||||||
|
// doAckBufFail sends a well-formed buf-type ACK with error code 1.
|
||||||
|
// The client's fail branch exits cleanly without reading response fields.
|
||||||
|
doAckBufFail(s, pkt.AckHandle, nil)
|
||||||
|
}
|
||||||
|
|
||||||
func handleMsgMhfGetEnhancedMinidata(s *Session, p mhfpacket.MHFPacket) {
|
func handleMsgMhfGetEnhancedMinidata(s *Session, p mhfpacket.MHFPacket) {
|
||||||
pkt := p.(*mhfpacket.MsgMhfGetEnhancedMinidata)
|
pkt := p.(*mhfpacket.MsgMhfGetEnhancedMinidata)
|
||||||
|
|||||||
@@ -526,13 +526,8 @@ func TestHandleMsgMhfUseUdShopCoin(t *testing.T) {
|
|||||||
server := createMockServer()
|
server := createMockServer()
|
||||||
session := createMockSession(1, server)
|
session := createMockSession(1, server)
|
||||||
|
|
||||||
defer func() {
|
pkt := &mhfpacket.MsgMhfUseUdShopCoin{AckHandle: 1}
|
||||||
if r := recover(); r != nil {
|
handleMsgMhfUseUdShopCoin(session, pkt)
|
||||||
t.Errorf("handleMsgMhfUseUdShopCoin panicked: %v", r)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
handleMsgMhfUseUdShopCoin(session, nil)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestHandleMsgMhfGetLobbyCrowd(t *testing.T) {
|
func TestHandleMsgMhfGetLobbyCrowd(t *testing.T) {
|
||||||
@@ -1028,7 +1023,6 @@ func TestEmptyHandlers_MiscFiles_Misc(t *testing.T) {
|
|||||||
fn func()
|
fn func()
|
||||||
}{
|
}{
|
||||||
|
|
||||||
{"handleMsgMhfUseUdShopCoin", func() { handleMsgMhfUseUdShopCoin(session, nil) }},
|
|
||||||
{"handleMsgMhfGetDailyMissionMaster", func() { handleMsgMhfGetDailyMissionMaster(session, nil) }},
|
{"handleMsgMhfGetDailyMissionMaster", func() { handleMsgMhfGetDailyMissionMaster(session, nil) }},
|
||||||
{"handleMsgMhfGetDailyMissionPersonal", func() { handleMsgMhfGetDailyMissionPersonal(session, nil) }},
|
{"handleMsgMhfGetDailyMissionPersonal", func() { handleMsgMhfGetDailyMissionPersonal(session, nil) }},
|
||||||
{"handleMsgMhfSetDailyMissionPersonal", func() { handleMsgMhfSetDailyMissionPersonal(session, nil) }},
|
{"handleMsgMhfSetDailyMissionPersonal", func() { handleMsgMhfSetDailyMissionPersonal(session, nil) }},
|
||||||
|
|||||||
@@ -752,7 +752,13 @@ func handleMsgMhfTransitMessage(s *Session, p mhfpacket.MHFPacket) {
|
|||||||
doAckBufSucceed(s, pkt.AckHandle, resp.Data())
|
doAckBufSucceed(s, pkt.AckHandle, resp.Data())
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMsgCaExchangeItem(s *Session, p mhfpacket.MHFPacket) {} // stub: unimplemented
|
func handleMsgCaExchangeItem(s *Session, p mhfpacket.MHFPacket) {
|
||||||
|
pkt := p.(*mhfpacket.MsgCaExchangeItem)
|
||||||
|
// TODO: full response format is not yet reverse-engineered.
|
||||||
|
// doAckBufFail sends a well-formed buf-type ACK with error code 1.
|
||||||
|
// The client's fail branch exits cleanly without reading response fields.
|
||||||
|
doAckBufFail(s, pkt.AckHandle, nil)
|
||||||
|
}
|
||||||
|
|
||||||
func handleMsgMhfServerCommand(s *Session, p mhfpacket.MHFPacket) {} // stub: unimplemented
|
func handleMsgMhfServerCommand(s *Session, p mhfpacket.MHFPacket) {} // stub: unimplemented
|
||||||
|
|
||||||
|
|||||||
@@ -559,13 +559,8 @@ func TestHandleMsgCaExchangeItem(t *testing.T) {
|
|||||||
server := createMockServer()
|
server := createMockServer()
|
||||||
session := createMockSession(1, server)
|
session := createMockSession(1, server)
|
||||||
|
|
||||||
defer func() {
|
pkt := &mhfpacket.MsgCaExchangeItem{AckHandle: 1}
|
||||||
if r := recover(); r != nil {
|
handleMsgCaExchangeItem(session, pkt)
|
||||||
t.Errorf("handleMsgCaExchangeItem panicked: %v", r)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
handleMsgCaExchangeItem(session, nil)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestHandleMsgMhfServerCommand(t *testing.T) {
|
func TestHandleMsgMhfServerCommand(t *testing.T) {
|
||||||
@@ -1068,7 +1063,6 @@ func TestEmptyHandlers_HandlersGo(t *testing.T) {
|
|||||||
{"handleMsgSysUpdateRight", func() { handleMsgSysUpdateRight(session, nil) }},
|
{"handleMsgSysUpdateRight", func() { handleMsgSysUpdateRight(session, nil) }},
|
||||||
{"handleMsgSysAuthQuery", func() { handleMsgSysAuthQuery(session, nil) }},
|
{"handleMsgSysAuthQuery", func() { handleMsgSysAuthQuery(session, nil) }},
|
||||||
{"handleMsgSysAuthTerminal", func() { handleMsgSysAuthTerminal(session, nil) }},
|
{"handleMsgSysAuthTerminal", func() { handleMsgSysAuthTerminal(session, nil) }},
|
||||||
{"handleMsgCaExchangeItem", func() { handleMsgCaExchangeItem(session, nil) }},
|
|
||||||
{"handleMsgMhfServerCommand", func() { handleMsgMhfServerCommand(session, nil) }},
|
{"handleMsgMhfServerCommand", func() { handleMsgMhfServerCommand(session, nil) }},
|
||||||
{"handleMsgMhfSetLoginwindow", func() { handleMsgMhfSetLoginwindow(session, nil) }},
|
{"handleMsgMhfSetLoginwindow", func() { handleMsgMhfSetLoginwindow(session, nil) }},
|
||||||
{"handleMsgSysTransBinary", func() { handleMsgSysTransBinary(session, nil) }},
|
{"handleMsgSysTransBinary", func() { handleMsgSysTransBinary(session, nil) }},
|
||||||
@@ -1106,7 +1100,6 @@ func TestEmptyHandlers_Concurrent(t *testing.T) {
|
|||||||
handleMsgSysUpdateRight,
|
handleMsgSysUpdateRight,
|
||||||
handleMsgSysAuthQuery,
|
handleMsgSysAuthQuery,
|
||||||
handleMsgSysAuthTerminal,
|
handleMsgSysAuthTerminal,
|
||||||
handleMsgCaExchangeItem,
|
|
||||||
handleMsgMhfServerCommand,
|
handleMsgMhfServerCommand,
|
||||||
handleMsgMhfSetLoginwindow,
|
handleMsgMhfSetLoginwindow,
|
||||||
handleMsgSysTransBinary,
|
handleMsgSysTransBinary,
|
||||||
|
|||||||
@@ -281,7 +281,6 @@ func TestEmptyHandlers_NoDb(t *testing.T) {
|
|||||||
{"handleMsgSysUpdateRight", handleMsgSysUpdateRight},
|
{"handleMsgSysUpdateRight", handleMsgSysUpdateRight},
|
||||||
{"handleMsgSysAuthQuery", handleMsgSysAuthQuery},
|
{"handleMsgSysAuthQuery", handleMsgSysAuthQuery},
|
||||||
{"handleMsgSysAuthTerminal", handleMsgSysAuthTerminal},
|
{"handleMsgSysAuthTerminal", handleMsgSysAuthTerminal},
|
||||||
{"handleMsgCaExchangeItem", handleMsgCaExchangeItem},
|
|
||||||
{"handleMsgMhfServerCommand", handleMsgMhfServerCommand},
|
{"handleMsgMhfServerCommand", handleMsgMhfServerCommand},
|
||||||
{"handleMsgMhfSetLoginwindow", handleMsgMhfSetLoginwindow},
|
{"handleMsgMhfSetLoginwindow", handleMsgMhfSetLoginwindow},
|
||||||
{"handleMsgSysTransBinary", handleMsgSysTransBinary},
|
{"handleMsgSysTransBinary", handleMsgSysTransBinary},
|
||||||
@@ -296,7 +295,6 @@ func TestEmptyHandlers_NoDb(t *testing.T) {
|
|||||||
{"handleMsgMhfKickExportForce", handleMsgMhfKickExportForce},
|
{"handleMsgMhfKickExportForce", handleMsgMhfKickExportForce},
|
||||||
{"handleMsgSysSetStatus", handleMsgSysSetStatus},
|
{"handleMsgSysSetStatus", handleMsgSysSetStatus},
|
||||||
{"handleMsgSysEcho", handleMsgSysEcho},
|
{"handleMsgSysEcho", handleMsgSysEcho},
|
||||||
{"handleMsgMhfUseUdShopCoin", handleMsgMhfUseUdShopCoin},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
|
|||||||
@@ -148,6 +148,11 @@ func (save *CharacterSaveData) updateSaveDataWithStruct() {
|
|||||||
if save.Mode >= cfg.F4 {
|
if save.Mode >= cfg.F4 {
|
||||||
copy(save.decompSave[save.Pointers[pRP]:save.Pointers[pRP]+saveFieldRP], rpBytes)
|
copy(save.decompSave[save.Pointers[pRP]:save.Pointers[pRP]+saveFieldRP], rpBytes)
|
||||||
}
|
}
|
||||||
|
if save.Mode >= cfg.S6 {
|
||||||
|
playtimeBytes := make([]byte, 4)
|
||||||
|
binary.LittleEndian.PutUint32(playtimeBytes, save.Playtime)
|
||||||
|
copy(save.decompSave[save.Pointers[pPlaytime]:save.Pointers[pPlaytime]+saveFieldPlaytime], playtimeBytes)
|
||||||
|
}
|
||||||
if save.Mode >= cfg.G10 {
|
if save.Mode >= cfg.G10 {
|
||||||
copy(save.decompSave[save.Pointers[pKQF]:save.Pointers[pKQF]+saveFieldKQF], save.KQF)
|
copy(save.decompSave[save.Pointers[pKQF]:save.Pointers[pKQF]+saveFieldKQF], save.KQF)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user