mirror of
https://github.com/CPunch/gopenfusion.git
synced 2025-10-01 07:20:05 +00:00
Compare commits
7 Commits
0a28dbcc3e
...
main
Author | SHA1 | Date | |
---|---|---|---|
cafca9093c | |||
d84fcd2c93 | |||
1f63f9856e | |||
de3e067b48 | |||
02afe67ac3 | |||
79f68187bf | |||
cd93a058ce |
@@ -43,7 +43,7 @@ type Service struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func RandomPort() (int, error) {
|
func RandomPort() (int, error) {
|
||||||
l, err := net.Listen("tcp", "127.0.0.1:0")
|
l, err := net.Listen("tcp", ":0")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
@@ -32,6 +32,7 @@ func TestMain(m *testing.M) {
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// this is fine since we don't defer anything
|
||||||
os.Exit(m.Run())
|
os.Exit(m.Run())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -18,6 +18,11 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
|
ret := 1
|
||||||
|
defer func() {
|
||||||
|
os.Exit(ret)
|
||||||
|
}()
|
||||||
|
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
psql, err := sqltestutil.StartPostgresContainer(ctx, "15")
|
psql, err := sqltestutil.StartPostgresContainer(ctx, "15")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -35,7 +40,7 @@ func TestMain(m *testing.M) {
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
os.Exit(m.Run())
|
ret = m.Run()
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDBAccount(t *testing.T) {
|
func TestDBAccount(t *testing.T) {
|
||||||
|
@@ -14,6 +14,11 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
|
ret := 1
|
||||||
|
defer func() {
|
||||||
|
os.Exit(ret)
|
||||||
|
}()
|
||||||
|
|
||||||
r, err := miniredis.Run()
|
r, err := miniredis.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
@@ -26,7 +31,7 @@ func TestMain(m *testing.M) {
|
|||||||
}
|
}
|
||||||
defer rh.Close()
|
defer rh.Close()
|
||||||
|
|
||||||
os.Exit(m.Run())
|
ret = m.Run()
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRedisLogin(t *testing.T) {
|
func TestRedisLogin(t *testing.T) {
|
||||||
|
68
internal/testutil/account.go
Normal file
68
internal/testutil/account.go
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
package testutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/CPunch/gopenfusion/cnet/protocol"
|
||||||
|
"github.com/CPunch/gopenfusion/internal/db"
|
||||||
|
"github.com/CPunch/gopenfusion/internal/redis"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
TestCharCreate = protocol.SP_CL2LS_REQ_CHAR_CREATE{
|
||||||
|
PCStyle: protocol.SPCStyle{
|
||||||
|
INameCheck: 1, SzFirstName: "Hector",
|
||||||
|
SzLastName: "Bannonvenom", IGender: 1, IFaceStyle: 1,
|
||||||
|
IHairStyle: 17, IHairColor: 11, ISkinColor: 10, IEyeColor: 2,
|
||||||
|
IHeight: 1, IBody: 0, IClass: 0,
|
||||||
|
},
|
||||||
|
SOn_Item: protocol.SOnItem{
|
||||||
|
IEquipHandID: 0, IEquipUBID: 53, IEquipLBID: 17, IEquipFootID: 58,
|
||||||
|
IEquipHeadID: 0, IEquipFaceID: 0, IEquipBackID: 0,
|
||||||
|
},
|
||||||
|
SOn_Item_Index: protocol.SOnItem_Index{
|
||||||
|
IEquipUBID_index: 15, IEquipLBID_index: 12, IEquipFootID_index: 17,
|
||||||
|
IFaceStyle: 2, IHairStyle: 18,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// creates a new account and player in the database
|
||||||
|
func MakeTestPlayer(db *db.DBHandler, id string, password string) (acc *db.Account, plr *db.Player, err error) {
|
||||||
|
acc, err = db.NewAccount(id, password)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var plrID int
|
||||||
|
plrID, err = db.NewPlayer(acc.AccountID, TestCharCreate.PCStyle.SzFirstName, TestCharCreate.PCStyle.SzLastName, 1)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
charCreate := TestCharCreate
|
||||||
|
charCreate.PCStyle.IPC_UID = int64(plrID)
|
||||||
|
err = db.FinishPlayer(&charCreate, acc.AccountID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
err = db.FinishTutorial(plrID, acc.AccountID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
plr, err = db.GetPlayer(plrID)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func QueueLogin(redisHndlr *redis.RedisHandler, FEKey []byte, plrID, accID int) (int64, error) {
|
||||||
|
key, err := protocol.GenSerialKey()
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return key, redisHndlr.QueueLogin(key, redis.LoginMetadata{
|
||||||
|
FEKey: FEKey,
|
||||||
|
PlayerID: int32(plrID),
|
||||||
|
AccountID: accID,
|
||||||
|
})
|
||||||
|
}
|
50
internal/testutil/dummy.go
Normal file
50
internal/testutil/dummy.go
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
package testutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"github.com/CPunch/gopenfusion/cnet"
|
||||||
|
"github.com/CPunch/gopenfusion/cnet/protocol"
|
||||||
|
"github.com/matryer/is"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DummyPeer struct {
|
||||||
|
Recv chan *cnet.PacketEvent
|
||||||
|
Peer *cnet.Peer
|
||||||
|
is *is.I
|
||||||
|
}
|
||||||
|
|
||||||
|
// MakeDummyPeer creates a new dummy peer and returns it
|
||||||
|
func MakeDummyPeer(ctx context.Context, is *is.I, port int) *DummyPeer {
|
||||||
|
conn, err := net.Dial("tcp", fmt.Sprintf("127.0.0.1:%d", port))
|
||||||
|
is.NoErr(err)
|
||||||
|
|
||||||
|
recv := make(chan *cnet.PacketEvent)
|
||||||
|
peer := cnet.NewPeer(ctx, conn)
|
||||||
|
go func() {
|
||||||
|
peer.Handler(recv)
|
||||||
|
}()
|
||||||
|
|
||||||
|
return &DummyPeer{Recv: recv, Peer: peer, is: is}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendAndRecv sends a packet (sID & out), waits for the expected response (rID) and decodes it into in
|
||||||
|
func (dp *DummyPeer) SendAndRecv(sID, rID uint32, out, in interface{}) {
|
||||||
|
// send out packet
|
||||||
|
err := dp.Peer.Send(sID, out)
|
||||||
|
dp.is.NoErr(err) // peer.Send() should not return an error
|
||||||
|
|
||||||
|
// receive response
|
||||||
|
evnt := <-dp.Recv
|
||||||
|
defer protocol.PutBuffer(evnt.Pkt)
|
||||||
|
|
||||||
|
dp.is.Equal(evnt.PktID, rID) // should receive expected type
|
||||||
|
dp.is.NoErr(protocol.NewPacket(evnt.Pkt).Decode(in)) // packet.Decode() should not return an error
|
||||||
|
}
|
||||||
|
|
||||||
|
// Kill closes the peer's connection
|
||||||
|
func (dp *DummyPeer) Kill() {
|
||||||
|
dp.Peer.Kill()
|
||||||
|
}
|
52
internal/testutil/env.go
Normal file
52
internal/testutil/env.go
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
package testutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/CPunch/gopenfusion/internal/db"
|
||||||
|
"github.com/CPunch/gopenfusion/internal/redis"
|
||||||
|
"github.com/alicebob/miniredis/v2"
|
||||||
|
"github.com/bitcomplete/sqltestutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SetupEnvironment spawns a postgres container and returns a db and redis handler
|
||||||
|
// along with a cleanup function
|
||||||
|
func SetupEnvironment(ctx context.Context) (*db.DBHandler, *redis.RedisHandler, func()) {
|
||||||
|
// spawn postgres container
|
||||||
|
psql, err := sqltestutil.StartPostgresContainer(ctx, "15")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// open db handler
|
||||||
|
testDB, err := db.OpenFromConnectionString("postgres", psql.ConnectionString()+"?sslmode=disable")
|
||||||
|
if err != nil {
|
||||||
|
psql.Shutdown(ctx)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = testDB.Setup(); err != nil {
|
||||||
|
psql.Shutdown(ctx)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// start miniredis
|
||||||
|
r, err := miniredis.Run()
|
||||||
|
if err != nil {
|
||||||
|
psql.Shutdown(ctx)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// open redis handler
|
||||||
|
rh, err := redis.OpenRedis(r.Addr())
|
||||||
|
if err != nil {
|
||||||
|
psql.Shutdown(ctx)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return testDB, rh, func() {
|
||||||
|
psql.Shutdown(ctx)
|
||||||
|
rh.Close()
|
||||||
|
r.Close()
|
||||||
|
}
|
||||||
|
}
|
@@ -1,101 +1,10 @@
|
|||||||
package testutil
|
package testutil
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/CPunch/gopenfusion/cnet"
|
|
||||||
"github.com/CPunch/gopenfusion/cnet/protocol"
|
|
||||||
"github.com/CPunch/gopenfusion/internal/db"
|
|
||||||
"github.com/CPunch/gopenfusion/internal/redis"
|
|
||||||
"github.com/alicebob/miniredis/v2"
|
|
||||||
"github.com/bitcomplete/sqltestutil"
|
|
||||||
"github.com/matryer/is"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type DummyPeer struct {
|
|
||||||
Recv chan *cnet.PacketEvent
|
|
||||||
Peer *cnet.Peer
|
|
||||||
}
|
|
||||||
|
|
||||||
// MakeDummyPeer creates a new dummy peer and returns it
|
|
||||||
func MakeDummyPeer(ctx context.Context, is *is.I, port int) *DummyPeer {
|
|
||||||
conn, err := net.Dial("tcp", fmt.Sprintf("127.0.0.1:%d", port))
|
|
||||||
is.NoErr(err)
|
|
||||||
|
|
||||||
recv := make(chan *cnet.PacketEvent)
|
|
||||||
peer := cnet.NewPeer(ctx, conn)
|
|
||||||
go func() {
|
|
||||||
peer.Handler(recv)
|
|
||||||
}()
|
|
||||||
|
|
||||||
return &DummyPeer{Recv: recv, Peer: peer}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SendAndRecv sends a packet (sID & out), waits for the expected response (rID) and decodes it into in
|
|
||||||
func (dp *DummyPeer) SendAndRecv(is *is.I, sID, rID uint32, out, in interface{}) {
|
|
||||||
// send out packet
|
|
||||||
err := dp.Peer.Send(sID, out)
|
|
||||||
is.NoErr(err) // peer.Send() should not return an error
|
|
||||||
|
|
||||||
// receive response
|
|
||||||
evnt := <-dp.Recv
|
|
||||||
defer protocol.PutBuffer(evnt.Pkt)
|
|
||||||
|
|
||||||
is.Equal(evnt.PktID, rID) // should receive expected type
|
|
||||||
is.NoErr(protocol.NewPacket(evnt.Pkt).Decode(in)) // packet.Decode() should not return an error
|
|
||||||
}
|
|
||||||
|
|
||||||
// Kill closes the peer's connection
|
|
||||||
func (dp *DummyPeer) Kill() {
|
|
||||||
dp.Peer.Kill()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetupEnvironment spawns a postgres container and returns a db and redis handler
|
|
||||||
// along with a cleanup function
|
|
||||||
func SetupEnvironment(ctx context.Context) (*db.DBHandler, *redis.RedisHandler, func()) {
|
|
||||||
// spawn postgres container
|
|
||||||
psql, err := sqltestutil.StartPostgresContainer(ctx, "15")
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// open db handler
|
|
||||||
testDB, err := db.OpenFromConnectionString("postgres", psql.ConnectionString()+"?sslmode=disable")
|
|
||||||
if err != nil {
|
|
||||||
psql.Shutdown(ctx)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = testDB.Setup(); err != nil {
|
|
||||||
psql.Shutdown(ctx)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// start miniredis
|
|
||||||
r, err := miniredis.Run()
|
|
||||||
if err != nil {
|
|
||||||
psql.Shutdown(ctx)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// open redis handler
|
|
||||||
rh, err := redis.OpenRedis(r.Addr())
|
|
||||||
if err != nil {
|
|
||||||
psql.Shutdown(ctx)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return testDB, rh, func() {
|
|
||||||
psql.Shutdown(ctx)
|
|
||||||
rh.Close()
|
|
||||||
r.Close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func SelectWithTimeout(ch <-chan struct{}, timeout time.Duration) bool {
|
func SelectWithTimeout(ch <-chan struct{}, timeout time.Duration) bool {
|
||||||
select {
|
select {
|
||||||
case <-ch:
|
case <-ch:
|
||||||
|
@@ -22,26 +22,17 @@ var (
|
|||||||
rh *redis.RedisHandler
|
rh *redis.RedisHandler
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
/*
|
||||||
testCharCreate = protocol.SP_CL2LS_REQ_CHAR_CREATE{
|
test data was scraped by dumping packets, just adding a println to the LoginService
|
||||||
PCStyle: protocol.SPCStyle{
|
to print the packet data
|
||||||
INameCheck: 1, SzFirstName: "Hector",
|
*/
|
||||||
SzLastName: "Bannonvenom", IGender: 1, IFaceStyle: 1,
|
|
||||||
IHairStyle: 17, IHairColor: 11, ISkinColor: 10, IEyeColor: 2,
|
|
||||||
IHeight: 1, IBody: 0, IClass: 0,
|
|
||||||
},
|
|
||||||
SOn_Item: protocol.SOnItem{
|
|
||||||
IEquipHandID: 0, IEquipUBID: 53, IEquipLBID: 17, IEquipFootID: 58,
|
|
||||||
IEquipHeadID: 0, IEquipFaceID: 0, IEquipBackID: 0,
|
|
||||||
},
|
|
||||||
SOn_Item_Index: protocol.SOnItem_Index{
|
|
||||||
IEquipUBID_index: 15, IEquipLBID_index: 12, IEquipFootID_index: 17,
|
|
||||||
IFaceStyle: 2, IHairStyle: 18,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
|
ret := 1
|
||||||
|
defer func() {
|
||||||
|
os.Exit(ret)
|
||||||
|
}()
|
||||||
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
@@ -70,10 +61,9 @@ func TestMain(m *testing.M) {
|
|||||||
|
|
||||||
// wait for login server to start, then start tests
|
// wait for login server to start, then start tests
|
||||||
<-loginSrv.Service().Started()
|
<-loginSrv.Service().Started()
|
||||||
ret := m.Run()
|
ret = m.Run()
|
||||||
cancel()
|
cancel()
|
||||||
<-loginSrv.Service().Stopped()
|
<-loginSrv.Service().Stopped()
|
||||||
os.Exit(ret)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// This test tries a typical login sequence.
|
// This test tries a typical login sequence.
|
||||||
@@ -87,7 +77,7 @@ func TestLoginSuccSequence(t *testing.T) {
|
|||||||
|
|
||||||
// send login request (this should create an account)
|
// send login request (this should create an account)
|
||||||
var resp protocol.SP_LS2CL_REP_LOGIN_SUCC
|
var resp protocol.SP_LS2CL_REP_LOGIN_SUCC
|
||||||
dummy.SendAndRecv(is, protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_SUCC,
|
dummy.SendAndRecv(protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_SUCC,
|
||||||
protocol.SP_CL2LS_REQ_LOGIN{
|
protocol.SP_CL2LS_REQ_LOGIN{
|
||||||
SzID: "testLoginSequence",
|
SzID: "testLoginSequence",
|
||||||
SzPassword: "test",
|
SzPassword: "test",
|
||||||
@@ -113,7 +103,7 @@ func TestLoginFailSequence(t *testing.T) {
|
|||||||
|
|
||||||
// send login request (this should not create an account)
|
// send login request (this should not create an account)
|
||||||
var resp protocol.SP_LS2CL_REP_LOGIN_FAIL
|
var resp protocol.SP_LS2CL_REP_LOGIN_FAIL
|
||||||
dummy.SendAndRecv(is, protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_FAIL,
|
dummy.SendAndRecv(protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_FAIL,
|
||||||
protocol.SP_CL2LS_REQ_LOGIN{
|
protocol.SP_CL2LS_REQ_LOGIN{
|
||||||
SzID: "",
|
SzID: "",
|
||||||
SzPassword: "",
|
SzPassword: "",
|
||||||
@@ -135,7 +125,7 @@ func TestCharacterSequence(t *testing.T) {
|
|||||||
|
|
||||||
// send login request (this should create an account)
|
// send login request (this should create an account)
|
||||||
var resp protocol.SP_LS2CL_REP_LOGIN_SUCC
|
var resp protocol.SP_LS2CL_REP_LOGIN_SUCC
|
||||||
dummy.SendAndRecv(is, protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_SUCC,
|
dummy.SendAndRecv(protocol.P_CL2LS_REQ_LOGIN, protocol.P_LS2CL_REP_LOGIN_SUCC,
|
||||||
protocol.SP_CL2LS_REQ_LOGIN{
|
protocol.SP_CL2LS_REQ_LOGIN{
|
||||||
SzID: "testCharacterSequence",
|
SzID: "testCharacterSequence",
|
||||||
SzPassword: "test",
|
SzPassword: "test",
|
||||||
@@ -159,28 +149,28 @@ func TestCharacterSequence(t *testing.T) {
|
|||||||
|
|
||||||
// send character name check request
|
// send character name check request
|
||||||
var charResp protocol.SP_LS2CL_REP_SAVE_CHAR_NAME_SUCC
|
var charResp protocol.SP_LS2CL_REP_SAVE_CHAR_NAME_SUCC
|
||||||
dummy.SendAndRecv(is, protocol.P_CL2LS_REQ_SAVE_CHAR_NAME, protocol.P_LS2CL_REP_SAVE_CHAR_NAME_SUCC,
|
dummy.SendAndRecv(protocol.P_CL2LS_REQ_SAVE_CHAR_NAME, protocol.P_LS2CL_REP_SAVE_CHAR_NAME_SUCC,
|
||||||
protocol.SP_CL2LS_REQ_SAVE_CHAR_NAME{
|
protocol.SP_CL2LS_REQ_SAVE_CHAR_NAME{
|
||||||
ISlotNum: 1,
|
ISlotNum: 1,
|
||||||
IGender: 1,
|
IGender: 1,
|
||||||
IFNCode: 260,
|
IFNCode: 260,
|
||||||
ILNCode: 551,
|
ILNCode: 551,
|
||||||
IMNCode: 33,
|
IMNCode: 33,
|
||||||
SzFirstName: testCharCreate.PCStyle.SzFirstName,
|
SzFirstName: testutil.TestCharCreate.PCStyle.SzFirstName,
|
||||||
SzLastName: testCharCreate.PCStyle.SzLastName,
|
SzLastName: testutil.TestCharCreate.PCStyle.SzLastName,
|
||||||
}, &charResp)
|
}, &charResp)
|
||||||
|
|
||||||
// verify response
|
// verify response
|
||||||
is.Equal(charResp.ISlotNum, int8(1)) // should have the same slot number
|
is.Equal(charResp.ISlotNum, int8(1)) // should have the same slot number
|
||||||
is.Equal(charResp.IGender, int8(1)) // should have the same gender
|
is.Equal(charResp.IGender, int8(1)) // should have the same gender
|
||||||
is.Equal(charResp.SzFirstName, testCharCreate.PCStyle.SzFirstName) // should have the same first name
|
is.Equal(charResp.SzFirstName, testutil.TestCharCreate.PCStyle.SzFirstName) // should have the same first name
|
||||||
is.Equal(charResp.SzLastName, testCharCreate.PCStyle.SzLastName) // should have the same last name
|
is.Equal(charResp.SzLastName, testutil.TestCharCreate.PCStyle.SzLastName) // should have the same last name
|
||||||
|
|
||||||
// send character create request
|
// send character create request
|
||||||
charCreate := testCharCreate
|
charCreate := testutil.TestCharCreate
|
||||||
charCreate.PCStyle.IPC_UID = charResp.IPC_UID
|
charCreate.PCStyle.IPC_UID = charResp.IPC_UID
|
||||||
var charCreateResp protocol.SP_LS2CL_REP_CHAR_CREATE_SUCC
|
var charCreateResp protocol.SP_LS2CL_REP_CHAR_CREATE_SUCC
|
||||||
dummy.SendAndRecv(is, protocol.P_CL2LS_REQ_CHAR_CREATE, protocol.P_LS2CL_REP_CHAR_CREATE_SUCC,
|
dummy.SendAndRecv(protocol.P_CL2LS_REQ_CHAR_CREATE, protocol.P_LS2CL_REP_CHAR_CREATE_SUCC,
|
||||||
charCreate, &charCreateResp)
|
charCreate, &charCreateResp)
|
||||||
|
|
||||||
// verify response
|
// verify response
|
||||||
|
@@ -18,7 +18,8 @@ func (server *ShardServer) LoadNPCs() {
|
|||||||
|
|
||||||
data, err := os.ReadFile(config.GetTDataPath() + "/NPCs.json")
|
data, err := os.ReadFile(config.GetTDataPath() + "/NPCs.json")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
log.Printf("Warning: failed to load NPCs: %v", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// yes, we have to do it this way so our NPCs IDs will be incremented and unique
|
// yes, we have to do it this way so our NPCs IDs will be incremented and unique
|
||||||
|
Reference in New Issue
Block a user