chart_repository.go 6.4 KB
package repository

import (
	"context"
	"github.com/jinzhu/copier"
	"github.com/pkg/errors"
	"github.com/tiptok/gocomm/pkg/cache"
	"gitlab.fjmaimaimai.com/allied-creation/sumifcc-bchart/cmd/chart-server/interanl/pkg/db/models"
	"gitlab.fjmaimaimai.com/allied-creation/sumifcc-bchart/cmd/chart-server/interanl/pkg/db/transaction"
	"gitlab.fjmaimaimai.com/allied-creation/sumifcc-bchart/cmd/chart-server/interanl/pkg/domain"
	"gorm.io/gorm"
)

type ChartRepository struct {
	*cache.CachedRepository
}

func (repository *ChartRepository) Insert(ctx context.Context, conn transaction.Conn, dm *domain.Chart) (*domain.Chart, error) {
	var (
		err error
		m   = &models.Chart{}
		tx  = conn.DB()
	)
	if m, err = repository.DomainModelToModel(dm); err != nil {
		return nil, err
	}
	if tx = tx.Model(m).Save(m); tx.Error != nil {
		return nil, tx.Error
	}
	dm.Id = m.Id
	return repository.ModelToDomainModel(m)

}

func (repository *ChartRepository) Update(ctx context.Context, conn transaction.Conn, dm *domain.Chart) (*domain.Chart, error) {
	var (
		err error
		m   *models.Chart
		tx  = conn.DB()
	)
	if m, err = repository.DomainModelToModel(dm); err != nil {
		return nil, err
	}
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(m).Updates(m)
		return nil, tx.Error
	}
	if _, err = repository.Query(queryFunc, m.CacheKeyFunc()); err != nil {
		return nil, err
	}
	return repository.ModelToDomainModel(m)
}

func (repository *ChartRepository) UpdateWithVersion(ctx context.Context, transaction transaction.Conn, dm *domain.Chart) (*domain.Chart, error) {
	var (
		err error
		m   *models.Chart
		tx  = transaction.DB()
	)
	if m, err = repository.DomainModelToModel(dm); err != nil {
		return nil, err
	}
	oldVersion := dm.Version
	m.Version += 1
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(m).Select("*").Where("id = ?", m.Id).Where("version = ?", oldVersion).Updates(m)
		if tx.RowsAffected == 0 {
			return nil, domain.ErrUpdateFail
		}
		return nil, tx.Error
	}
	if _, err = repository.Query(queryFunc, m.CacheKeyFunc()); err != nil {
		return nil, err
	}
	return repository.ModelToDomainModel(m)
}

func (repository *ChartRepository) Delete(ctx context.Context, conn transaction.Conn, dm *domain.Chart) (*domain.Chart, error) {
	var (
		tx = conn.DB()
		m  = &models.Chart{Id: dm.Identify().(int64)}
	)
	queryFunc := func() (interface{}, error) {
		tx = tx.Where("id = ?", m.Id).Delete(m)
		return m, tx.Error
	}
	if _, err := repository.Query(queryFunc, m.CacheKeyFunc()); err != nil {
		return dm, err
	}
	return repository.ModelToDomainModel(m)
}

func (repository *ChartRepository) FindOne(ctx context.Context, conn transaction.Conn, id int64) (*domain.Chart, error) {
	var (
		err error
		tx  = conn.DB()
		m   = new(models.Chart)
	)
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(m).Where("id = ?", id).First(m)
		if errors.Is(tx.Error, gorm.ErrRecordNotFound) {
			return nil, domain.ErrNotFound
		}
		return m, tx.Error
	}
	cacheModel := new(models.Chart)
	cacheModel.Id = id
	if err = repository.QueryCache(cacheModel.CacheKeyFunc, m, queryFunc); err != nil {
		return nil, err
	}
	return repository.ModelToDomainModel(m)
}

