✨ Mixed feed API
This commit is contained in:
parent
02d6801a7f
commit
e836a97435
@ -2,23 +2,36 @@ package api
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"time"
|
||||||
|
|
||||||
"git.solsynth.dev/hydrogen/interactive/pkg/internal/database"
|
"git.solsynth.dev/hydrogen/interactive/pkg/internal/database"
|
||||||
"git.solsynth.dev/hydrogen/interactive/pkg/internal/models"
|
"git.solsynth.dev/hydrogen/interactive/pkg/internal/models"
|
||||||
"git.solsynth.dev/hydrogen/interactive/pkg/internal/services"
|
"git.solsynth.dev/hydrogen/interactive/pkg/internal/services"
|
||||||
"github.com/gofiber/fiber/v2"
|
"github.com/gofiber/fiber/v2"
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type FeedRecord struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Data map[string]any `json:"data"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
}
|
||||||
|
|
||||||
func listFeed(c *fiber.Ctx) error {
|
func listFeed(c *fiber.Ctx) error {
|
||||||
take := c.QueryInt("take", 0)
|
take := c.QueryInt("take", 0)
|
||||||
offset := c.QueryInt("offset", 0)
|
offset := c.QueryInt("offset", 0)
|
||||||
realmId := c.QueryInt("realmId", 0)
|
realmId := c.QueryInt("realmId", 0)
|
||||||
|
|
||||||
tx := database.C
|
postTx := services.FilterPostDraft(database.C)
|
||||||
|
articleTx := services.FilterArticleDraft(database.C)
|
||||||
|
|
||||||
if realmId > 0 {
|
if realmId > 0 {
|
||||||
if realm, err := services.GetRealmWithExtID(uint(realmId)); err != nil {
|
if realm, err := services.GetRealmWithExtID(uint(realmId)); err != nil {
|
||||||
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("realm was not found: %v", err))
|
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("realm was not found: %v", err))
|
||||||
} else {
|
} else {
|
||||||
tx = services.FilterPostWithRealm(tx, realm.ID)
|
postTx = services.FilterPostWithRealm(postTx, realm.ID)
|
||||||
|
articleTx = services.FilterArticleWithRealm(articleTx, realm.ID)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -27,28 +40,80 @@ func listFeed(c *fiber.Ctx) error {
|
|||||||
if err := database.C.Where(&models.Account{Name: c.Query("authorId")}).First(&author).Error; err != nil {
|
if err := database.C.Where(&models.Account{Name: c.Query("authorId")}).First(&author).Error; err != nil {
|
||||||
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
}
|
}
|
||||||
tx = tx.Where("author_id = ?", author.ID)
|
postTx = postTx.Where("author_id = ?", author.ID)
|
||||||
|
articleTx = articleTx.Where("author_id = ?", author.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(c.Query("category")) > 0 {
|
if len(c.Query("category")) > 0 {
|
||||||
tx = services.FilterPostWithCategory(tx, c.Query("category"))
|
postTx = services.FilterPostWithCategory(postTx, c.Query("category"))
|
||||||
|
articleTx = services.FilterArticleWithCategory(articleTx, c.Query("category"))
|
||||||
}
|
}
|
||||||
if len(c.Query("tag")) > 0 {
|
if len(c.Query("tag")) > 0 {
|
||||||
tx = services.FilterPostWithTag(tx, c.Query("tag"))
|
postTx = services.FilterPostWithTag(postTx, c.Query("tag"))
|
||||||
|
articleTx = services.FilterArticleWithTag(articleTx, c.Query("tag"))
|
||||||
}
|
}
|
||||||
|
|
||||||
count, err := services.CountPost(tx)
|
postCountTx := postTx
|
||||||
|
articleCountTx := articleTx
|
||||||
|
|
||||||
|
postCount, err := services.CountPost(postCountTx)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusInternalServerError, err.Error())
|
||||||
|
}
|
||||||
|
articleCount, err := services.CountArticle(articleCountTx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fiber.NewError(fiber.StatusInternalServerError, err.Error())
|
return fiber.NewError(fiber.StatusInternalServerError, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
items, err := services.ListPost(tx, take, offset)
|
postItems, err := services.ListPost(postTx, take, offset)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
}
|
||||||
|
articleItems, err := services.ListArticle(articleTx, take, offset)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var feed []FeedRecord
|
||||||
|
|
||||||
|
encodeToFeed := func(t string, in any, createdAt time.Time) FeedRecord {
|
||||||
|
var result map[string]any
|
||||||
|
raw, _ := jsoniter.Marshal(in)
|
||||||
|
jsoniter.Unmarshal(raw, &result)
|
||||||
|
|
||||||
|
return FeedRecord{
|
||||||
|
Type: t,
|
||||||
|
Data: result,
|
||||||
|
CreatedAt: createdAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, post := range postItems {
|
||||||
|
feed = append(feed, encodeToFeed("post", post, post.CreatedAt))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, article := range articleItems {
|
||||||
|
feed = append(feed, encodeToFeed("article", article, article.CreatedAt))
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Slice(feed, func(i, j int) bool {
|
||||||
|
return feed[i].CreatedAt.After(feed[j].CreatedAt)
|
||||||
|
})
|
||||||
|
|
||||||
|
start := offset
|
||||||
|
end := start + take
|
||||||
|
if start > len(feed) {
|
||||||
|
return c.JSON(fiber.Map{
|
||||||
|
"count": postCount + articleCount,
|
||||||
|
"data": []FeedRecord{},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if end > len(feed) {
|
||||||
|
end = len(feed)
|
||||||
|
}
|
||||||
|
|
||||||
return c.JSON(fiber.Map{
|
return c.JSON(fiber.Map{
|
||||||
"count": count,
|
"count": postCount + articleCount,
|
||||||
"data": items,
|
"data": feed[start:end],
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user