feat: news router

This commit is contained in:
ericprd 2025-03-02 04:36:17 +08:00
parent f4ae93bc48
commit 915f12eaf7
49 changed files with 863 additions and 132 deletions

22
.vscode/launch.json vendored Normal file
View File

@ -0,0 +1,22 @@
{
"version": "0.2.0",
"configurations": [
{
"name": "Launch Package",
"type": "go",
"request": "launch",
"mode": "debug",
"program": "${workspaceFolder}/cmd/legalgo/",
"dlvLoadConfig":{
"followPointers": true,
"maxVariableRecurse": 1,
"maxStringLen": 512, //
"maxArrayValues": 64,
"maxStructFields": -1
},
"dlvFlags": ["--check-go-version=false"],
"env": {
"TZ": ""
}
}]
}

View File

@ -29,6 +29,8 @@ func main() {
log.Fatalf("failed to connect to database: %v", err) log.Fatalf("failed to connect to database: %v", err)
} }
// db.DropTables()
if err := db.Migrate(); err != nil { if err := db.Migrate(); err != nil {
log.Fatal("Migration failed: ", err) log.Fatal("Migration failed: ", err)
} }

View File

@ -1,11 +1,21 @@
package database package database
import "time" import (
"time"
)
type Category struct { type Category struct {
ID string `gorm:"primaryKey;not null" json:"id"` ID string `gorm:"primaryKey" json:"id"`
Code string `gorm:"not null;unique" json:"code"` Code string `gorm:"not null;unique" json:"code"`
Name string `gorm:"not null" json:"name"` Name string `gorm:"not null" json:"name"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"` CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"` UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
} }
type CategoryModel struct {
ID string `gorm:"primaryKey" json:"id"`
Name string `gorm:"not null;unique" json:"name"`
Code string `gorm:"not null" json:"code"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
}

View File

@ -36,6 +36,18 @@ func NewDB() (*DB, error) {
return &DB{db}, nil return &DB{db}, nil
} }
func (db *DB) DropTables() error {
// Auto Migrate the User model
return db.Migrator().DropTable(
// &Staff{},
// &SubscribePlan{},
// &Subscribe{},
// &User{},
&Tag{},
&Category{},
&News{},
)
}
func (db *DB) Migrate() error { func (db *DB) Migrate() error {
// Auto Migrate the User model // Auto Migrate the User model
return db.AutoMigrate( return db.AutoMigrate(
@ -43,7 +55,11 @@ func (db *DB) Migrate() error {
&SubscribePlan{}, &SubscribePlan{},
&Subscribe{}, &Subscribe{},
&User{}, &User{},
&Tag{}, // &Tag{},
&Category{}, // &Category{},
// &News{},
&NewsModel{},
&TagModel{},
&CategoryModel{},
) )
} }

View File

@ -5,13 +5,31 @@ import (
) )
type News struct { type News struct {
ID string `gorm:"primaryKey" json:"id"` ID string `gorm:"primaryKey" json:"id"`
AuthorID string `gorm:"not null" json:"author_id"` Title string `gorm:"default:null" json:"title"`
Title string `gorm:"default:null" json:"title"` Tags []Tag `gorm:"many2many:news_tags" json:"tags"`
Content string `gorm:"default:null" json:"content"` Categories []Category `gorm:"many2many:news_categories" json:"categories"`
IsPremium bool `gorm:"default:false" json:"is_premium"` Content string `gorm:"default:null" json:"content"`
Slug string `gorm:"default:null" json:"slug"` LiveAt time.Time `gorm:"not null" json:"live_at"`
FeaturedImage string `gorm:"default:null" json:"featured_image"` AuthorID string `gorm:"not null" json:"author_id"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"` IsPremium bool `gorm:"default:false" json:"is_premium"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"` Slug string `gorm:"default:null" json:"slug"`
FeaturedImage string `gorm:"default:null" json:"featured_image"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
}
type NewsModel struct {
ID string `gorm:"primaryKey" json:"id"`
Title string `json:"title"`
Content string `json:"content"`
Categories []CategoryModel `gorm:"many2many:news_categories" json:"categories"`
Tags []TagModel `gorm:"many2many:news_tags" json:"tags"`
IsPremium bool `gorm:"default:false" json:"is_premium"`
Slug string `gorm:"default:null" json:"slug"`
FeaturedImage string `gorm:"default:null" json:"featured_image"`
AuthorID string `gorm:"not null" json:"author_id"`
LiveAt time.Time `gorm:"not null" json:"live_at"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
} }

View File

@ -5,9 +5,17 @@ import (
) )
type Tag struct { type Tag struct {
ID string `gorm:"primaryKey;not null" json:"id"` ID string `gorm:"primaryKey" json:"id"`
Code string `gorm:"not null;unique" json:"code"` Code string `gorm:"not null;unique" json:"code"`
Name string `gorm:"not null" json:"name"` Name string `gorm:"not null" json:"name"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"` CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"` UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
} }
type TagModel struct {
ID string `gorm:"primaryKey" json:"id"`
Name string `gorm:"not null;unique" json:"name"`
Code string `gorm:"not null" json:"code"`
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_at"`
UpdatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"updated_at"`
}

View File

@ -5,7 +5,7 @@ import (
) )
type User struct { type User struct {
ID string `gorm:"primaryKey" json:"id"` ID string `gorm:"primaryKey;not null" json:"id"`
SubscribeID string `gorm:"not null" json:"subscribe_id"` SubscribeID string `gorm:"not null" json:"subscribe_id"`
Email string `gorm:"unique;not null" json:"email"` Email string `gorm:"unique;not null" json:"email"`
Password string `gorm:"not null" json:"password"` Password string `gorm:"not null" json:"password"`

View File

@ -0,0 +1,22 @@
package categoryrepository
import (
"legalgo-BE-go/database"
categorydomain "legalgo-BE-go/internal/domain/category"
"github.com/google/uuid"
)
func (a *accessor) CreateModel(spec categorydomain.CategoryReq) error {
data := database.CategoryModel{
ID: uuid.NewString(),
Name: spec.Name,
Code: spec.Code,
}
if err := a.DB.Create(&data).Error; err != nil {
return err
}
return nil
}

View File

@ -0,0 +1,15 @@
package categoryrepository
import (
"legalgo-BE-go/database"
)
func (a *accessor) GetAllModel() ([]database.CategoryModel, error) {
var categories []database.CategoryModel
if err := a.DB.Find(&categories).Error; err != nil {
return nil, err
}
return categories, nil
}

View File

@ -0,0 +1,15 @@
package categoryrepository
import (
categorydomain "legalgo-BE-go/internal/domain/category"
)
func (a *accessor) GetByIDs(ids []string) ([]categorydomain.Category, error) {
var categories []categorydomain.Category
if err := a.DB.Find(&categories, "id IN ?", ids).Error; err != nil {
return nil, err
}
return categories, nil
}

View File

@ -0,0 +1,15 @@
package categoryrepository
import (
"legalgo-BE-go/database"
)
func (a *accessor) GetBulks(ids []string) ([]database.CategoryModel, error) {
var categories []database.CategoryModel
if err := a.DB.Find(&categories, "id IN ?", ids).Error; err != nil {
return nil, err
}
return categories, nil
}

View File

@ -11,7 +11,13 @@ type accessor struct {
type Category interface { type Category interface {
Create(categorydomain.CategoryReq) error Create(categorydomain.CategoryReq) error
CreateModel(categorydomain.CategoryReq) error
GetAll() ([]categorydomain.Category, error) GetAll() ([]categorydomain.Category, error)
GetAllModel() ([]database.CategoryModel, error)
GetByIDs([]string) ([]categorydomain.Category, error)
GetBulks([]string) ([]database.CategoryModel, error)
} }
func New( func New(

View File

@ -2,6 +2,7 @@ package repository
import ( import (
categoryrepository "legalgo-BE-go/internal/accessor/category" categoryrepository "legalgo-BE-go/internal/accessor/category"
newsrepository "legalgo-BE-go/internal/accessor/news"
redisaccessor "legalgo-BE-go/internal/accessor/redis" redisaccessor "legalgo-BE-go/internal/accessor/redis"
staffrepository "legalgo-BE-go/internal/accessor/staff" staffrepository "legalgo-BE-go/internal/accessor/staff"
subscriberepository "legalgo-BE-go/internal/accessor/subscribe" subscriberepository "legalgo-BE-go/internal/accessor/subscribe"
@ -20,4 +21,5 @@ var Module = fx.Module("repository", fx.Provide(
subscriberepository.New, subscriberepository.New,
tagrepository.New, tagrepository.New,
categoryrepository.New, categoryrepository.New,
newsrepository.New,
)) ))

View File

@ -0,0 +1,13 @@
package newsrepository
import (
"fmt"
newsdomain "legalgo-BE-go/internal/domain/news"
)
func (a *accessor) Create(spec *newsdomain.News) error {
if err := a.db.Create(&spec).Error; err != nil {
return fmt.Errorf("failed to create news: %w", err)
}
return nil
}

View File

@ -0,0 +1,13 @@
package newsrepository
import (
"fmt"
"legalgo-BE-go/database"
)
func (a *accessor) CreateModel(spec database.NewsModel) error {
if err := a.db.Create(&spec).Error; err != nil {
return fmt.Errorf("failed to create news: %w", err)
}
return nil
}

View File

@ -0,0 +1,13 @@
package newsrepository
import newsdomain "legalgo-BE-go/internal/domain/news"
func (a *accessor) GetAll() ([]newsdomain.News, error) {
var news []newsdomain.News
if err := a.db.Preload("Tags").Preload("Categories").Find(&news).Error; err != nil {
return nil, err
}
return news, nil
}

View File

@ -0,0 +1,13 @@
package newsrepository
import "legalgo-BE-go/database"
func (a *accessor) GetAllModel() ([]database.NewsModel, error) {
var news []database.NewsModel
if err := a.db.Preload("Tags").Preload("Categories").Find(&news).Error; err != nil {
return nil, err
}
return news, nil
}

View File

@ -0,0 +1,21 @@
package newsrepository
import (
"legalgo-BE-go/database"
newsdomain "legalgo-BE-go/internal/domain/news"
)
type accessor struct {
db *database.DB
}
type News interface {
GetAll() ([]newsdomain.News, error)
GetAllModel() ([]database.NewsModel, error)
Create(*newsdomain.News) error
CreateModel(database.NewsModel) error
}
func New(db *database.DB) News {
return &accessor{db}
}

View File

@ -0,0 +1,22 @@
package tagrepository
import (
"legalgo-BE-go/database"
tagdomain "legalgo-BE-go/internal/domain/tag"
"github.com/google/uuid"
)
func (acc *accessor) CreateModel(spec tagdomain.TagReq) error {
data := &database.TagModel{
ID: uuid.NewString(),
Code: spec.Code,
Name: spec.Name,
}
if err := acc.DB.Create(&data).Error; err != nil {
return err
}
return nil
}

View File

@ -0,0 +1,13 @@
package tagrepository
import "legalgo-BE-go/database"
func (acc *accessor) GetAllModel() ([]database.TagModel, error) {
var tags []database.TagModel
if err := acc.DB.Find(&tags).Error; err != nil {
return nil, err
}
return tags, nil
}

View File

@ -0,0 +1,13 @@
package tagrepository
import tagdomain "legalgo-BE-go/internal/domain/tag"
func (a *accessor) GetByIDs(ids []string) ([]tagdomain.Tag, error) {
var tags []tagdomain.Tag
if err := a.DB.Find(&tags, "id IN ?", ids).Error; err != nil {
return nil, err
}
return tags, nil
}

View File

@ -0,0 +1,13 @@
package tagrepository
import "legalgo-BE-go/database"
func (a *accessor) GetBulks(ids []string) ([]database.TagModel, error) {
var tags []database.TagModel
if err := a.DB.Find(&tags, "id IN ?", ids).Error; err != nil {
return nil, err
}
return tags, nil
}

View File

@ -11,7 +11,11 @@ type accessor struct {
type TagAccessor interface { type TagAccessor interface {
Create(tagdomain.TagReq) error Create(tagdomain.TagReq) error
CreateModel(tagdomain.TagReq) error
GetAll() ([]tagdomain.Tag, error) GetAll() ([]tagdomain.Tag, error)
GetAllModel() ([]database.TagModel, error)
GetByIDs([]string) ([]tagdomain.Tag, error)
GetBulks(ids []string) ([]database.TagModel, error)
} }
func New( func New(

View File

@ -15,7 +15,7 @@ func (ur *UserRepository) GetUserByEmail(email string) (*authdomain.User, error)
return nil, errors.New("email is empty") return nil, errors.New("email is empty")
} }
if err := ur.DB.First(&user, "email = ?", email).Error; err != nil { if err := ur.DB.Find(&user, "email = ?", email).Error; err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) { if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, errors.New("user not found") return nil, errors.New("user not found")
} }

View File

@ -1,12 +1,10 @@
package authhttp package authhttp
import ( import (
"errors"
authsvc "legalgo-BE-go/internal/services/auth" authsvc "legalgo-BE-go/internal/services/auth"
"legalgo-BE-go/internal/utilities/response" "legalgo-BE-go/internal/utilities/response"
"legalgo-BE-go/internal/utilities/utils" "legalgo-BE-go/internal/utilities/utils"
"net/http" "net/http"
"strings"
"github.com/go-chi/chi/v5" "github.com/go-chi/chi/v5"
) )
@ -17,46 +15,7 @@ func GetStaffProfile(
) { ) {
router.Get("/staff/profile", func(w http.ResponseWriter, r *http.Request) { router.Get("/staff/profile", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context() ctx := r.Context()
authHeader := r.Header.Get("Authorization") destructedToken, err := utils.GetTokenDetail(r)
if authHeader == "" {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("provided auth is empty"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"required params is not provided",
)
return
}
if !strings.HasPrefix(authHeader, "Bearer") {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("invalid authorization token"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"invalid required token",
)
return
}
token := strings.Split(authHeader, " ")
if len(token) < 2 {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("invalid authorization"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"invalid required token",
)
return
}
destructedToken, err := utils.DestructToken(token[1])
if err != nil { if err != nil {
response.ResponseWithErrorCode( response.ResponseWithErrorCode(
ctx, ctx,
@ -92,46 +51,7 @@ func GetUserProfile(
) { ) {
router.Get("/user/profile", func(w http.ResponseWriter, r *http.Request) { router.Get("/user/profile", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context() ctx := r.Context()
authHeader := r.Header.Get("Authorization") destructedToken, err := utils.GetTokenDetail(r)
if authHeader == "" {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("provided auth is empty"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"required params is not provided",
)
return
}
if !strings.HasPrefix(authHeader, "Bearer") {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("invalid authorization token"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"invalid required token",
)
return
}
token := strings.Split(authHeader, " ")
if len(token) < 2 {
response.ResponseWithErrorCode(
ctx,
w,
errors.New("invalid authorization"),
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
"invalid required token",
)
return
}
destructedToken, err := utils.DestructToken(token[1])
if err != nil { if err != nil {
response.ResponseWithErrorCode( response.ResponseWithErrorCode(
ctx, ctx,

View File

@ -14,7 +14,8 @@ func GetAll(
) { ) {
router.Get("/category", func(w http.ResponseWriter, r *http.Request) { router.Get("/category", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context() ctx := r.Context()
subsPlan, err := categorySvc.GetAll() subsPlan, err := categorySvc.GetAllModel()
// subsPlan, err := categorySvc.GetAll()
if err != nil { if err != nil {
response.ResponseWithErrorCode( response.ResponseWithErrorCode(
ctx, ctx,

View File

@ -0,0 +1,93 @@
package newshttp
import (
staffrepository "legalgo-BE-go/internal/accessor/staff"
newsdomain "legalgo-BE-go/internal/domain/news"
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"
"github.com/go-playground/validator/v10"
)
func Create(
validate *validator.Validate,
newsSvc newssvc.News,
staffRepo staffrepository.StaffIntf,
router chi.Router,
) {
router.Post("/news/create", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
var spec newsdomain.NewsReq
if err := utils.UnmarshalBody(r, &spec); err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
if err := validate.Struct(spec); err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.(validator.ValidationErrors).Error(),
)
return
}
destructedToken, err := utils.GetTokenDetail(r)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
staffProfile, err := staffRepo.GetStaffByEmail(destructedToken.Email)
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
if err := newsSvc.CreateModel(spec, staffProfile.ID); 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 created successfully.",
})
})
}

View File

@ -0,0 +1,32 @@
package newshttp
import (
newssvc "legalgo-BE-go/internal/services/news"
"legalgo-BE-go/internal/utilities/response"
"net/http"
"github.com/go-chi/chi/v5"
)
func GetAll(
router chi.Router,
newsSvc newssvc.News,
) {
router.Get("/news", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()
news, err := newsSvc.GetAllModel()
if err != nil {
response.ResponseWithErrorCode(
ctx,
w,
err,
response.ErrBadRequest.Code,
response.ErrBadRequest.HttpCode,
err.Error(),
)
return
}
response.RespondJsonSuccess(ctx, w, news)
})
}

View File

@ -0,0 +1,8 @@
package newshttp
import "go.uber.org/fx"
var Module = fx.Module("news", fx.Invoke(
GetAll,
Create,
))

View File

@ -3,6 +3,7 @@ package internalhttp
import ( import (
authhttp "legalgo-BE-go/internal/api/http/auth" authhttp "legalgo-BE-go/internal/api/http/auth"
categoryhttp "legalgo-BE-go/internal/api/http/category" categoryhttp "legalgo-BE-go/internal/api/http/category"
newshttp "legalgo-BE-go/internal/api/http/news"
subscribeplanhttp "legalgo-BE-go/internal/api/http/subscribe_plan" subscribeplanhttp "legalgo-BE-go/internal/api/http/subscribe_plan"
taghttp "legalgo-BE-go/internal/api/http/tag" taghttp "legalgo-BE-go/internal/api/http/tag"
@ -23,6 +24,7 @@ var Module = fx.Module("router",
subscribeplanhttp.Module, subscribeplanhttp.Module,
taghttp.Module, taghttp.Module,
categoryhttp.Module, categoryhttp.Module,
newshttp.Module,
) )
func initRouter() chi.Router { func initRouter() chi.Router {

View File

@ -14,7 +14,8 @@ func GetAll(
) { ) {
router.Get("/tag", func(w http.ResponseWriter, r *http.Request) { router.Get("/tag", func(w http.ResponseWriter, r *http.Request) {
ctx := r.Context() ctx := r.Context()
subsPlan, err := tagSvc.GetAll() tags, err := tagSvc.GetAllModel()
// tags, err := tagSvc.GetAll()
if err != nil { if err != nil {
response.ResponseWithErrorCode( response.ResponseWithErrorCode(
ctx, ctx,
@ -27,6 +28,6 @@ func GetAll(
return return
} }
response.RespondJsonSuccess(ctx, w, subsPlan) response.RespondJsonSuccess(ctx, w, tags)
}) })
} }

View File

@ -1,7 +1,7 @@
package categorydomain package categorydomain
type Category struct { type Category struct {
ID string `json:"id"` ID string `json:"id" gorm:"primaryKey"`
Name string `json:"name"` Name string `json:"name"`
Code string `json:"code"` Code string `json:"code"`
} }

View File

@ -1,12 +0,0 @@
package newsdomain
type News struct {
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"`
Slug string `json:"slug"`
Author string `json:"author"`
}

