Post channel modes

This commit is contained in:
LittleSheep 2025-02-22 13:29:42 +08:00
parent be1beb713c
commit a3cb407d89
4 changed files with 34 additions and 10 deletions

View File

@ -26,9 +26,9 @@ func getPostInsight(c *fiber.Ctx) error {
tx := services.FilterPostDraft(database.C) tx := services.FilterPostDraft(database.C)
if user, authenticated := c.Locals("user").(authm.Account); authenticated { if user, authenticated := c.Locals("user").(authm.Account); authenticated {
tx = services.FilterPostWithUserContext(tx, &user, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, &user)
} else { } else {
tx = services.FilterPostWithUserContext(tx, nil, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, nil)
} }
if numericId, paramErr := strconv.Atoi(id); paramErr == nil { if numericId, paramErr := strconv.Atoi(id); paramErr == nil {

View File

@ -24,9 +24,9 @@ func UniversalPostFilter(c *fiber.Ctx, tx *gorm.DB) (*gorm.DB, error) {
tx = services.FilterPostDraft(tx) tx = services.FilterPostDraft(tx)
if user, authenticated := c.Locals("user").(authm.Account); authenticated { if user, authenticated := c.Locals("user").(authm.Account); authenticated {
tx = services.FilterPostWithUserContext(tx, &user, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, &user)
} else { } else {
tx = services.FilterPostWithUserContext(tx, nil, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, nil)
} }
if c.QueryBool("noReply", true) { if c.QueryBool("noReply", true) {
@ -71,9 +71,9 @@ func getPost(c *fiber.Ctx) error {
tx := services.FilterPostDraft(database.C) tx := services.FilterPostDraft(database.C)
if user, authenticated := c.Locals("user").(authm.Account); authenticated { if user, authenticated := c.Locals("user").(authm.Account); authenticated {
tx = services.FilterPostWithUserContext(tx, &user, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, &user)
} else { } else {
tx = services.FilterPostWithUserContext(tx, nil, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, nil)
} }
if numericId, paramErr := strconv.Atoi(id); paramErr == nil { if numericId, paramErr := strconv.Atoi(id); paramErr == nil {

View File

@ -20,7 +20,7 @@ func getWhatsNew(c *fiber.Ctx) error {
} }
tx := services.FilterPostDraft(database.C) tx := services.FilterPostDraft(database.C)
tx = services.FilterPostWithUserContext(tx, &user, len(c.Query("realm")) > 0) tx = services.FilterPostWithUserContext(c, tx, &user)
tx = tx.Where("id > ?", pivot) tx = tx.Where("id > ?", pivot)

View File

@ -5,6 +5,7 @@ import (
"errors" "errors"
"fmt" "fmt"
"git.solsynth.dev/hypernet/nexus/pkg/nex" "git.solsynth.dev/hypernet/nexus/pkg/nex"
"github.com/gofiber/fiber/v2"
"regexp" "regexp"
"strconv" "strconv"
"strings" "strings"
@ -29,7 +30,7 @@ import (
"gorm.io/gorm" "gorm.io/gorm"
) )
func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool) *gorm.DB { func FilterPostWithUserContext(c *fiber.Ctx, tx *gorm.DB, user *authm.Account) *gorm.DB {
if user == nil { if user == nil {
return tx.Where("visibility = ? AND realm_id IS NULL", models.PostVisibilityAll) return tx.Where("visibility = ? AND realm_id IS NULL", models.PostVisibilityAll)
} }
@ -44,6 +45,7 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
type userContextState struct { type userContextState struct {
Allowlist []uint `json:"allow"` Allowlist []uint `json:"allow"`
InvisibleList []uint `json:"invisible"` InvisibleList []uint `json:"invisible"`
FollowList []uint `json:"follow"`
RealmList []uint `json:"realm"` RealmList []uint `json:"realm"`
} }
@ -51,7 +53,7 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
marshal := marshaler.New(cacheManager) marshal := marshaler.New(cacheManager)
ctx := context.Background() ctx := context.Background()
var allowlist, invisibleList, realmList []uint var allowlist, invisibleList, followList, realmList []uint
statusCacheKey := fmt.Sprintf("post-user-context-query#%d", user.ID) statusCacheKey := fmt.Sprintf("post-user-context-query#%d", user.ID)
statusCache, err := marshal.Get(ctx, statusCacheKey, new(userContextState)) statusCache, err := marshal.Get(ctx, statusCacheKey, new(userContextState))
@ -59,6 +61,7 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
state := statusCache.(*userContextState) state := statusCache.(*userContextState)
allowlist = state.Allowlist allowlist = state.Allowlist
invisibleList = state.InvisibleList invisibleList = state.InvisibleList
followList = state.FollowList
realmList = state.RealmList realmList = state.RealmList
} else { } else {
// Getting the relationships // Getting the relationships
@ -106,6 +109,19 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
models.PublisherTypePersonal, models.PublisherTypePersonal,
).Find(&publishers) ).Find(&publishers)
// Getting the follow list
{
var subs []models.Subscription
if err := database.C.Where("follower_id = ? AND account_id IS NOT NULL", user.ID).Find(&subs).Error; err != nil {
log.Error().Err(err).Msg("An error occurred when getting subscriptions...")
}
followList = lo.Map(lo.Filter(subs, func(item models.Subscription, index int) bool {
return item.AccountID != nil
}), func(item models.Subscription, index int) uint {
return *item.AccountID
})
}
allowlist = lo.Map(lo.Filter(publishers, func(item models.Publisher, index int) bool { allowlist = lo.Map(lo.Filter(publishers, func(item models.Publisher, index int) bool {
if item.AccountID == nil { if item.AccountID == nil {
return false return false
@ -130,6 +146,7 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
Allowlist: allowlist, Allowlist: allowlist,
InvisibleList: invisibleList, InvisibleList: invisibleList,
RealmList: realmList, RealmList: realmList,
FollowList: followList,
}, },
store.WithExpiration(2*time.Minute), store.WithExpiration(2*time.Minute),
store.WithTags([]string{"post-user-context-query", fmt.Sprintf("user#%d", user.ID)}), store.WithTags([]string{"post-user-context-query", fmt.Sprintf("user#%d", user.ID)}),
@ -154,7 +171,7 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
if len(invisibleList) > 0 { if len(invisibleList) > 0 {
tx = tx.Where("publisher_id NOT IN ?", invisibleList) tx = tx.Where("publisher_id NOT IN ?", invisibleList)
} }
if !withRealm { if len(c.Query("realm")) == 0 {
if len(realmList) > 0 { if len(realmList) > 0 {
tx = tx.Where("realm_id IN ? OR realm_id IS NULL", realmList) tx = tx.Where("realm_id IN ? OR realm_id IS NULL", realmList)
} else { } else {
@ -162,6 +179,13 @@ func FilterPostWithUserContext(tx *gorm.DB, user *authm.Account, withRealm bool)
} }
} }
switch c.Query("channel") {
case "friends":
tx = tx.Where("publisher_id IN ?", allowlist)
case "following":
tx = tx.Where("publisher_id IN ?", followList)
}
return tx return tx
} }