feat: update news

This commit is contained in:
ericprd 2025-03-06 23:55:46 +08:00
parent 2046021e01
commit e3a9b67e28
17 changed files with 249 additions and 21 deletions

1
.gitignore vendored
View File

@ -1,3 +1,4 @@
bin
.env
.DS_Store
/cmd/legalgo/env

View File

@ -6,7 +6,7 @@ import (
type Staff struct {
ID string `gorm:"primaryKey" json:"id"`
Name string `gorm:"default:null;unique" json:"name"`
Name string `gorm:"default:null" json:"name"`
ProfilePicture string `gorm:"default:null" json:"profile_picture"`
Email string `gorm:"unique;not null" json:"email"`
Password string `gorm:"not null" json:"password"`

View File

@ -13,6 +13,7 @@ type News interface {
GetAll() ([]newsdomain.News, error)
GetBySlug(string) (*newsdomain.News, error)
Create(newsdomain.News) error
Update(newsdomain.News) error
Delete(string) error
}

View File

@ -0,0 +1,66 @@
package newsrepository
import (
"fmt"
newsdomain "legalgo-BE-go/internal/domain/news"
"gorm.io/gorm/clause"
)
func (a *accessor) Update(spec newsdomain.News) error {
tx := a.db.Begin()
if err := tx.Clauses(clause.OnConflict{
Columns: []clause.Column{{Name: "id"}},
DoUpdates: clause.AssignmentColumns([]string{
"title",
"content",
"featured_image",
"is_premium",
"slug",
"author_id",
"live_at",
"updated_at",
}),
}).Select(
"title",
"content",
"featured_image",
"is_premium",
"slug",
"author_id",
"live_at",
"updated_at",
).Save(&spec).Error; err != nil {
tx.Rollback()
return fmt.Errorf("failed to update news: %v", err)
}
if len(spec.Tags) < 1 {
if err := tx.Model(&spec).Association("Tags").Clear(); err != nil {
tx.Rollback()
return fmt.Errorf("failed to clear tags: %v", err)
}
}
if len(spec.Categories) < 1 {
if err := tx.Model(&spec).Association("Categories").Clear(); err != nil {
tx.Rollback()
return fmt.Errorf("failed to clear categories: %v", err)
}
}
if err := tx.Model(&spec).Association("Tags").Append(spec.Tags); err != nil {
tx.Rollback()
return fmt.Errorf("failed to add tags: %v", err)
}
if err := tx.Model(&spec).Association("Categories").Append(spec.Categories); err != nil {
tx.Rollback()
return fmt.Errorf("failed to add categories: %v", err)
}
if err := tx.Commit().Error; err != nil {
return fmt.Errorf("failed to commit transaction: %v", err)
}
return nil
}

View File

@ -4,8 +4,8 @@ import (
userdomain "legalgo-BE-go/internal/domain/user"
)
func (ur *UserRepository) CreateUser(spec userdomain.User) error {
if err := ur.DB.Create(&spec).Error; err != nil {
func (ur *accessor) CreateUser(spec userdomain.User) error {
if err := ur.db.Create(&spec).Error; err != nil {
return err
}

View File

@ -7,14 +7,14 @@ import (
"gorm.io/gorm"
)
func (ur *UserRepository) GetUserByEmail(email string) (*userdomain.User, error) {
func (ur *accessor) GetUserByEmail(email string) (*userdomain.User, error) {
var user *userdomain.User
if email == "" {
return nil, errors.New("email is empty")
}
if err := ur.DB.First(&user, "email = ?", email).Error; err != nil {
if err := ur.db.First(&user, "email = ?", email).Error; err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, errors.New("user not found")
}

View File

@ -6,17 +6,17 @@ import (
userdomain "legalgo-BE-go/internal/domain/user"
)
func (ur *UserRepository) GetUserByID(email string) (*userdomain.User, error) {
func (ur *accessor) GetUserByID(id string) (*userdomain.User, error) {
var user userdomain.User
if email == "" {
return nil, errors.New("email is empty")
if id == "" {
return nil, errors.New("id is empty")
}
if err := ur.DB.
if err := ur.db.
Preload("Subscribe").
Preload("Subscribe.SubscribePlan").
First(&user, "email = ?", email).Error; err != nil {
First(&user, "id = ?", id).Error; err != nil {
return nil, err
}

View File

@ -5,14 +5,14 @@ import (
userdomain "legalgo-BE-go/internal/domain/user"
)
func (ur *UserRepository) GetUserProfile(email string) (*userdomain.UserProfile, error) {
func (ur *accessor) GetUserProfile(email string) (*userdomain.UserProfile, error) {
var user *userdomain.User
if email == "" {
return nil, errors.New("email is empty")
}
if err := ur.DB.
if err := ur.db.
Preload("Subscribe").
Preload("Subscribe.SubscribePlan").
First(&user, "email = ?", email).

View File

@ -5,11 +5,11 @@ import (
userdomain "legalgo-BE-go/internal/domain/user"
)
type UserRepository struct {
DB *database.DB
type accessor struct {
db *database.DB
}
type UserIntf interface {
type User interface {
GetUserByEmail(string) (*userdomain.User, error)
GetUserByID(string) (*userdomain.User, error)
GetUserProfile(string) (*userdomain.UserProfile, error)
@ -18,6 +18,6 @@ type UserIntf interface {
func New(
db *database.DB,
) UserIntf {
return &UserRepository{db}
) User {
return &accessor{db}
}

View File

@ -6,5 +6,6 @@ var Module = fx.Module("news", fx.Invoke(
GetAll,
GetBySlug,
Create,
Update,
Delete,
))

View File

@ -0,0 +1,97 @@
package newshttp
import (
"fmt"
authmiddleware "legalgo-BE-go/internal/api/http/middleware/auth"
newsdomain "legalgo-BE-go/internal/domain/news"
authsvc "legalgo-BE-go/internal/services/auth"
newssvc "legalgo-BE-go/internal/services/news"
"legalgo-BE-go/internal/utilities/response"
"legalgo-BE-go/internal/utilities/utils"
"net/http"
"github.com/go-chi/chi/v5"
)
func Update(
router chi.Router,
newsSvc newssvc.News,
authSvc authsvc.Auth,
) {
router.With(authmiddleware.Authorize()).
Put("/news/{news_id}/update", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
newsID := chi.URLParam(r, "news_id")
if newsID == "" {
response.ResponseWithErrorCode(
ctx,
w,
fmt.Errorf("news id is not provided"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"news id is not provided",
)
return
}
destructedToken, err := utils.GetTokenDetail(r)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
staff, err := authSvc.GetStaffProfile(destructedToken.Email)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
var spec newsdomain.NewsUpdate
if err := utils.UnmarshalBody(r, &spec); err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
spec.ID = newsID
if err := newsSvc.Update(staff.ID, spec); err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
response.RespondJsonSuccess(ctx, w, struct {
Message string
}{
Message: "news updated successfully.",
})
})
}

View File

@ -38,3 +38,14 @@ type News struct {
Author Staff `json:"author"`
}
type NewsUpdate struct {
ID string `json:"id"`
Title string `json:"title"`
Content string `json:"content"`
FeaturedImage string `json:"featured_image"`
Tags []string `json:"tags"`
Categories []string `json:"categories"`
IsPremium bool `json:"is_premium"`
LiveAt time.Time `json:"live_at"`
}

View File

@ -11,7 +11,7 @@ import (
type impl struct {
staffRepo staffrepository.Staff
userRepo userrepository.UserIntf
userRepo userrepository.User
subsRepo subscriberepository.SubsIntf
subsPlanRepo subscribeplanrepository.SubsPlanIntf
}
@ -29,7 +29,7 @@ type Auth interface {
func New(
staffRepo staffrepository.Staff,
userRepo userrepository.UserIntf,
userRepo userrepository.User,
subsRepo subscriberepository.SubsIntf,
subsPlanRepo subscribeplanrepository.SubsPlanIntf,
) Auth {

View File

@ -2,14 +2,14 @@ package newssvc
import (
newsdomain "legalgo-BE-go/internal/domain/news"
"strings"
"legalgo-BE-go/internal/utilities/utils"
"time"
"github.com/google/uuid"
)
func (i *impl) Create(spec newsdomain.NewsReq, staffId string) error {
slug := strings.ToLower(strings.ReplaceAll(spec.Title, " ", "-"))
slug := utils.TitleToSlug(spec.Title)
tags, err := i.tagRepo.GetByIDs(spec.Tags)
if err != nil {

View File

@ -4,6 +4,7 @@ import (
categoryrepository "legalgo-BE-go/internal/accessor/category"
newsrepository "legalgo-BE-go/internal/accessor/news"
tagrepository "legalgo-BE-go/internal/accessor/tag"
userrepository "legalgo-BE-go/internal/accessor/user_repository"
newsdomain "legalgo-BE-go/internal/domain/news"
)
@ -11,12 +12,14 @@ type impl struct {
newsRepo newsrepository.News
tagRepo tagrepository.TagAccessor
categoryRepo categoryrepository.Category
userRepo userrepository.User
}
type News interface {
GetAll() ([]newsdomain.News, error)
GetBySlug(string) (*newsdomain.News, error)
Create(newsdomain.NewsReq, string) error
Update(string, newsdomain.NewsUpdate) error
Delete(string) error
}
@ -24,10 +27,12 @@ func New(
newsRepo newsrepository.News,
tagRepo tagrepository.TagAccessor,
categoryRepo categoryrepository.Category,
userRepo userrepository.User,
) News {
return &impl{
newsRepo,
tagRepo,
categoryRepo,
userRepo,
}
}

View File

@ -0,0 +1,39 @@
package newssvc
import (
newsdomain "legalgo-BE-go/internal/domain/news"
timeutils "legalgo-BE-go/internal/utilities/time_utils"
"legalgo-BE-go/internal/utilities/utils"
)
func (i *impl) Update(id string, spec newsdomain.NewsUpdate) error {
tags, err := i.tagRepo.GetByIDs(spec.Tags)
if err != nil {
return err
}
categories, err := i.categoryRepo.GetByIDs(spec.Categories)
if err != nil {
return err
}
newSpec := newsdomain.News{
ID: spec.ID,
AuthorID: id,
UpdatedAt: timeutils.Now(),
Slug: utils.TitleToSlug(spec.Title),
LiveAt: spec.LiveAt,
Title: spec.Title,
IsPremium: spec.IsPremium,
Content: spec.Content,
FeaturedImage: spec.FeaturedImage,
Tags: tags,
Categories: categories,
}
if err := i.newsRepo.Update(newSpec); err != nil {
return err
}
return nil
}

View File

@ -0,0 +1,7 @@
package utils
import "strings"
func TitleToSlug(title string) string {
return strings.ToLower(strings.ReplaceAll(title, " ", "-"))
}