View File

@ -0,0 +1,32 @@
package newsdomain
import (
categorydomain "legalgo-BE-go/internal/domain/category"
tagdomain "legalgo-BE-go/internal/domain/tag"
"time"
)
type NewsReq struct {
Title string `json:"title" validate:"required"`
Content string `json:"content"`
FeaturedImage string `json:"featured_image"`
Tags []string `json:"tags"`
Categories []string `json:"categories"`
IsPremium bool `json:"is_premium"`
LiveAt string `json:"live_at" validate:"required"`
}
type News struct {
ID string `json:"id"`
Title string `json:"title"`
Content string `json:"content"`
FeaturedImage string `json:"featured_image"`
Tags []tagdomain.Tag `json:"tags"`
Categories []categorydomain.Category `json:"categories"`
IsPremium bool `json:"is_premium"`
Slug string `json:"slug"`
AuthorID string `json:"author_id"`
LiveAt time.Time `json:"live_at"`
CreatedAt time.Time `json:"created_at"`
UpdatedAt time.Time `json:"updated_at"`
}

View File

@ -6,7 +6,7 @@ type TagReq struct {
} }
type Tag struct { type Tag struct {
ID string `json:"id"` ID string `json:"id" gorm:"primaryKey"`
Code string `json:"code"` Code string `json:"code"`
Name string `json:"name"` Name string `json:"name"`
} }

