You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
133 lines
4.2 KiB
Go
133 lines
4.2 KiB
Go
package storage
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/jackc/pgx/v5"
|
|
"github.com/jackc/pgx/v5/pgxpool"
|
|
|
|
"github.com/denisovdennis/autohero/internal/model"
|
|
)
|
|
|
|
// GearStore handles all gear CRUD operations against PostgreSQL.
|
|
type GearStore struct {
|
|
pool *pgxpool.Pool
|
|
}
|
|
|
|
// NewGearStore creates a new GearStore backed by the given connection pool.
|
|
func NewGearStore(pool *pgxpool.Pool) *GearStore {
|
|
return &GearStore{pool: pool}
|
|
}
|
|
|
|
// CreateItem inserts a new gear item into the database and populates item.ID.
|
|
func (s *GearStore) CreateItem(ctx context.Context, item *model.GearItem) error {
|
|
err := s.pool.QueryRow(ctx, `
|
|
INSERT INTO gear (slot, form_id, name, subtype, rarity, ilvl, base_primary, primary_stat,
|
|
stat_type, speed_modifier, crit_chance, agility_bonus, set_name, special_effect)
|
|
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14)
|
|
RETURNING id
|
|
`,
|
|
string(item.Slot), item.FormID, item.Name, item.Subtype,
|
|
string(item.Rarity), item.Ilvl, item.BasePrimary, item.PrimaryStat,
|
|
item.StatType, item.SpeedModifier, item.CritChance, item.AgilityBonus,
|
|
item.SetName, item.SpecialEffect,
|
|
).Scan(&item.ID)
|
|
if err != nil {
|
|
return fmt.Errorf("create gear item: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetItem loads a single gear item by ID. Returns (nil, nil) if not found.
|
|
func (s *GearStore) GetItem(ctx context.Context, id int64) (*model.GearItem, error) {
|
|
var item model.GearItem
|
|
var slot, rarity string
|
|
err := s.pool.QueryRow(ctx, `
|
|
SELECT id, slot, form_id, name, subtype, rarity, ilvl,
|
|
base_primary, primary_stat, stat_type,
|
|
speed_modifier, crit_chance, agility_bonus,
|
|
set_name, special_effect
|
|
FROM gear WHERE id = $1
|
|
`, id).Scan(
|
|
&item.ID, &slot, &item.FormID, &item.Name, &item.Subtype,
|
|
&rarity, &item.Ilvl,
|
|
&item.BasePrimary, &item.PrimaryStat, &item.StatType,
|
|
&item.SpeedModifier, &item.CritChance, &item.AgilityBonus,
|
|
&item.SetName, &item.SpecialEffect,
|
|
)
|
|
if err != nil {
|
|
if errors.Is(err, pgx.ErrNoRows) {
|
|
return nil, nil
|
|
}
|
|
return nil, fmt.Errorf("get gear item: %w", err)
|
|
}
|
|
item.Slot = model.EquipmentSlot(slot)
|
|
item.Rarity = model.Rarity(rarity)
|
|
return &item, nil
|
|
}
|
|
|
|
// GetHeroGear returns all equipped gear for a hero, keyed by slot.
|
|
func (s *GearStore) GetHeroGear(ctx context.Context, heroID int64) (map[model.EquipmentSlot]*model.GearItem, error) {
|
|
rows, err := s.pool.Query(ctx, `
|
|
SELECT g.id, g.slot, g.form_id, g.name, g.subtype, g.rarity, g.ilvl,
|
|
g.base_primary, g.primary_stat, g.stat_type,
|
|
g.speed_modifier, g.crit_chance, g.agility_bonus,
|
|
g.set_name, g.special_effect
|
|
FROM hero_gear hg
|
|
JOIN gear g ON hg.gear_id = g.id
|
|
WHERE hg.hero_id = $1
|
|
`, heroID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get hero gear: %w", err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
gear := make(map[model.EquipmentSlot]*model.GearItem)
|
|
for rows.Next() {
|
|
var item model.GearItem
|
|
var slot, rarity string
|
|
if err := rows.Scan(
|
|
&item.ID, &slot, &item.FormID, &item.Name, &item.Subtype,
|
|
&rarity, &item.Ilvl,
|
|
&item.BasePrimary, &item.PrimaryStat, &item.StatType,
|
|
&item.SpeedModifier, &item.CritChance, &item.AgilityBonus,
|
|
&item.SetName, &item.SpecialEffect,
|
|
); err != nil {
|
|
return nil, fmt.Errorf("scan gear item: %w", err)
|
|
}
|
|
item.Slot = model.EquipmentSlot(slot)
|
|
item.Rarity = model.Rarity(rarity)
|
|
gear[item.Slot] = &item
|
|
}
|
|
if err := rows.Err(); err != nil {
|
|
return nil, fmt.Errorf("hero gear rows: %w", err)
|
|
}
|
|
return gear, nil
|
|
}
|
|
|
|
// EquipItem equips a gear item into the given slot for a hero (upsert).
|
|
func (s *GearStore) EquipItem(ctx context.Context, heroID int64, slot model.EquipmentSlot, gearID int64) error {
|
|
_, err := s.pool.Exec(ctx, `
|
|
INSERT INTO hero_gear (hero_id, slot, gear_id)
|
|
VALUES ($1, $2, $3)
|
|
ON CONFLICT (hero_id, slot) DO UPDATE SET gear_id = $3
|
|
`, heroID, string(slot), gearID)
|
|
if err != nil {
|
|
return fmt.Errorf("equip gear item: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// UnequipSlot removes the gear from the given slot for a hero.
|
|
func (s *GearStore) UnequipSlot(ctx context.Context, heroID int64, slot model.EquipmentSlot) error {
|
|
_, err := s.pool.Exec(ctx, `
|
|
DELETE FROM hero_gear WHERE hero_id = $1 AND slot = $2
|
|
`, heroID, string(slot))
|
|
if err != nil {
|
|
return fmt.Errorf("unequip gear slot: %w", err)
|
|
}
|
|
return nil
|
|
}
|