Compare commits

..

8 Commits

Author SHA1 Message Date
ericprd
3311b80943 Merge branch 'main' of github.com:ardeman/project-legalgo-go 2025-03-24 22:16:17 +08:00
ericprd
16fef8706a fix: register staff return message only 2025-03-24 22:15:41 +08:00
Ardeman
4fbfa9c2c9 Merge branch 'main' of https://github.com/ardeman/project-legalgo-go
* 'main' of https://github.com/ardeman/project-legalgo-go:
  fix: search news title by query
2025-03-24 18:18:53 +08:00
ericprd
a68780f09e fix: search news title by query 2025-03-24 18:11:08 +08:00
Ardeman
49dc0e4a7b refactor: update GetAll method to return NewsResponse and rename clicked to views 2025-03-24 15:24:52 +08:00
ericprd
09d9c4efb8 fix: error clicked when create new news 2025-03-24 14:02:20 +08:00
ericprd
990026cece fix: error clicked when create new news 2025-03-24 13:35:46 +08:00
ericprd
11051ae89c feat: staff get news by id 2025-03-24 13:32:05 +08:00
11 changed files with 135 additions and 31 deletions

View File

@ -12,9 +12,9 @@ const (
notActive is_active = "false" notActive is_active = "false"
) )
func (a *accessor) GetAll(filter newsdomain.NewsFilter) ([]newsdomain.News, error) { func (a *accessor) GetAll(filter newsdomain.NewsFilter) ([]newsdomain.NewsResponse, error) {
var news []newsdomain.News var news []newsdomain.NewsResponse
query := a.db. query := a.db.Table("news").
Preload("Tags"). Preload("Tags").
Preload("Categories"). Preload("Categories").
Preload("Author"). Preload("Author").
@ -34,8 +34,12 @@ func (a *accessor) GetAll(filter newsdomain.NewsFilter) ([]newsdomain.News, erro
query = query.Where("news.live_at <= ?", timeutils.Now()) query = query.Where("news.live_at <= ?", timeutils.Now())
} }
if filter.Search != "" {
query = query.Where("news.title ILIKE ?", "%"+filter.Search+"%")
}
query. query.
Select("news.*, COUNT(content_logs.content_id) as clicked"). Select("news.*, COUNT(content_logs.content_id) as views").
Group("news.id"). Group("news.id").
Order("news.created_at DESC") Order("news.created_at DESC")

View File

@ -0,0 +1,17 @@
package newsrepository
import newsdomain "legalgo-BE-go/internal/domain/news"
func (a *accessor) GetByID(id string) (*newsdomain.News, error) {
var news newsdomain.News
if err := a.db.
Preload("Tags").
Preload("Categories").
Preload("Author").
First(&news, "id = ?", id).Error; err != nil {
return nil, err
}
return &news, nil
}

View File

@ -10,8 +10,9 @@ type accessor struct {
} }
type News interface { type News interface {
GetAll(filter newsdomain.NewsFilter) ([]newsdomain.News, error) GetAll(filter newsdomain.NewsFilter) ([]newsdomain.NewsResponse, error)
GetBySlug(string) (*newsdomain.News, error) GetBySlug(string) (*newsdomain.News, error)
GetByID(string) (*newsdomain.News, error)
Create(newsdomain.News) error Create(newsdomain.News) error
Update(newsdomain.News) error Update(newsdomain.News) error
Delete(string) error Delete(string) error

View File

@ -15,7 +15,7 @@ func GetAll(
) { ) {
router.Get("/news", func(w http.ResponseWriter, r *http.Request) { router.Get("/news", func(w http.ResponseWriter, r *http.Request) {
var ( var (
news []newsdomain.News news []newsdomain.NewsResponse
err error err error
) )
ctx := r.Context() ctx := r.Context()
@ -24,11 +24,13 @@ func GetAll(
category := query.Get("categories") category := query.Get("categories")
tags := query.Get("tags") tags := query.Get("tags")
activeOnly := query.Get("active") activeOnly := query.Get("active")
search := query.Get("q")
news, err = newsSvc.GetAll(newsdomain.Filter{ news, err = newsSvc.GetAll(newsdomain.Filter{
Category: category, Category: category,
Tags: tags, Tags: tags,
Active: activeOnly, Active: activeOnly,
Search: search,
}) })
if err != nil { if err != nil {

View File

@ -0,0 +1,77 @@
package newshttp
import (
authmiddleware "legalgo-BE-go/internal/api/http/middleware/auth"
logssvc "legalgo-BE-go/internal/services/logs"
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 GetByID(
router chi.Router,
newsSvc newssvc.News,
logSvc logssvc.Log,
) {
router.With(authmiddleware.Authorize()).Get("/staff/news/{id}", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
id := chi.URLParam(r, "id")
token, err := utils.GetToken(r)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
staffDetail, err := utils.DestructToken(token)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
if staffDetail.Role != "staff" {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrUnauthorized.Code,
response.ErrUnauthorized.HttpCode,
"unauthorized",
)
return
}
news, err := newsSvc.GetByID(id)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
response.RespondJsonSuccess(ctx, w, news)
})
}

View File

@ -5,6 +5,7 @@ import "go.uber.org/fx"
var Module = fx.Module("news", fx.Invoke( var Module = fx.Module("news", fx.Invoke(
GetAll, GetAll,
GetBySlug, GetBySlug,
GetByID,
Create, Create,
Update, Update,
Delete, Delete,

View File

@ -3,7 +3,6 @@ package staffhttp
import ( import (
"net/http" "net/http"
responsedomain "legalgo-BE-go/internal/domain/reponse"
staffdomain "legalgo-BE-go/internal/domain/staff" staffdomain "legalgo-BE-go/internal/domain/staff"
staffsvc "legalgo-BE-go/internal/services/staffsvc" staffsvc "legalgo-BE-go/internal/services/staffsvc"
"legalgo-BE-go/internal/utilities/response" "legalgo-BE-go/internal/utilities/response"
@ -49,7 +48,7 @@ func Register(
return return
} }
token, err := authSvc.Register(spec) _, err := authSvc.Register(spec)
if err != nil { if err != nil {
response.ResponseWithErrorCode( response.ResponseWithErrorCode(
ctx, ctx,
@ -62,21 +61,10 @@ func Register(
return return
} }
if err := utils.StoreTokenRedis(ctx, rdb, token, spec.Email); err != nil { response.RespondJsonSuccess(ctx, w, struct {
response.ResponseWithErrorCode( Message string
ctx, }{
w, Message: "staff created successfully.",
err, })
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
responsePayload := &responsedomain.Auth{
Token: token,
}
response.RespondJsonSuccess(ctx, w, responsePayload)
}) })
} }

View File

@ -27,19 +27,23 @@ type News struct {
Title string `json:"title"` Title string `json:"title"`
Content string `json:"content"` Content string `json:"content"`
FeaturedImage string `json:"featured_image"` FeaturedImage string `json:"featured_image"`
Tags []tagdomain.Tag `gorm:"many2many:news_tags" json:"tags"` Tags []tagdomain.Tag `gorm:"many2many:news_tags;foreignKey:ID;joinForeignKey:news_id;References:ID;joinReferences:tag_id" json:"tags"`
Categories []categorydomain.Category `gorm:"many2many:news_categories" json:"categories"` Categories []categorydomain.Category `gorm:"many2many:news_categories;foreignKey:ID;joinForeignKey:news_id;References:ID;joinReferences:category_id" json:"categories"`
IsPremium bool `json:"is_premium"` IsPremium bool `json:"is_premium"`
Slug string `json:"slug"` Slug string `json:"slug"`
AuthorID string `json:"author_id"` AuthorID string `json:"author_id"`
LiveAt time.Time `json:"live_at"` LiveAt time.Time `json:"live_at"`
CreatedAt time.Time `json:"created_at"` CreatedAt time.Time `json:"created_at"`
UpdatedAt time.Time `json:"updated_at"` UpdatedAt time.Time `json:"updated_at"`
Clicked int64 `json:"clicked"`
Author Staff `json:"author"` Author Staff `json:"author"`
} }
type NewsResponse struct {
News
Views int64 `json:"views"`
}
type NewsUpdate struct { type NewsUpdate struct {
ID string `json:"id"` ID string `json:"id"`
Title string `json:"title"` Title string `json:"title"`
@ -52,11 +56,12 @@ type NewsUpdate struct {
} }
type Filter struct { type Filter struct {
Category, Tags, Active string Category, Tags, Active, Search string
} }
type NewsFilter struct { type NewsFilter struct {
Tags []string Tags []string
Category []string Category []string
Active string Active string
Search string
} }

View File

@ -5,12 +5,12 @@ import (
"strings" "strings"
) )
func (i *impl) GetAll(filter newsdomain.Filter) ([]newsdomain.News, error) { func (i *impl) GetAll(filter newsdomain.Filter) ([]newsdomain.NewsResponse, error) {
var err error var err error
categories := []string{} categories := []string{}
tags := []string{} tags := []string{}
news := []newsdomain.News{} news := []newsdomain.NewsResponse{}
tagCodeArr := strings.Split(filter.Tags, " ") tagCodeArr := strings.Split(filter.Tags, " ")
categoryCodeArr := strings.Split(filter.Category, " ") categoryCodeArr := strings.Split(filter.Category, " ")
@ -41,6 +41,7 @@ func (i *impl) GetAll(filter newsdomain.Filter) ([]newsdomain.News, error) {
Tags: tags, Tags: tags,
Category: categories, Category: categories,
Active: filter.Active, Active: filter.Active,
Search: filter.Search,
} }
return i.newsRepo.GetAll(filterSpec) return i.newsRepo.GetAll(filterSpec)
} }

View File

@ -0,0 +1,7 @@
package newssvc
import newsdomain "legalgo-BE-go/internal/domain/news"
func (i *impl) GetByID(id string) (*newsdomain.News, error) {
return i.newsRepo.GetByID(id)
}

View File

@ -16,8 +16,9 @@ type impl struct {
} }
type News interface { type News interface {
GetAll(filter newsdomain.Filter) ([]newsdomain.News, error) GetAll(filter newsdomain.Filter) ([]newsdomain.NewsResponse, error)
GetBySlug(string) (*newsdomain.News, error) GetBySlug(string) (*newsdomain.News, error)
GetByID(string) (*newsdomain.News, error)
Create(newsdomain.NewsReq, string) error Create(newsdomain.NewsReq, string) error
Update(string, newsdomain.NewsUpdate) error Update(string, newsdomain.NewsUpdate) error
Delete(string) error Delete(string) error