func (repository *ChartRepository) Find(ctx context.Context, conn transaction.Conn, queryOptions map[string]interface{}) (int64, []*domain.Chart, error) {
	var (
		tx    = conn.DB()
		ms    []*models.Chart
		dms   = make([]*domain.Chart, 0)
		total int64
	)
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(&ms).Order("updated_at desc") //.Order("pid asc").Order("sort asc")
		if v, ok := queryOptions["ids"]; ok {
			tx.Where("id in (?)", v)
		}
		if v, ok := queryOptions["tenantId"]; ok {
			tx.Where("tenant_id = ?", v)
		}
		if v, ok := queryOptions["pid"]; ok {
			tx.Where("pid = ?", v)
		}
		if total, tx = transaction.PaginationAndCount(ctx, tx, queryOptions, &ms); tx.Error != nil {
			return dms, tx.Error
		}
		return dms, nil
	}

	if _, err := repository.Query(queryFunc); err != nil {
		return 0, nil, err
	}

	for _, item := range ms {
		if dm, err := repository.ModelToDomainModel(item); err != nil {
			return 0, dms, err
		} else {
			dms = append(dms, dm)
		}
	}
	return total, dms, nil
}

// FindByTypeAndName 按类型、名称搜索图表(判断重复的图表)
func (repository *ChartRepository) FindByTypeAndName(ctx context.Context, conn transaction.Conn, tenantId int64, t string, name string) (int64, []*domain.Chart, error) {
	var (
		tx  = conn.DB()
		ms  []*models.Chart
		dms = make([]*domain.Chart, 0)
		//total int64
	)
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(&ms).Order("id desc") //.Order("pid asc").Order("sort asc")
		tx.Where("tenant_id = ?", tenantId)
		tx.Where("type = ?", t)
		tx.Where("name = ?", name)
		if tx = tx.Find(&ms); tx.Error != nil {
			return dms, tx.Error
		}
		return dms, nil
	}

	if _, err := repository.Query(queryFunc); err != nil {
		return 0, nil, err
	}

	for _, item := range ms {
		if dm, err := repository.ModelToDomainModel(item); err != nil {
			return 0, dms, err
		} else {
			dms = append(dms, dm)
		}
	}
	return int64(len(dms)), dms, nil
}

func (repository *ChartRepository) FindOneByGroup(ctx context.Context, conn transaction.Conn, tenantId, pid int64) (*domain.Chart, error) {
	var (
		err error
		tx  = conn.DB()
		m   = new(models.Chart)
	)
	queryFunc := func() (interface{}, error) {
		tx = tx.Model(m).Where("tenant_id = ?", tenantId).Where("pid = ?", pid).Order("sort desc").Limit(1).First(m)
		if errors.Is(tx.Error, gorm.ErrRecordNotFound) {
			return nil, domain.ErrNotFound
		}
		return m, tx.Error
	}
	if _, err = repository.Query(queryFunc); err != nil {
		return nil, err
	}
	return repository.ModelToDomainModel(m)
}

// FindCompanyIds 获取所有的公司ID
func (repository *ChartRepository) FindCompanyIds(ctx context.Context, conn transaction.Conn) ([]int64, error) {
	var (
		err error
		tx  = conn.DB()
		m   = new(models.Chart)
	)
	list := make([]int64, 0)
	err = tx.Model(&m).Group("tenant_id").Pluck("tenant_id", &list).Error
	return list, err
}

func (repository *ChartRepository) ModelToDomainModel(from *models.Chart) (*domain.Chart, error) {
	to := &domain.Chart{}
	err := copier.Copy(to, from)
	return to, err
}

func (repository *ChartRepository) DomainModelToModel(from *domain.Chart) (*models.Chart, error) {
	to := &models.Chart{}
	err := copier.Copy(to, from)
	return to, err
}

func NewChartRepository(cache *cache.CachedRepository) domain.ChartRepository {
	return &ChartRepository{CachedRepository: cache}
}