View File

@ -3,5 +3,6 @@ package categorysvc
import categorydomain "legalgo-BE-go/internal/domain/category" import categorydomain "legalgo-BE-go/internal/domain/category"
func (i *impl) Create(spec categorydomain.CategoryReq) error { func (i *impl) Create(spec categorydomain.CategoryReq) error {
return i.categoryRepo.Create(spec) return i.categoryRepo.CreateModel(spec)
// return i.categoryRepo.Create(spec)
} }

View File

@ -0,0 +1,9 @@
package categorysvc
import (
"legalgo-BE-go/database"
)
func (i *impl) GetAllModel() ([]database.CategoryModel, error) {
return i.categoryRepo.GetAllModel()
}

View File

@ -1,6 +1,7 @@
package categorysvc package categorysvc
import ( import (
"legalgo-BE-go/database"
categoryrepository "legalgo-BE-go/internal/accessor/category" categoryrepository "legalgo-BE-go/internal/accessor/category"
categorydomain "legalgo-BE-go/internal/domain/category" categorydomain "legalgo-BE-go/internal/domain/category"
) )
@ -12,6 +13,7 @@ type impl struct {
type Category interface { type Category interface {
Create(categorydomain.CategoryReq) error Create(categorydomain.CategoryReq) error
GetAll() ([]categorydomain.Category, error) GetAll() ([]categorydomain.Category, error)
GetAllModel() ([]database.CategoryModel, error)
} }
func New( func New(

View File

@ -3,6 +3,7 @@ package services
import ( import (
serviceauth "legalgo-BE-go/internal/services/auth" serviceauth "legalgo-BE-go/internal/services/auth"
categorysvc "legalgo-BE-go/internal/services/category" categorysvc "legalgo-BE-go/internal/services/category"
newssvc "legalgo-BE-go/internal/services/news"
subscribesvc "legalgo-BE-go/internal/services/subscribe" subscribesvc "legalgo-BE-go/internal/services/subscribe"
subscribeplansvc "legalgo-BE-go/internal/services/subscribe_plan" subscribeplansvc "legalgo-BE-go/internal/services/subscribe_plan"
tagsvc "legalgo-BE-go/internal/services/tag" tagsvc "legalgo-BE-go/internal/services/tag"
@ -17,5 +18,6 @@ var Module = fx.Module("services",
subscribesvc.New, subscribesvc.New,
tagsvc.New, tagsvc.New,
categorysvc.New, categorysvc.New,
newssvc.New,
), ),
) )

View File

@ -0,0 +1,47 @@
package newssvc
import (
newsdomain "legalgo-BE-go/internal/domain/news"
"strings"
"time"
"github.com/google/uuid"
)
func (i *impl) Create(spec newsdomain.NewsReq, staffId string) error {
slug := strings.ToLower(strings.ReplaceAll(spec.Title, " ", "-"))
tags, err := i.tagRepo.GetByIDs(spec.Tags)
if err != nil {
return err
}
categories, err := i.categoryRepo.GetByIDs(spec.Categories)
if err != nil {
return err
}
parsedTime, err := time.Parse(time.RFC3339, spec.LiveAt)
if err != nil {
return err
}
newSpec := &newsdomain.News{
ID: uuid.NewString(),
Title: spec.Title,
Content: spec.Content,
FeaturedImage: spec.FeaturedImage,
IsPremium: spec.IsPremium,
Slug: slug,
LiveAt: parsedTime,
AuthorID: staffId,
Tags: tags,
Categories: categories,
}
if err := i.newsRepo.Create(newSpec); err != nil {
return err
}
return nil
}

View File

@ -0,0 +1,48 @@
package newssvc
import (
"legalgo-BE-go/database"
newsdomain "legalgo-BE-go/internal/domain/news"
"strings"
"time"
"github.com/google/uuid"
)
func (i *impl) CreateModel(spec newsdomain.NewsReq, staffId string) error {
slug := strings.ToLower(strings.ReplaceAll(spec.Title, " ", "-"))
tags, err := i.tagRepo.GetBulks(spec.Tags)
if err != nil {
return err
}
categories, err := i.categoryRepo.GetBulks(spec.Categories)
if err != nil {
return err
}
parsedTime, err := time.Parse(time.RFC3339, spec.LiveAt)
if err != nil {
return err
}
newSpec := database.NewsModel{
ID: uuid.NewString(),
Title: spec.Title,
Content: spec.Content,
FeaturedImage: spec.FeaturedImage,
IsPremium: spec.IsPremium,
Slug: slug,
LiveAt: parsedTime,
AuthorID: staffId,
Tags: tags,
Categories: categories,
}
if err := i.newsRepo.CreateModel(newSpec); err != nil {
return err
}
return nil
}

View File

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

View File

@ -0,0 +1,9 @@
package newssvc
import (
"legalgo-BE-go/database"
)
func (i *impl) GetAllModel() ([]database.NewsModel, error) {
return i.newsRepo.GetAllModel()
}

View File

@ -0,0 +1,34 @@
package newssvc
import (
"legalgo-BE-go/database"
categoryrepository "legalgo-BE-go/internal/accessor/category"
newsrepository "legalgo-BE-go/internal/accessor/news"
tagrepository "legalgo-BE-go/internal/accessor/tag"
newsdomain "legalgo-BE-go/internal/domain/news"
)
type impl struct {
newsRepo newsrepository.News
tagRepo tagrepository.TagAccessor
categoryRepo categoryrepository.Category
}
type News interface {
GetAll() ([]newsdomain.News, error)
GetAllModel() ([]database.NewsModel, error)
Create(newsdomain.NewsReq, string) error
CreateModel(newsdomain.NewsReq, string) error
}
func New(
newsRepo newsrepository.News,
tagRepo tagrepository.TagAccessor,
categoryRepo categoryrepository.Category,
) News {
return &impl{
newsRepo,
tagRepo,
categoryRepo,
}
}

View File

@ -3,5 +3,6 @@ package tagsvc
import tagdomain "legalgo-BE-go/internal/domain/tag" import tagdomain "legalgo-BE-go/internal/domain/tag"
func (i *impl) Create(spec tagdomain.TagReq) error { func (i *impl) Create(spec tagdomain.TagReq) error {
return i.tagRepo.Create(spec) // return i.tagRepo.Create(spec)
return i.tagRepo.CreateModel(spec)
} }

View File

@ -0,0 +1,9 @@
package tagsvc
import (
"legalgo-BE-go/database"
)
func (i *impl) GetAllModel() ([]database.TagModel, error) {
return i.tagRepo.GetAllModel()
}

View File

@ -1,6 +1,7 @@
package tagsvc package tagsvc
import ( import (
"legalgo-BE-go/database"
tagrepository "legalgo-BE-go/internal/accessor/tag" tagrepository "legalgo-BE-go/internal/accessor/tag"
tagdomain "legalgo-BE-go/internal/domain/tag" tagdomain "legalgo-BE-go/internal/domain/tag"
) )
@ -12,6 +13,7 @@ type impl struct {
type TagIntf interface { type TagIntf interface {
Create(tagdomain.TagReq) error Create(tagdomain.TagReq) error
GetAll() ([]tagdomain.Tag, error) GetAll() ([]tagdomain.Tag, error)
GetAllModel() ([]database.TagModel, error)
} }
func New( func New(

View File

@ -0,0 +1,34 @@
package utils
import (
"errors"
authdomain "legalgo-BE-go/internal/domain/auth"
"net/http"
"strings"
)
func GetTokenDetail(r *http.Request) (authdomain.AuthToken, error) {
authHeader := r.Header.Get("Authorization")
var data authdomain.AuthToken
if authHeader == "" {
return data, errors.New("unauthorized")
}
if !strings.HasPrefix(authHeader, "Bearer") {
return data, errors.New("invalid token")
}
token := strings.Split(authHeader, " ")
if len(token) < 2 {
return data, errors.New("invalid token")
}
data, err := DestructToken(token[1])
if err != nil {
return data, err
}
return data, nil
}

View File

@ -10,13 +10,6 @@ paths:
summary: "get staff profile" summary: "get staff profile"
tags: tags:
- Staff - Staff
parameters:
- name: "id"
in: query
description: "staff id"
required: true
schema:
type: string
responses: responses:
"200": "200":
description: Success get profile description: Success get profile
@ -169,13 +162,6 @@ paths:
summary: "get staff profile" summary: "get staff profile"
tags: tags:
- User - User
parameters:
- name: "id"
in: query
description: "user id"
required: true
schema:
type: string
responses: responses:
"200": "200":
description: Success get profile description: Success get profile
@ -574,7 +560,7 @@ paths:
post: post:
summary: Create a new category summary: Create a new category
tags: tags:
- Tags - Category
requestBody: requestBody:
required: true required: true
content: content:
@ -631,3 +617,177 @@ paths:
type: string type: string
message: message:
type: string type: string
/news/create:
post:
summary: Create a new news article
tags:
- News
requestBody:
required: true
content:
application/json:
schema:
type: object
properties:
title:
type: string
content:
type: string
featured_image:
type: string
format: uri
tags:
type: array
items:
type: string
categories:
type: array
items:
type: string
is_premium:
type: boolean
live_at:
type: string
format: date-time
required:
- title
- content
- featured_image
- tags
- categories
- is_premium
- live_at
responses:
"201":
description: News article created
content:
application/json:
schema:
type: object
properties:
data:
type: object
properties:
message:
type: string
example: "news created successfully."
"400":
description: Bad request
content:
application/json:
schema:
type: object
properties:
error:
type: object
properties:
code:
type: string
message:
type: string
"409":
description: Conflict (e.g., duplicate title or invalid tag/category)
content:
application/json:
schema:
type: object
properties:
error:
type: object
properties:
code:
type: string
message:
type: string
/news:
get:
summary: Get all news articles
tags:
- News
responses:
"200":
description: Successfully retrieved all news articles
content:
application/json:
schema:
type: object
properties:
data:
type: array
items:
type: object
properties:
id:
type: string
title:
type: string
content:
type: string
categories:
type: array
items:
type: object
properties:
id:
type: string
name:
type: string
code:
type: string
created_at:
type: string
format: date-time
updated_at:
type: string
format: date-time
tags:
type: array
items:
type: object
properties:
id:
type: string
name:
type: string
code:
type: string
created_at:
type: string
format: date-time
updated_at:
type: string
format: date-time
is_premium:
type: boolean
slug:
type: string
featured_image:
type: string
format: uri
author_id:
type: string
live_at:
type: string
format: date-time
created_at:
type: string
format: date-time
updated_at:
type: string
format: date-time
"400":
description: Bad request
content:
application/json:
schema:
type: object
properties:
error:
type: object
properties:
code:
type: string
message:
type: string