feat: add product func without testing

This commit is contained in:
王性驊 2025-03-25 09:28:05 +00:00
parent 84e608f4bb
commit eafd691dab
4 changed files with 366 additions and 26 deletions

View File

@ -27,6 +27,8 @@ type Base interface {
// List 根據查詢條件取得 Tag 資料列表
// 回傳值分別為資料列表與符合條件的總筆數
List(ctx context.Context, params TagQueryParams) ([]*entity.Tags, int64, error)
// GetByIDs 根據查詢條件取得 Tag 資料列表
GetByIDs(ctx context.Context, ids []string) ([]*entity.Tags, error)
}
// TagQueryParams 為查詢 Tags 時的參數結構
@ -52,6 +54,8 @@ type TagBindingRepo interface {
BindTags(ctx context.Context, binding []*entity.TagsBindingTable) error
// UnbindTag 刪除一筆綁定資料
UnbindTag(ctx context.Context, tagID, referenceID string) error
// UnbindTagByReferenceID 刪除一筆綁定資料
UnbindTagByReferenceID(ctx context.Context, referenceID string) error
// GetBindingsByReference 根據參照 ID 取得所有綁定資料
GetBindingsByReference(ctx context.Context, referenceID string) ([]*entity.TagsBindingTable, error)
// ListTagBinding 根據查詢條件取得 tag binding 的資料列表

View File

@ -172,6 +172,33 @@ func (repo *TagsRepository) List(ctx context.Context, params repository.TagQuery
return tags, count, nil
}
func (repo *TagsRepository) GetByIDs(ctx context.Context, ids []string) ([]*entity.Tags, error) {
// 轉換字串 ID 為 ObjectID
objectIDs := make([]primitive.ObjectID, 0, len(ids))
for _, id := range ids {
oid, err := primitive.ObjectIDFromHex(id)
if err != nil {
continue
}
objectIDs = append(objectIDs, oid)
}
if len(objectIDs) == 0 {
return nil, nil
}
// 構建查詢過濾器
filter := bson.M{"_id": bson.M{"$in": objectIDs}}
// 查詢符合條件的文件
var results []*entity.Tags
err := repo.Tags.GetClient().Find(ctx, &results, filter)
if err != nil {
return nil, err
}
return results, nil
}
func (repo *TagsRepository) IndexTags20250317001UP(ctx context.Context) (*mongo.Cursor, error) {
// 等價於 db.account.createIndex({"create_at": 1})
repo.Tags.PopulateIndex(ctx, "show_type", 1, false)

View File

@ -42,6 +42,16 @@ func (repo *TagsRepository) UnbindTag(ctx context.Context, tagID, referenceID st
return nil
}
func (repo *TagsRepository) UnbindTagByReferenceID(ctx context.Context, referenceID string) error {
filter := bson.M{"reference_id": referenceID}
_, err := repo.TageBinding.GetClient().DeleteMany(ctx, filter)
if err != nil {
return err
}
return nil
}
func (repo *TagsRepository) GetBindingsByReference(ctx context.Context, referenceID string) ([]*entity.TagsBindingTable, error) {
var result []*entity.TagsBindingTable
filter := bson.M{"reference_id": referenceID}

View File

@ -91,6 +91,15 @@ func (use *ProductUseCase) Create(ctx context.Context, product *usecase.Product)
insert.CustomFields = cf
}
// 綁定 Tags
tagsBinding := make([]*entity.TagsBindingTable, 0, len(product.Tags))
for _, tag := range product.Tags {
tagsBinding = append(tagsBinding, &entity.TagsBindingTable{
ReferenceID: insert.ID.Hex(),
TagID: tag,
})
}
// Transaction 設定:只做必要寫入
opts := options.Transaction().SetReadConcern(readconcern.Local())
err := use.ProductRepo.Transaction(ctx, func(sessCtx mongo.SessionContext) (any, error) {
@ -128,24 +137,17 @@ func (use *ProductUseCase) Create(ctx context.Context, product *usecase.Product)
return nil, e
}
// 過濾 Tag
// 綁定 Tags
//for _, tag := range product.Tags {
// if err := use.TagBinding.BindTag(sessCtx, &entity.TagsBindingTable{
// ReferenceID: insert.ID.Hex(),
// TagID: tag,
// }); err != nil {
// _ = errs.DBErrorL(logx.WithContext(ctx),
// []logx.LogField{
// {Key: "ReferenceID", Value: insert.ID.Hex()},
// {Key: "TagID", Value: tag},
// {Key: "func", Value: "TagBinding.BindTag"},
// {Key: "err", Value: err.Error()},
// }, "")
//
// continue
// }
//}
if err := use.TagBinding.BindTags(sessCtx, tagsBinding); err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "ReferenceID", Value: insert.ID.Hex()},
{Key: "tags", Value: product.Tags},
{Key: "func", Value: "TagBinding.BindTag"},
{Key: "err", Value: err.Error()},
}, "failed to binding product tags")
return nil, e
}
return nil, nil
}, opts)
@ -158,23 +160,320 @@ func (use *ProductUseCase) Create(ctx context.Context, product *usecase.Product)
}
func (use *ProductUseCase) Update(ctx context.Context, id string, product *usecase.Product) error {
//TODO implement me
panic("implement me")
// 資料前處理:準備 entity.Product 實體
update := &repository.ProductUpdateParams{}
if product.Title != nil {
update.Title = product.Title
}
if product.IsPublished != nil {
update.IsPublished = product.IsPublished
}
if product.Category != nil {
update.Category = product.Category
}
if product.ShortTitle != nil {
update.ShortTitle = product.ShortTitle
}
if product.Details != nil {
update.Details = product.Details
}
if product.ShortDescription != nil {
update.ShortDescription = *product.ShortDescription
}
if product.Slug != nil {
update.Slug = product.Slug
}
if product.Amount != 0 {
update.Amount = &product.Amount
}
if product.StartTime != nil {
st := utils.Rfc3339ToUnix(utils.ToValue(product.StartTime))
update.StartTime = &st
}
if product.EndTime != nil {
et := utils.Rfc3339ToUnix(utils.ToValue(product.EndTime))
update.EndTime = &et
}
if len(product.Media) > 0 {
medias := make([]entity.Media, 0, len(product.Media))
for _, m := range product.Media {
medias = append(medias, entity.Media{
Sort: m.Sort,
URL: m.URL,
Type: m.Type,
})
}
update.Media = medias
}
if len(product.CustomFields) > 0 {
cf := make([]entity.CustomFields, 0, len(product.CustomFields))
for _, field := range product.CustomFields {
cf = append(cf, entity.CustomFields{
Key: field.Key,
Value: field.Value,
})
}
update.CustomFields = cf
}
// 綁定 Tags
tagsBinding := make([]*entity.TagsBindingTable, 0, len(product.Tags))
for _, tag := range product.Tags {
tagsBinding = append(tagsBinding, &entity.TagsBindingTable{
ReferenceID: id,
TagID: tag,
})
}
// Transaction 設定:只做必要寫入
opts := options.Transaction().SetReadConcern(readconcern.Local())
err := use.ProductRepo.Transaction(ctx, func(sessCtx mongo.SessionContext) (any, error) {
_, err := use.ProductRepo.Update(sessCtx, id, update)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "req", Value: product},
{Key: "func", Value: "ProductRepo.Update"},
{Key: "err", Value: err.Error()},
},
"failed to update product")
return nil, e
}
if err := use.TagBinding.UnbindTagByReferenceID(sessCtx, id); err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "ReferenceID", Value: id},
{Key: "func", Value: "TagBinding.UnbindTagByReferenceID"},
{Key: "err", Value: err.Error()},
}, "failed to unbind tags")
return nil, e
}
if err := use.TagBinding.BindTags(sessCtx, tagsBinding); err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "ReferenceID", Value: id},
{Key: "tags", Value: product.Tags},
{Key: "func", Value: "TagBinding.BindTags"},
{Key: "err", Value: err.Error()},
}, "failed to binding product tags")
return nil, e
}
return nil, nil
}, opts)
if err != nil {
return err
}
return nil
}
func (use *ProductUseCase) Delete(ctx context.Context, id string) error {
//TODO implement me
panic("implement me")
// Transaction 設定:只做必要寫入
opts := options.Transaction().SetReadConcern(readconcern.Local())
err := use.ProductRepo.Transaction(ctx, func(sessCtx mongo.SessionContext) (any, error) {
err := use.ProductRepo.Delete(sessCtx, id)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "ReferenceID", Value: id},
{Key: "func", Value: "ProductRepo.Delete"},
{Key: "err", Value: err.Error()},
}, "failed to delete product")
return nil, e
}
err = use.TagRepo.UnbindTagByReferenceID(sessCtx, id)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "ReferenceID", Value: id},
{Key: "func", Value: "TagBinding.UnbindTagByReferenceID"},
{Key: "err", Value: err.Error()},
}, "failed to unbind tags")
return nil, e
}
return nil, nil
}, opts)
if err != nil {
return err
}
return nil
}
func (use *ProductUseCase) Get(ctx context.Context, id string) (*usecase.ProductResp, error) {
//TODO implement me
panic("implement me")
product, err := use.ProductRepo.FindOneByID(ctx, id)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "product_id", Value: id},
{Key: "func", Value: "ProductRepo.FindOneByID"},
{Key: "err", Value: err.Error()},
}, "failed to find product")
return nil, e
}
productStatistics, err := use.ProductStatisticsRepo.GetByID(ctx, id)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "product_id", Value: id},
{Key: "func", Value: "ProductStatisticsRepo.GetByID"},
{Key: "err", Value: err.Error()},
}, "failed to get product statistics")
return nil, e
}
tags, err := use.TagRepo.GetBindingsByReference(ctx, id)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "product_id", Value: id},
{Key: "func", Value: "TagRepo.GetBindingsByReference"},
{Key: "err", Value: err.Error()},
}, "failed to get tags")
return nil, e
}
t := make([]string, 0, len(tags))
for _, item := range tags {
t = append(t, item.TagID)
}
tagsInfo, err := use.TagRepo.GetByIDs(ctx, t)
if err != nil {
e := errs.DBErrorL(logx.WithContext(ctx),
[]logx.LogField{
{Key: "product_id", Value: id},
{Key: "func", Value: "TagRepo.GetByIDs"},
{Key: "err", Value: err.Error()},
}, "failed to get tags")
return nil, e
}
// 組合資料
result := &usecase.ProductResp{
ID: product.ID.Hex(),
UID: product.UID,
Title: product.Title,
ShortDescription: product.ShortDescription,
IsPublished: product.IsPublished,
Amount: product.Amount,
Category: product.Category,
Orders: productStatistics.Orders,
AverageRating: productStatistics.AverageRating,
FansCount: productStatistics.FansCount,
UpdatedAt: utils.UnixToRfc3339(product.UpdatedAt),
CreatedAt: utils.UnixToRfc3339(product.CreatedAt),
}
for _, tag := range tagsInfo {
item := usecase.Tags{
ID: tag.ID.Hex(),
Types: tag.Types,
Name: tag.Name,
ShowType: tag.ShowType,
UpdatedAt: utils.UnixToRfc3339(tag.UpdatedAt),
CreatedAt: utils.UnixToRfc3339(tag.CreatedAt),
}
if tag.Cover != nil {
item.Cover = *tag.Cover
}
result.Tags = append(result.Tags, item)
}
if len(product.Media) > 0 {
medias := make([]usecase.Media, 0, len(product.Media))
for _, m := range product.Media {
medias = append(medias, usecase.Media{
Sort: m.Sort,
URL: m.URL,
Type: m.Type,
})
}
result.Media = medias
}
if len(product.CustomFields) > 0 {
cf := make([]usecase.CustomFields, 0, len(product.CustomFields))
for _, field := range product.CustomFields {
cf = append(cf, usecase.CustomFields{
Key: field.Key,
Value: field.Value,
})
}
result.CustomFields = cf
}
if product.Slug != nil {
result.Slug = *product.Slug
}
if product.ShortTitle != nil {
result.ShortTitle = *product.ShortTitle
}
if product.Details != nil {
result.Details = *product.Details
}
if product.StartTime != nil {
result.StartTime = utils.UnixToRfc3339(*product.StartTime)
}
if product.EndTime != nil {
result.EndTime = utils.UnixToRfc3339(*product.EndTime)
}
if productStatistics.OrdersUpdateTime > 0 {
result.OrdersUpdateTime = utils.UnixToRfc3339(productStatistics.OrdersUpdateTime)
}
if productStatistics.FansCountUpdateTime > 0 {
result.FansCountUpdateTime = utils.UnixToRfc3339(productStatistics.FansCountUpdateTime)
}
if productStatistics.AverageRatingUpdateTime > 0 {
result.AverageRatingUpdateTime = utils.UnixToRfc3339(productStatistics.AverageRatingUpdateTime)
}
return result, nil
}
func (use *ProductUseCase) List(ctx context.Context, data usecase.ProductQueryParams) ([]*usecase.ProductResp, int64, error) {
//TODO implement me
panic("implement me")
query := &repository.ProductQueryParams{
PageSize: data.PageSize,
PageIndex: data.PageIndex,
}
if data.Slug != nil {
query.Slug = data.Slug
}
if data.UID != nil {
query.UID = data.UID
}
if data.IsPublished != nil {
query.IsPublished = data.IsPublished
}
if data.Category != nil {
query.Category = data.Category
}
if data.StartTime != nil {
query.StartTime = data.StartTime
}
EndTime * int64 // 結束時間Unix 時間戳)
Slug * string // URL 後綴
product, i, err := use.ProductRepo.ListProduct(ctx, &repository.ProductQueryParams{})
if err != nil {
return nil, 0, err
}
}
func (use *ProductUseCase) IncOrders(ctx context.Context, productID string, count int64) error {