2024-09-16 16:12:09 +00:00
|
|
|
package services
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2024-12-15 15:50:54 +00:00
|
|
|
|
2024-11-02 05:41:51 +00:00
|
|
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/database"
|
|
|
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/gap"
|
|
|
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/models"
|
2024-12-15 15:50:54 +00:00
|
|
|
"git.solsynth.dev/hypernet/nexus/pkg/proto"
|
2024-10-31 14:41:32 +00:00
|
|
|
"git.solsynth.dev/hypernet/passport/pkg/authkit"
|
|
|
|
authm "git.solsynth.dev/hypernet/passport/pkg/authkit/models"
|
|
|
|
"git.solsynth.dev/hypernet/pusher/pkg/pushkit"
|
2024-12-15 15:50:54 +00:00
|
|
|
"github.com/samber/lo"
|
2024-09-16 16:12:09 +00:00
|
|
|
"gorm.io/gorm"
|
|
|
|
)
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func GetSubscriptionOnUser(user authm.Account, target models.Publisher) (*models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND account_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("unable to get subscription: %v", err)
|
|
|
|
}
|
|
|
|
return &subscription, nil
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func GetSubscriptionOnTag(user authm.Account, target models.Tag) (*models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND tag_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("unable to get subscription: %v", err)
|
|
|
|
}
|
|
|
|
return &subscription, nil
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func GetSubscriptionOnCategory(user authm.Account, target models.Category) (*models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND category_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("unable to get subscription: %v", err)
|
|
|
|
}
|
|
|
|
return &subscription, nil
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func SubscribeToUser(user authm.Account, target models.Publisher) (models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND account_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
2024-09-16 17:43:01 +00:00
|
|
|
if !errors.Is(err, gorm.ErrRecordNotFound) {
|
2024-09-16 16:12:09 +00:00
|
|
|
return subscription, fmt.Errorf("subscription already exists")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
subscription = models.Subscription{
|
|
|
|
FollowerID: user.ID,
|
|
|
|
AccountID: &target.ID,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Save(&subscription).Error
|
|
|
|
return subscription, err
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func SubscribeToTag(user authm.Account, target models.Tag) (models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND tag_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
2024-09-16 17:43:01 +00:00
|
|
|
if !errors.Is(err, gorm.ErrRecordNotFound) {
|
2024-09-16 16:12:09 +00:00
|
|
|
return subscription, fmt.Errorf("subscription already exists")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
subscription = models.Subscription{
|
|
|
|
FollowerID: user.ID,
|
|
|
|
TagID: &target.ID,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Save(&subscription).Error
|
|
|
|
return subscription, err
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func SubscribeToCategory(user authm.Account, target models.Category) (models.Subscription, error) {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND category_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
2024-09-16 17:43:01 +00:00
|
|
|
if !errors.Is(err, gorm.ErrRecordNotFound) {
|
2024-09-16 16:12:09 +00:00
|
|
|
return subscription, fmt.Errorf("subscription already exists")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
subscription = models.Subscription{
|
|
|
|
FollowerID: user.ID,
|
|
|
|
CategoryID: &target.ID,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Save(&subscription).Error
|
|
|
|
return subscription, err
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func UnsubscribeFromUser(user authm.Account, target models.Publisher) error {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND account_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return fmt.Errorf("subscription does not exist")
|
|
|
|
}
|
|
|
|
return fmt.Errorf("unable to check subscription is exists or not: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Delete(&subscription).Error
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func UnsubscribeFromTag(user authm.Account, target models.Tag) error {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND tag_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return fmt.Errorf("subscription does not exist")
|
|
|
|
}
|
|
|
|
return fmt.Errorf("unable to check subscription is exists or not: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Delete(&subscription).Error
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
func UnsubscribeFromCategory(user authm.Account, target models.Category) error {
|
2024-09-16 16:12:09 +00:00
|
|
|
var subscription models.Subscription
|
|
|
|
if err := database.C.Where("follower_id = ? AND category_id = ?", user.ID, target.ID).First(&subscription).Error; err != nil {
|
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
|
|
return fmt.Errorf("subscription does not exist")
|
|
|
|
}
|
|
|
|
return fmt.Errorf("unable to check subscription is exists or not: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := database.C.Delete(&subscription).Error
|
|
|
|
return err
|
|
|
|
}
|
2024-09-16 16:35:42 +00:00
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
func NotifyUserSubscription(poster models.Publisher, item models.Post, content string, title *string) error {
|
|
|
|
if item.Visibility == models.PostVisibilityNone {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-16 16:35:42 +00:00
|
|
|
var subscriptions []models.Subscription
|
|
|
|
if err := database.C.Where("account_id = ?", poster.ID).Preload("Follower").Find(&subscriptions).Error; err != nil {
|
|
|
|
return fmt.Errorf("unable to get subscriptions: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nTitle := fmt.Sprintf("New post from %s (%s)", poster.Nick, poster.Name)
|
|
|
|
nSubtitle := "From your subscription"
|
|
|
|
|
2024-10-14 13:40:28 +00:00
|
|
|
body := TruncatePostContentShort(content)
|
2024-09-16 16:35:42 +00:00
|
|
|
if title != nil {
|
|
|
|
body = fmt.Sprintf("%s\n%s", *title, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
userIDs := make([]uint64, 0, len(subscriptions))
|
|
|
|
for _, subscription := range subscriptions {
|
|
|
|
userIDs = append(userIDs, uint64(subscription.Follower.ID))
|
|
|
|
}
|
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
if item.Visibility == models.PostVisibilitySelected {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return lo.Contains(item.VisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if item.Visibility == models.PostVisibilityFiltered {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(item.InvisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if invisibleList := ListPostInvisibleUser(poster, item.Visibility); len(invisibleList) > 0 {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(invisibleList, uint(entry))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
err := authkit.NotifyUserBatch(gap.Nx, userIDs, pushkit.Notification{
|
|
|
|
Topic: "interactive.subscription",
|
|
|
|
Title: nTitle,
|
|
|
|
Subtitle: nSubtitle,
|
|
|
|
Body: body,
|
|
|
|
Priority: 3,
|
2024-09-16 16:35:42 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
func NotifyTagSubscription(poster models.Tag, og models.Publisher, item models.Post, content string, title *string) error {
|
|
|
|
if item.Visibility == models.PostVisibilityNone {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-16 16:35:42 +00:00
|
|
|
var subscriptions []models.Subscription
|
|
|
|
if err := database.C.Where("tag_id = ?", poster.ID).Preload("Follower").Find(&subscriptions).Error; err != nil {
|
|
|
|
return fmt.Errorf("unable to get subscriptions: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nTitle := fmt.Sprintf("New post in %s by %s (%s)", poster.Name, og.Nick, og.Name)
|
|
|
|
nSubtitle := "From your subscription"
|
|
|
|
|
2024-10-14 13:40:28 +00:00
|
|
|
body := TruncatePostContentShort(content)
|
2024-09-16 16:35:42 +00:00
|
|
|
if title != nil {
|
|
|
|
body = fmt.Sprintf("%s\n%s", *title, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
userIDs := make([]uint64, 0, len(subscriptions))
|
|
|
|
for _, subscription := range subscriptions {
|
|
|
|
userIDs = append(userIDs, uint64(subscription.Follower.ID))
|
|
|
|
}
|
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
if item.Visibility == models.PostVisibilitySelected {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return lo.Contains(item.VisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if item.Visibility == models.PostVisibilityFiltered {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(item.InvisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if invisibleList := ListPostInvisibleUser(item.Publisher, item.Visibility); len(invisibleList) > 0 {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(invisibleList, uint(entry))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
err := authkit.NotifyUserBatch(gap.Nx, userIDs, pushkit.Notification{
|
|
|
|
Topic: "interactive.subscription",
|
|
|
|
Title: nTitle,
|
|
|
|
Subtitle: nSubtitle,
|
|
|
|
Body: body,
|
|
|
|
Priority: 3,
|
2024-09-16 16:35:42 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
func NotifyCategorySubscription(poster models.Category, og models.Publisher, item models.Post, content string, title *string) error {
|
|
|
|
if item.Visibility == models.PostVisibilityNone {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-16 16:35:42 +00:00
|
|
|
var subscriptions []models.Subscription
|
|
|
|
if err := database.C.Where("category_id = ?", poster.ID).Preload("Follower").Find(&subscriptions).Error; err != nil {
|
|
|
|
return fmt.Errorf("unable to get subscriptions: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nTitle := fmt.Sprintf("New post in %s by %s (%s)", poster.Name, og.Nick, og.Name)
|
|
|
|
nSubtitle := "From your subscription"
|
|
|
|
|
2024-10-14 13:40:28 +00:00
|
|
|
body := TruncatePostContentShort(content)
|
|
|
|
if title != nil {
|
|
|
|
body = fmt.Sprintf("%s\n%s", *title, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
userIDs := make([]uint64, 0, len(subscriptions))
|
|
|
|
for _, subscription := range subscriptions {
|
|
|
|
userIDs = append(userIDs, uint64(subscription.Follower.ID))
|
2024-09-16 16:35:42 +00:00
|
|
|
}
|
2024-10-14 13:40:28 +00:00
|
|
|
|
2024-12-15 15:50:54 +00:00
|
|
|
if item.Visibility == models.PostVisibilitySelected {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return lo.Contains(item.VisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if item.Visibility == models.PostVisibilityFiltered {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(item.InvisibleUsers, uint(entry))
|
|
|
|
})
|
|
|
|
} else if invisibleList := ListPostInvisibleUser(item.Publisher, item.Visibility); len(invisibleList) > 0 {
|
|
|
|
userIDs = lo.Filter(userIDs, func(entry uint64, index int) bool {
|
|
|
|
return !lo.Contains(invisibleList, uint(entry))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-10-31 14:41:32 +00:00
|
|
|
err := authkit.NotifyUserBatch(gap.Nx, userIDs, pushkit.Notification{
|
|
|
|
Topic: "interactive.subscription",
|
|
|
|
Title: nTitle,
|
|
|
|
Subtitle: nSubtitle,
|
|
|
|
Body: body,
|
|
|
|
Priority: 3,
|
2024-09-16 16:35:42 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
2024-12-15 15:50:54 +00:00
|
|
|
|
|
|
|
// ListPostInvisibleUser will return a list of users which should not be notified the post.
|
|
|
|
// NOTICE If the visibility is PostVisibilitySelected, PostVisibilityFiltered or PostVisibilityNone, you need do extra steps to filter users
|
|
|
|
// WARNING This function won't use cache, be careful of the queries
|
|
|
|
func ListPostInvisibleUser(og models.Publisher, visibility models.PostVisibilityLevel) []uint {
|
|
|
|
switch visibility {
|
|
|
|
case models.PostVisibilityAll:
|
|
|
|
return []uint{}
|
|
|
|
case models.PostVisibilityFriends:
|
|
|
|
if og.AccountID == nil {
|
|
|
|
return []uint{}
|
|
|
|
}
|
|
|
|
userFriends, _ := authkit.ListRelative(gap.Nx, *og.AccountID, int32(authm.RelationshipFriend), true)
|
|
|
|
return lo.Map(userFriends, func(item *proto.UserInfo, index int) uint {
|
|
|
|
return uint(item.GetId())
|
|
|
|
})
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|