All checks were successful
构建并部署到测试环境(无 SSH) / build-and-deploy (push) Successful in 6m33s
**变更说明**: - 删除所有 *_test.go 文件(单元测试、集成测试、验收测试、流程测试) - 删除整个 tests/ 目录 - 更新 CLAUDE.md:用"测试禁令"章节替换所有测试要求 - 删除测试生成 Skill (openspec-generate-acceptance-tests) - 删除测试生成命令 (opsx:gen-tests) - 更新 tasks.md:删除所有测试相关任务 **新规范**: - ❌ 禁止编写任何形式的自动化测试 - ❌ 禁止创建 *_test.go 文件 - ❌ 禁止在任务中包含测试相关工作 - ✅ 仅当用户明确要求时才编写测试 **原因**: 业务系统的正确性通过人工验证和生产环境监控保证,测试代码维护成本高于价值。 Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
136 lines
4.5 KiB
Go
136 lines
4.5 KiB
Go
package postgres
|
||
|
||
import (
|
||
"context"
|
||
|
||
"gorm.io/gorm"
|
||
|
||
"github.com/break/junhong_cmp_fiber/internal/model"
|
||
"github.com/break/junhong_cmp_fiber/internal/store"
|
||
"github.com/break/junhong_cmp_fiber/pkg/constants"
|
||
"github.com/break/junhong_cmp_fiber/pkg/middleware"
|
||
)
|
||
|
||
type PackageStore struct {
|
||
db *gorm.DB
|
||
}
|
||
|
||
func NewPackageStore(db *gorm.DB) *PackageStore {
|
||
return &PackageStore{db: db}
|
||
}
|
||
|
||
func (s *PackageStore) Create(ctx context.Context, pkg *model.Package) error {
|
||
// GORM 对零值字段有特殊处理,先创建然后立即更新 enable_realname_activation 字段确保正确设置
|
||
if err := s.db.WithContext(ctx).Omit("enable_realname_activation").Create(pkg).Error; err != nil {
|
||
return err
|
||
}
|
||
// 明确更新 enable_realname_activation 字段(包括零值 false)
|
||
return s.db.WithContext(ctx).Model(pkg).Update("enable_realname_activation", pkg.EnableRealnameActivation).Error
|
||
}
|
||
|
||
func (s *PackageStore) GetByID(ctx context.Context, id uint) (*model.Package, error) {
|
||
var pkg model.Package
|
||
if err := s.db.WithContext(ctx).First(&pkg, id).Error; err != nil {
|
||
return nil, err
|
||
}
|
||
return &pkg, nil
|
||
}
|
||
|
||
func (s *PackageStore) GetByCode(ctx context.Context, code string) (*model.Package, error) {
|
||
var pkg model.Package
|
||
if err := s.db.WithContext(ctx).Where("package_code = ?", code).First(&pkg).Error; err != nil {
|
||
return nil, err
|
||
}
|
||
return &pkg, nil
|
||
}
|
||
|
||
func (s *PackageStore) Update(ctx context.Context, pkg *model.Package) error {
|
||
return s.db.WithContext(ctx).Save(pkg).Error
|
||
}
|
||
|
||
func (s *PackageStore) Delete(ctx context.Context, id uint) error {
|
||
return s.db.WithContext(ctx).Delete(&model.Package{}, id).Error
|
||
}
|
||
|
||
func (s *PackageStore) List(ctx context.Context, opts *store.QueryOptions, filters map[string]interface{}) ([]*model.Package, int64, error) {
|
||
var packages []*model.Package
|
||
var total int64
|
||
|
||
query := s.db.WithContext(ctx).Model(&model.Package{})
|
||
|
||
// 代理用户额外过滤:只能看到已分配的套餐
|
||
userType := middleware.GetUserTypeFromContext(ctx)
|
||
shopID := middleware.GetShopIDFromContext(ctx)
|
||
if userType == constants.UserTypeAgent && shopID > 0 {
|
||
query = query.Joins("INNER JOIN tb_shop_package_allocation ON tb_shop_package_allocation.package_id = tb_package.id").
|
||
Where("tb_shop_package_allocation.shop_id = ? AND tb_shop_package_allocation.status = ?",
|
||
shopID, constants.StatusEnabled)
|
||
}
|
||
|
||
if packageName, ok := filters["package_name"].(string); ok && packageName != "" {
|
||
query = query.Where("tb_package.package_name LIKE ?", "%"+packageName+"%")
|
||
}
|
||
if seriesID, ok := filters["series_id"].(uint); ok && seriesID > 0 {
|
||
query = query.Where("tb_package.series_id = ?", seriesID)
|
||
}
|
||
if status, ok := filters["status"]; ok {
|
||
query = query.Where("tb_package.status = ?", status)
|
||
}
|
||
if shelfStatus, ok := filters["shelf_status"]; ok {
|
||
query = query.Where("tb_package.shelf_status = ?", shelfStatus)
|
||
}
|
||
if packageType, ok := filters["package_type"].(string); ok && packageType != "" {
|
||
query = query.Where("tb_package.package_type = ?", packageType)
|
||
}
|
||
|
||
if err := query.Count(&total).Error; err != nil {
|
||
return nil, 0, err
|
||
}
|
||
|
||
if opts == nil {
|
||
opts = store.DefaultQueryOptions()
|
||
}
|
||
offset := (opts.Page - 1) * opts.PageSize
|
||
query = query.Offset(offset).Limit(opts.PageSize)
|
||
|
||
if opts.OrderBy != "" {
|
||
query = query.Order(opts.OrderBy)
|
||
}
|
||
|
||
if err := query.Find(&packages).Error; err != nil {
|
||
return nil, 0, err
|
||
}
|
||
|
||
return packages, total, nil
|
||
}
|
||
|
||
func (s *PackageStore) UpdateStatus(ctx context.Context, id uint, status int) error {
|
||
return s.db.WithContext(ctx).Model(&model.Package{}).Where("id = ?", id).Update("status", status).Error
|
||
}
|
||
|
||
func (s *PackageStore) UpdateShelfStatus(ctx context.Context, id uint, shelfStatus int) error {
|
||
return s.db.WithContext(ctx).Model(&model.Package{}).Where("id = ?", id).Update("shelf_status", shelfStatus).Error
|
||
}
|
||
|
||
// GetByIDUnscoped 根据ID获取套餐(包括已删除的记录)
|
||
// 用于关联查询场景,确保已删除的套餐信息仍能被展示
|
||
func (s *PackageStore) GetByIDUnscoped(ctx context.Context, id uint) (*model.Package, error) {
|
||
var pkg model.Package
|
||
if err := s.db.WithContext(ctx).Unscoped().First(&pkg, id).Error; err != nil {
|
||
return nil, err
|
||
}
|
||
return &pkg, nil
|
||
}
|
||
|
||
// GetByIDsUnscoped 批量获取套餐(包括已删除的记录)
|
||
func (s *PackageStore) GetByIDsUnscoped(ctx context.Context, ids []uint) ([]*model.Package, error) {
|
||
if len(ids) == 0 {
|
||
return nil, nil
|
||
}
|
||
var packages []*model.Package
|
||
if err := s.db.WithContext(ctx).Unscoped().Where("id IN ?", ids).Find(&packages).Error; err != nil {
|
||
return nil, err
|
||
}
|
||
return packages, nil
|
||
}
|