feat: news router
This commit is contained in:
parent
f4ae93bc48
commit
915f12eaf7
22
.vscode/launch.json
vendored
Normal file
22
.vscode/launch.json
vendored
Normal 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": ""
|
||||
}
|
||||
}]
|
||||
}
|
||||
@ -29,6 +29,8 @@ func main() {
|
||||
log.Fatalf("failed to connect to database: %v", err)
|
||||
}
|
||||
|
||||
// db.DropTables()
|
||||
|
||||
if err := db.Migrate(); err != nil {
|
||||
log.Fatal("Migration failed: ", err)
|
||||
}
|
||||
|
||||
@ -1,11 +1,21 @@
|
||||
package database
|
||||
|
||||
import "time"
|
||||
import (
|
||||
"time"
|
||||
)
|
||||
|
||||
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"`
|
||||
Name string `gorm:"not null" json:"name"`
|
||||
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_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"`
|
||||
}
|
||||
|
||||
@ -36,6 +36,18 @@ func NewDB() (*DB, error) {
|
||||
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 {
|
||||
// Auto Migrate the User model
|
||||
return db.AutoMigrate(
|
||||
@ -43,7 +55,11 @@ func (db *DB) Migrate() error {
|
||||
&SubscribePlan{},
|
||||
&Subscribe{},
|
||||
&User{},
|
||||
&Tag{},
|
||||
&Category{},
|
||||
// &Tag{},
|
||||
// &Category{},
|
||||
// &News{},
|
||||
&NewsModel{},
|
||||
&TagModel{},
|
||||
&CategoryModel{},
|
||||
)
|
||||
}
|
||||
|
||||
@ -5,13 +5,31 @@ import (
|
||||
)
|
||||
|
||||
type News struct {
|
||||
ID string `gorm:"primaryKey" json:"id"`
|
||||
AuthorID string `gorm:"not null" json:"author_id"`
|
||||
Title string `gorm:"default:null" json:"title"`
|
||||
Content string `gorm:"default:null" json:"content"`
|
||||
IsPremium bool `gorm:"default:false" json:"is_premium"`
|
||||
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"`
|
||||
ID string `gorm:"primaryKey" json:"id"`
|
||||
Title string `gorm:"default:null" json:"title"`
|
||||
Tags []Tag `gorm:"many2many:news_tags" json:"tags"`
|
||||
Categories []Category `gorm:"many2many:news_categories" json:"categories"`
|
||||
Content string `gorm:"default:null" json:"content"`
|
||||
LiveAt time.Time `gorm:"not null" json:"live_at"`
|
||||
AuthorID string `gorm:"not null" json:"author_id"`
|
||||
IsPremium bool `gorm:"default:false" json:"is_premium"`
|
||||
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"`
|
||||
}
|
||||
|
||||
@ -5,9 +5,17 @@ import (
|
||||
)
|
||||
|
||||
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"`
|
||||
Name string `gorm:"not null" json:"name"`
|
||||
CreatedAt time.Time `gorm:"default:CURRENT_TIMESTAMP" json:"created_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"`
|
||||
}
|
||||
|
||||
@ -5,7 +5,7 @@ import (
|
||||
)
|
||||
|
||||
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"`
|
||||
Email string `gorm:"unique;not null" json:"email"`
|
||||
Password string `gorm:"not null" json:"password"`
|
||||
|
||||
22
internal/accessor/category/create_model.go
Normal file
22
internal/accessor/category/create_model.go
Normal 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
|
||||
}
|
||||
15
internal/accessor/category/get_all_model.go
Normal file
15
internal/accessor/category/get_all_model.go
Normal 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
|
||||
}
|
||||
15
internal/accessor/category/get_bulk.go
Normal file
15
internal/accessor/category/get_bulk.go
Normal 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
|
||||
}
|
||||
15
internal/accessor/category/get_bulk_model.go
Normal file
15
internal/accessor/category/get_bulk_model.go
Normal 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
|
||||
}
|
||||
@ -11,7 +11,13 @@ type accessor struct {
|
||||
|
||||
type Category interface {
|
||||
Create(categorydomain.CategoryReq) error
|
||||
CreateModel(categorydomain.CategoryReq) error
|
||||
|
||||
GetAll() ([]categorydomain.Category, error)
|
||||
GetAllModel() ([]database.CategoryModel, error)
|
||||
|
||||
GetByIDs([]string) ([]categorydomain.Category, error)
|
||||
GetBulks([]string) ([]database.CategoryModel, error)
|
||||
}
|
||||
|
||||
func New(
|
||||
|
||||
@ -2,6 +2,7 @@ package repository
|
||||
|
||||
import (
|
||||
categoryrepository "legalgo-BE-go/internal/accessor/category"
|
||||
newsrepository "legalgo-BE-go/internal/accessor/news"
|
||||
redisaccessor "legalgo-BE-go/internal/accessor/redis"
|
||||
staffrepository "legalgo-BE-go/internal/accessor/staff"
|
||||
subscriberepository "legalgo-BE-go/internal/accessor/subscribe"
|
||||
@ -20,4 +21,5 @@ var Module = fx.Module("repository", fx.Provide(
|
||||
subscriberepository.New,
|
||||
tagrepository.New,
|
||||
categoryrepository.New,
|
||||
newsrepository.New,
|
||||
))
|
||||
|
||||
13
internal/accessor/news/create.go
Normal file
13
internal/accessor/news/create.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/news/create_model.go
Normal file
13
internal/accessor/news/create_model.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/news/get_all.go
Normal file
13
internal/accessor/news/get_all.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/news/get_all_model.go
Normal file
13
internal/accessor/news/get_all_model.go
Normal 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
|
||||
}
|
||||
21
internal/accessor/news/impl.go
Normal file
21
internal/accessor/news/impl.go
Normal 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}
|
||||
}
|
||||
22
internal/accessor/tag/create_model.go
Normal file
22
internal/accessor/tag/create_model.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/tag/get_all_model.go
Normal file
13
internal/accessor/tag/get_all_model.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/tag/get_bulk.go
Normal file
13
internal/accessor/tag/get_bulk.go
Normal 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
|
||||
}
|
||||
13
internal/accessor/tag/get_bulk_model.go
Normal file
13
internal/accessor/tag/get_bulk_model.go
Normal 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
|
||||
}
|
||||
@ -11,7 +11,11 @@ type accessor struct {
|
||||
|
||||
type TagAccessor interface {
|
||||
Create(tagdomain.TagReq) error
|
||||
CreateModel(tagdomain.TagReq) error
|
||||
GetAll() ([]tagdomain.Tag, error)
|
||||
GetAllModel() ([]database.TagModel, error)
|
||||
GetByIDs([]string) ([]tagdomain.Tag, error)
|
||||
GetBulks(ids []string) ([]database.TagModel, error)
|
||||
}
|
||||
|
||||
func New(
|
||||
|
||||
@ -15,7 +15,7 @@ func (ur *UserRepository) GetUserByEmail(email string) (*authdomain.User, error)
|
||||
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) {
|
||||
return nil, errors.New("user not found")
|
||||
}
|
||||
|
||||
@ -1,12 +1,10 @@
|
||||
package authhttp
|
||||
|
||||
import (
|
||||
"errors"
|
||||
authsvc "legalgo-BE-go/internal/services/auth"
|
||||
"legalgo-BE-go/internal/utilities/response"
|
||||
"legalgo-BE-go/internal/utilities/utils"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
"github.com/go-chi/chi/v5"
|
||||
)
|
||||
@ -17,46 +15,7 @@ func GetStaffProfile(
|
||||
) {
|
||||
router.Get("/staff/profile", func(w http.ResponseWriter, r *http.Request) {
|
||||
ctx := r.Context()
|
||||
authHeader := r.Header.Get("Authorization")
|
||||
|
||||
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])
|
||||
destructedToken, err := utils.GetTokenDetail(r)
|
||||
if err != nil {
|
||||
response.ResponseWithErrorCode(
|
||||
ctx,
|
||||
@ -92,46 +51,7 @@ func GetUserProfile(
|
||||
) {
|
||||
router.Get("/user/profile", func(w http.ResponseWriter, r *http.Request) {
|
||||
ctx := r.Context()
|
||||
authHeader := r.Header.Get("Authorization")
|
||||
|
||||
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])
|
||||
destructedToken, err := utils.GetTokenDetail(r)
|
||||
if err != nil {
|
||||
response.ResponseWithErrorCode(
|
||||
ctx,
|
||||
|
||||
@ -14,7 +14,8 @@ func GetAll(
|
||||
) {
|
||||
router.Get("/category", func(w http.ResponseWriter, r *http.Request) {
|
||||
ctx := r.Context()
|
||||
subsPlan, err := categorySvc.GetAll()
|
||||
subsPlan, err := categorySvc.GetAllModel()
|
||||
// subsPlan, err := categorySvc.GetAll()
|
||||
if err != nil {
|
||||
response.ResponseWithErrorCode(
|
||||
ctx,
|
||||
|
||||
93
internal/api/http/news/create.go
Normal file
93
internal/api/http/news/create.go
Normal 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.",
|
||||
})
|
||||
})
|
||||
}
|
||||
32
internal/api/http/news/get_all.go
Normal file
32
internal/api/http/news/get_all.go
Normal 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)
|
||||
})
|
||||
}
|
||||
8
internal/api/http/news/module.go
Normal file
8
internal/api/http/news/module.go
Normal file
@ -0,0 +1,8 @@
|
||||
package newshttp
|
||||
|
||||
import "go.uber.org/fx"
|
||||
|
||||
var Module = fx.Module("news", fx.Invoke(
|
||||
GetAll,
|
||||
Create,
|
||||
))
|
||||
@ -3,6 +3,7 @@ package internalhttp
|
||||
import (
|
||||
authhttp "legalgo-BE-go/internal/api/http/auth"
|
||||
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"
|
||||
taghttp "legalgo-BE-go/internal/api/http/tag"
|
||||
|
||||
@ -23,6 +24,7 @@ var Module = fx.Module("router",
|
||||
subscribeplanhttp.Module,
|
||||
taghttp.Module,
|
||||
categoryhttp.Module,
|
||||
newshttp.Module,
|
||||
)
|
||||
|
||||
func initRouter() chi.Router {
|
||||
|
||||
@ -14,7 +14,8 @@ func GetAll(
|
||||
) {
|
||||
router.Get("/tag", func(w http.ResponseWriter, r *http.Request) {
|
||||
ctx := r.Context()
|
||||
subsPlan, err := tagSvc.GetAll()
|
||||
tags, err := tagSvc.GetAllModel()
|
||||
// tags, err := tagSvc.GetAll()
|
||||
if err != nil {
|
||||
response.ResponseWithErrorCode(
|
||||
ctx,
|
||||
@ -27,6 +28,6 @@ func GetAll(
|
||||
return
|
||||
}
|
||||
|
||||
response.RespondJsonSuccess(ctx, w, subsPlan)
|
||||
response.RespondJsonSuccess(ctx, w, tags)
|
||||
})
|
||||
}
|
||||
|
||||
@ -1,7 +1,7 @@
|
||||
package categorydomain
|
||||
|
||||
type Category struct {
|
||||
ID string `json:"id"`
|
||||
ID string `json:"id" gorm:"primaryKey"`
|
||||
Name string `json:"name"`
|
||||
Code string `json:"code"`
|
||||
}
|
||||
|
||||
@ -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"`
|
||||
}
|
||||
32
internal/domain/news/spec.go
Normal file
32
internal/domain/news/spec.go
Normal 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"`
|
||||
}
|
||||
@ -6,7 +6,7 @@ type TagReq struct {
|
||||
}
|
||||
|
||||
type Tag struct {
|
||||
ID string `json:"id"`
|
||||
ID string `json:"id" gorm:"primaryKey"`
|
||||
Code string `json:"code"`
|
||||
Name string `json:"name"`
|
||||
}
|
||||
|
||||
@ -3,5 +3,6 @@ package categorysvc
|
||||
import categorydomain "legalgo-BE-go/internal/domain/category"
|
||||
|
||||
func (i *impl) Create(spec categorydomain.CategoryReq) error {
|
||||
return i.categoryRepo.Create(spec)
|
||||
return i.categoryRepo.CreateModel(spec)
|
||||
// return i.categoryRepo.Create(spec)
|
||||
}
|
||||
|
||||
9
internal/services/category/get_all_model.go
Normal file
9
internal/services/category/get_all_model.go
Normal file
@ -0,0 +1,9 @@
|
||||
package categorysvc
|
||||
|
||||
import (
|
||||
"legalgo-BE-go/database"
|
||||
)
|
||||
|
||||
func (i *impl) GetAllModel() ([]database.CategoryModel, error) {
|
||||
return i.categoryRepo.GetAllModel()
|
||||
}
|
||||
@ -1,6 +1,7 @@
|
||||
package categorysvc
|
||||
|
||||
import (
|
||||
"legalgo-BE-go/database"
|
||||
categoryrepository "legalgo-BE-go/internal/accessor/category"
|
||||
categorydomain "legalgo-BE-go/internal/domain/category"
|
||||
)
|
||||
@ -12,6 +13,7 @@ type impl struct {
|
||||
type Category interface {
|
||||
Create(categorydomain.CategoryReq) error
|
||||
GetAll() ([]categorydomain.Category, error)
|
||||
GetAllModel() ([]database.CategoryModel, error)
|
||||
}
|
||||
|
||||
func New(
|
||||
|
||||
@ -3,6 +3,7 @@ package services
|
||||
import (
|
||||
serviceauth "legalgo-BE-go/internal/services/auth"
|
||||
categorysvc "legalgo-BE-go/internal/services/category"
|
||||
newssvc "legalgo-BE-go/internal/services/news"
|
||||
subscribesvc "legalgo-BE-go/internal/services/subscribe"
|
||||
subscribeplansvc "legalgo-BE-go/internal/services/subscribe_plan"
|
||||
tagsvc "legalgo-BE-go/internal/services/tag"
|
||||
@ -17,5 +18,6 @@ var Module = fx.Module("services",
|
||||
subscribesvc.New,
|
||||
tagsvc.New,
|
||||
categorysvc.New,
|
||||
newssvc.New,
|
||||
),
|
||||
)
|
||||
|
||||
47
internal/services/news/create.go
Normal file
47
internal/services/news/create.go
Normal 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
|
||||
}
|
||||
48
internal/services/news/create_model.go
Normal file
48
internal/services/news/create_model.go
Normal 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
|
||||
}
|
||||
7
internal/services/news/get_all.go
Normal file
7
internal/services/news/get_all.go
Normal 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()
|
||||
}
|
||||
9
internal/services/news/get_all_model.go
Normal file
9
internal/services/news/get_all_model.go
Normal file
@ -0,0 +1,9 @@
|
||||
package newssvc
|
||||
|
||||
import (
|
||||
"legalgo-BE-go/database"
|
||||
)
|
||||
|
||||
func (i *impl) GetAllModel() ([]database.NewsModel, error) {
|
||||
return i.newsRepo.GetAllModel()
|
||||
}
|
||||
34
internal/services/news/impl.go
Normal file
34
internal/services/news/impl.go
Normal 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,
|
||||
}
|
||||
}
|
||||
@ -3,5 +3,6 @@ package tagsvc
|
||||
import tagdomain "legalgo-BE-go/internal/domain/tag"
|
||||
|
||||
func (i *impl) Create(spec tagdomain.TagReq) error {
|
||||
return i.tagRepo.Create(spec)
|
||||
// return i.tagRepo.Create(spec)
|
||||
return i.tagRepo.CreateModel(spec)
|
||||
}
|
||||
|
||||
9
internal/services/tag/get_all_model.go
Normal file
9
internal/services/tag/get_all_model.go
Normal file
@ -0,0 +1,9 @@
|
||||
package tagsvc
|
||||
|
||||
import (
|
||||
"legalgo-BE-go/database"
|
||||
)
|
||||
|
||||
func (i *impl) GetAllModel() ([]database.TagModel, error) {
|
||||
return i.tagRepo.GetAllModel()
|
||||
}
|
||||
@ -1,6 +1,7 @@
|
||||
package tagsvc
|
||||
|
||||
import (
|
||||
"legalgo-BE-go/database"
|
||||
tagrepository "legalgo-BE-go/internal/accessor/tag"
|
||||
tagdomain "legalgo-BE-go/internal/domain/tag"
|
||||
)
|
||||
@ -12,6 +13,7 @@ type impl struct {
|
||||
type TagIntf interface {
|
||||
Create(tagdomain.TagReq) error
|
||||
GetAll() ([]tagdomain.Tag, error)
|
||||
GetAllModel() ([]database.TagModel, error)
|
||||
}
|
||||
|
||||
func New(
|
||||
|
||||
34
internal/utilities/utils/get_token_detail.go
Normal file
34
internal/utilities/utils/get_token_detail.go
Normal 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
|
||||
}
|
||||
190
openapi.yml
190
openapi.yml
@ -10,13 +10,6 @@ paths:
|
||||
summary: "get staff profile"
|
||||
tags:
|
||||
- Staff
|
||||
parameters:
|
||||
- name: "id"
|
||||
in: query
|
||||
description: "staff id"
|
||||
required: true
|
||||
schema:
|
||||
type: string
|
||||
responses:
|
||||
"200":
|
||||
description: Success get profile
|
||||
@ -169,13 +162,6 @@ paths:
|
||||
summary: "get staff profile"
|
||||
tags:
|
||||
- User
|
||||
parameters:
|
||||
- name: "id"
|
||||
in: query
|
||||
description: "user id"
|
||||
required: true
|
||||
schema:
|
||||
type: string
|
||||
responses:
|
||||
"200":
|
||||
description: Success get profile
|
||||
@ -574,7 +560,7 @@ paths:
|
||||
post:
|
||||
summary: Create a new category
|
||||
tags:
|
||||
- Tags
|
||||
- Category
|
||||
requestBody:
|
||||
required: true
|
||||
content:
|
||||
@ -631,3 +617,177 @@ paths:
|
||||
type: string
|
||||
message:
|
||||
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
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user