邹方成 a7a0f639e1 feat: 新增取消发货功能并优化任务中心
fix: 修复微信通知字段截断导致的编码错误
feat: 添加有效邀请相关字段和任务中心常量
refactor: 重构一番赏奖品格位逻辑
perf: 优化道具卡列表聚合显示
docs: 更新项目说明文档和API文档
test: 添加字符串截断工具测试
2025-12-23 22:26:07 +08:00

143 lines
6.4 KiB
Go

package strategy
import (
"context"
"crypto/sha256"
"fmt"
"testing"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
"bindbox-game/internal/repository/mysql/dao"
"bindbox-game/internal/repository/mysql/model"
)
func TestIchibanSelectItemBySlot_Deterministic(t *testing.T) {
db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
if err != nil { t.Fatalf("db open err: %v", err) }
if err := db.Exec(`CREATE TABLE activities (
id INTEGER PRIMARY KEY AUTOINCREMENT,
commitment_seed_master BLOB,
commitment_state_version INTEGER
);`).Error; err != nil { t.Fatalf("migrate err: %v", err) }
if err := db.Exec(`CREATE TABLE activity_reward_settings (
id INTEGER PRIMARY KEY AUTOINCREMENT,
created_at DATETIME,
updated_at DATETIME,
issue_id INTEGER NOT NULL,
product_id INTEGER,
name TEXT NOT NULL,
weight INTEGER NOT NULL,
quantity INTEGER NOT NULL,
original_qty INTEGER NOT NULL,
level INTEGER NOT NULL,
sort INTEGER NOT NULL,
is_boss INTEGER NOT NULL,
deleted_at DATETIME
);`).Error; err != nil { t.Fatalf("migrate err: %v", err) }
q := dao.Use(db)
// seed activity commitment
_ = db.Exec(`INSERT INTO activities (id, commitment_seed_master, commitment_state_version) VALUES (9, x'7365656476616c7565', 1);`).Error
// rewards with original qty
r1 := &model.ActivityRewardSettings{IssueID: 1, Name: "A", Weight: 10, Quantity: 10, OriginalQty: 2, Level: 1, Sort: 1}
r2 := &model.ActivityRewardSettings{IssueID: 1, Name: "B", Weight: 20, Quantity: 10, OriginalQty: 3, Level: 2, Sort: 2}
if err := q.ActivityRewardSettings.Create(r1, r2); err != nil { t.Fatalf("create rewards err: %v", err) }
s := NewIchiban(q, q)
ctx := context.Background()
total := int64(r1.OriginalQty + r2.OriginalQty)
got1 := make([]int64, total)
got2 := make([]int64, total)
for i := int64(0); i < total; i++ {
id, _, err := s.SelectItemBySlot(ctx, 9, 1, i)
if err != nil { t.Fatalf("select err: %v", err) }
got1[i] = id
}
for i := int64(0); i < total; i++ {
id, _, err := s.SelectItemBySlot(ctx, 9, 1, i)
if err != nil { t.Fatalf("select err: %v", err) }
got2[i] = id
}
for i := int64(0); i < total; i++ {
if got1[i] != got2[i] { t.Fatalf("non-deterministic mapping at %d: %d != %d", i, got1[i], got2[i]) }
}
}
func TestIchibanSelectItemBySlot_OutOfRange(t *testing.T) {
db, _ := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
_ = db.Exec(`CREATE TABLE activities (id INTEGER PRIMARY KEY AUTOINCREMENT, commitment_seed_master BLOB, commitment_state_version INTEGER);`).Error
_ = db.Exec(`CREATE TABLE activity_reward_settings (id INTEGER PRIMARY KEY AUTOINCREMENT, created_at DATETIME, updated_at DATETIME, issue_id INTEGER NOT NULL, product_id INTEGER, name TEXT NOT NULL, weight INTEGER NOT NULL, quantity INTEGER NOT NULL, original_qty INTEGER NOT NULL, level INTEGER NOT NULL, sort INTEGER NOT NULL, is_boss INTEGER NOT NULL, deleted_at DATETIME);`).Error
q := dao.Use(db)
_ = db.Exec(`INSERT INTO activities (id, commitment_seed_master, commitment_state_version) VALUES (9, x'7365656476616c7565', 1);`).Error
_ = q.ActivityRewardSettings.Create(&model.ActivityRewardSettings{IssueID: 2, Name: "A", OriginalQty: 1, Quantity: 1, Sort: 1})
s := NewIchiban(q, q)
if _, _, err := s.SelectItemBySlot(context.Background(), 9, 2, 5); err == nil { t.Fatalf("expected out of range error") }
}
func TestIchibanGrantReward_Decrement(t *testing.T) {
db, _ := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
_ = db.Exec(`CREATE TABLE activities (id INTEGER PRIMARY KEY AUTOINCREMENT, commitment_seed_master BLOB, commitment_state_version INTEGER);`).Error
_ = db.Exec(`CREATE TABLE activity_reward_settings (id INTEGER PRIMARY KEY AUTOINCREMENT, created_at DATETIME, updated_at DATETIME, issue_id INTEGER NOT NULL, product_id INTEGER, name TEXT NOT NULL, weight INTEGER NOT NULL, quantity INTEGER NOT NULL, original_qty INTEGER NOT NULL, level INTEGER NOT NULL, sort INTEGER NOT NULL, is_boss INTEGER NOT NULL, deleted_at DATETIME);`).Error
q := dao.Use(db)
r := &model.ActivityRewardSettings{IssueID: 3, Name: "A", OriginalQty: 1, Quantity: 2, Sort: 1}
_ = q.ActivityRewardSettings.Create(r)
s := NewIchiban(q, q)
if err := s.GrantReward(context.Background(), 100, r.ID); err != nil { t.Fatalf("grant err: %v", err) }
got, _ := q.ActivityRewardSettings.Where(q.ActivityRewardSettings.ID.Eq(r.ID)).First()
if got.Quantity != 1 { t.Fatalf("quantity not decremented: %d", got.Quantity) }
}
func TestIchibanProofHasSeedHash(t *testing.T) {
// 1. Setup In-Memory DB
db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{})
if err != nil {
t.Fatalf("db open err: %v", err)
}
// 2. Migrate Tables
db.Exec(`CREATE TABLE activities (id INTEGER PRIMARY KEY AUTOINCREMENT, commitment_seed_master BLOB, commitment_state_version INTEGER, deleted_at DATETIME);`)
db.Exec(`CREATE TABLE activity_reward_settings (id INTEGER PRIMARY KEY AUTOINCREMENT, created_at DATETIME, updated_at DATETIME, issue_id INTEGER NOT NULL, product_id INTEGER, name TEXT NOT NULL, weight INTEGER NOT NULL, quantity INTEGER NOT NULL, original_qty INTEGER NOT NULL, level INTEGER NOT NULL, sort INTEGER NOT NULL, is_boss INTEGER NOT NULL, deleted_at DATETIME);`)
q := dao.Use(db)
// 3. Insert Test Data
seedBytes := []byte("testseedvalue")
expectedHash := fmt.Sprintf("%x", sha256.Sum256(seedBytes))
db.Exec("INSERT INTO activities (id, commitment_seed_master, commitment_state_version) VALUES (?, ?, ?)", 100, seedBytes, 1)
r1 := &model.ActivityRewardSettings{IssueID: 10, Name: "A", Weight: 10, Quantity: 10, OriginalQty: 5, Level: 1, Sort: 1}
q.ActivityRewardSettings.Create(r1)
// 4. Test SelectItemBySlot
s := NewIchiban(q, q)
ctx := context.Background()
_, proof, err := s.SelectItemBySlot(ctx, 100, 10, 0)
if err != nil {
t.Fatalf("SelectItemBySlot failed: %v", err)
}
// 5. Verify seed_hash is in proof
if proof == nil {
t.Fatal("proof is nil")
}
val, ok := proof["seed_hash"]
if !ok {
t.Fatal("seed_hash missing from proof")
}
seedHash, ok := val.(string)
if !ok {
t.Fatalf("seed_hash is not a string, got %T", val)
}
if seedHash != expectedHash {
t.Fatalf("seed_hash mismatch. got %s, want %s", seedHash, expectedHash)
}
t.Logf("Success: seed_hash found in proof: %s", seedHash)
}