|
|
|
// Copyright 2022 ROC. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package base
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/alimy/mir/v4"
|
|
|
|
"github.com/cockroachdb/errors"
|
|
|
|
"github.com/getsentry/sentry-go"
|
|
|
|
sentrygin "github.com/getsentry/sentry-go/gin"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/conf"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/core"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/core/cs"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/core/ms"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/dao"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/dao/cache"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/events"
|
|
|
|
"github.com/rocboss/paopao-ce/internal/model/joint"
|
|
|
|
"github.com/rocboss/paopao-ce/pkg/app"
|
|
|
|
"github.com/rocboss/paopao-ce/pkg/xerror"
|
|
|
|
)
|
|
|
|
|
|
|
|
type BaseServant struct {
|
|
|
|
bindAny func(c *gin.Context, obj any) mir.Error
|
|
|
|
}
|
|
|
|
|
|
|
|
type DaoServant struct {
|
|
|
|
*BaseServant
|
|
|
|
|
|
|
|
Dsa core.WebDataServantA
|
|
|
|
Ds core.DataService
|
|
|
|
Ts core.TweetSearchService
|
|
|
|
Redis core.RedisCache
|
|
|
|
}
|
|
|
|
|
|
|
|
type SentryHubSetter interface {
|
|
|
|
SetSentryHub(hub *sentry.Hub)
|
|
|
|
}
|
|
|
|
|
|
|
|
type UserSetter interface {
|
|
|
|
SetUser(*ms.User)
|
|
|
|
}
|
|
|
|
|
|
|
|
type UserIdSetter interface {
|
|
|
|
SetUserId(int64)
|
|
|
|
}
|
|
|
|
|
|
|
|
type PageInfoSetter interface {
|
|
|
|
SetPageInfo(page, pageSize int)
|
|
|
|
}
|
|
|
|
|
|
|
|
func UserFrom(c *gin.Context) (*ms.User, bool) {
|
|
|
|
if u, exists := c.Get("USER"); exists {
|
|
|
|
user, ok := u.(*ms.User)
|
|
|
|
return user, ok
|
|
|
|
}
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func UserIdFrom(c *gin.Context) (int64, bool) {
|
|
|
|
if uid, exists := c.Get("UID"); exists {
|
|
|
|
v, ok := uid.(int64)
|
|
|
|
return v, ok
|
|
|
|
}
|
|
|
|
return -1, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func UserNameFrom(c *gin.Context) (string, bool) {
|
|
|
|
if username, exists := c.Get("USERNAME"); exists {
|
|
|
|
v, ok := username.(string)
|
|
|
|
return v, ok
|
|
|
|
}
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
|
|
|
func bindAny(c *gin.Context, obj any) mir.Error {
|
|
|
|
var errs xerror.ValidErrors
|
|
|
|
err := c.ShouldBind(obj)
|
|
|
|
if err != nil {
|
|
|
|
return mir.NewError(xerror.InvalidParams.StatusCode(), xerror.InvalidParams.WithDetails(errs.Error()))
|
|
|
|
}
|
|
|
|
// setup *core.User if needed
|
|
|
|
if setter, ok := obj.(UserSetter); ok {
|
|
|
|
user, _ := UserFrom(c)
|
|
|
|
setter.SetUser(user)
|
|
|
|
}
|
|
|
|
// setup UserId if needed
|
|
|
|
if setter, ok := obj.(UserIdSetter); ok {
|
|
|
|
uid, _ := UserIdFrom(c)
|
|
|
|
setter.SetUserId(uid)
|
|
|
|
}
|
|
|
|
// setup PageInfo if needed
|
|
|
|
if setter, ok := obj.(PageInfoSetter); ok {
|
|
|
|
page, pageSize := app.GetPageInfo(c)
|
|
|
|
setter.SetPageInfo(page, pageSize)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func bindAnySentry(c *gin.Context, obj any) mir.Error {
|
|
|
|
hub := sentrygin.GetHubFromContext(c)
|
|
|
|
var errs xerror.ValidErrors
|
|
|
|
err := c.ShouldBind(obj)
|
|
|
|
if err != nil {
|
|
|
|
xerr := mir.NewError(xerror.InvalidParams.StatusCode(), xerror.InvalidParams.WithDetails(errs.Error()))
|
|
|
|
if hub != nil {
|
|
|
|
hub.CaptureException(errors.Wrap(xerr, "bind object"))
|
|
|
|
}
|
|
|
|
return xerr
|
|
|
|
}
|
|
|
|
// setup sentry hub if needed
|
|
|
|
if setter, ok := obj.(SentryHubSetter); ok && hub != nil {
|
|
|
|
setter.SetSentryHub(hub)
|
|
|
|
}
|
|
|
|
// setup *core.User if needed
|
|
|
|
if setter, ok := obj.(UserSetter); ok {
|
|
|
|
user, _ := UserFrom(c)
|
|
|
|
setter.SetUser(user)
|
|
|
|
}
|
|
|
|
// setup UserId if needed
|
|
|
|
if setter, ok := obj.(UserIdSetter); ok {
|
|
|
|
uid, _ := UserIdFrom(c)
|
|
|
|
setter.SetUserId(uid)
|
|
|
|
}
|
|
|
|
// setup PageInfo if needed
|
|
|
|
if setter, ok := obj.(PageInfoSetter); ok {
|
|
|
|
page, pageSize := app.GetPageInfo(c)
|
|
|
|
setter.SetPageInfo(page, pageSize)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func RenderAny(c *gin.Context, data any, err mir.Error) {
|
|
|
|
if err == nil {
|
|
|
|
c.JSON(http.StatusOK, &joint.JsonResp{
|
|
|
|
Code: 0,
|
|
|
|
Msg: "success",
|
|
|
|
Data: data,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
c.JSON(xerror.HttpStatusCode(err), &joint.JsonResp{
|
|
|
|
Code: err.StatusCode(),
|
|
|
|
Msg: err.Error(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *BaseServant) Bind(c *gin.Context, obj any) mir.Error {
|
|
|
|
return s.bindAny(c, obj)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *BaseServant) Render(c *gin.Context, data any, err mir.Error) {
|
|
|
|
if err == nil {
|
|
|
|
c.JSON(http.StatusOK, &joint.JsonResp{
|
|
|
|
Code: 0,
|
|
|
|
Msg: "success",
|
|
|
|
Data: data,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
c.JSON(xerror.HttpStatusCode(err), &joint.JsonResp{
|
|
|
|
Code: err.StatusCode(),
|
|
|
|
Msg: err.Error(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) GetTweetBy(id int64) (*ms.PostFormated, error) {
|
|
|
|
post, err := s.Ds.GetPostByID(id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
postContents, err := s.Ds.GetPostContentsByIDs([]int64{post.ID})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
users, err := s.Ds.GetUsersByIDs([]int64{post.UserID})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// 数据整合
|
|
|
|
postFormated := post.Format()
|
|
|
|
for _, user := range users {
|
|
|
|
postFormated.User = user.Format()
|
|
|
|
}
|
|
|
|
for _, content := range postContents {
|
|
|
|
if content.PostID == post.ID {
|
|
|
|
postFormated.Contents = append(postFormated.Contents, content.Format())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return postFormated, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) PushAllPostToSearch() {
|
|
|
|
events.OnEvent(&pushAllPostToSearchEvent{
|
|
|
|
fn: s.pushAllPostToSearch,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) pushAllPostToSearch() error {
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := s.Redis.SetPushToSearchJob(ctx); err == nil {
|
|
|
|
defer s.Redis.DelPushToSearchJob(ctx)
|
|
|
|
|
|
|
|
splitNum := 1000
|
|
|
|
conditions := ms.ConditionsT{
|
|
|
|
"visibility IN ?": []core.PostVisibleT{core.PostVisitPublic, core.PostVisitFriend},
|
|
|
|
}
|
|
|
|
totalRows, _ := s.Ds.GetPostCount(conditions)
|
|
|
|
pages := math.Ceil(float64(totalRows) / float64(splitNum))
|
|
|
|
nums := int(pages)
|
|
|
|
for i := 0; i < nums; i++ {
|
|
|
|
posts, postsFormated, err := s.GetTweetList(conditions, i*splitNum, splitNum)
|
|
|
|
if err != nil || len(posts) != len(postsFormated) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for i, pf := range postsFormated {
|
|
|
|
contentFormated := ""
|
|
|
|
for _, content := range pf.Contents {
|
|
|
|
if content.Type == ms.ContentTypeText || content.Type == ms.ContentTypeTitle {
|
|
|
|
contentFormated = contentFormated + content.Content + "\n"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
docs := []core.TsDocItem{{
|
|
|
|
Post: posts[i],
|
|
|
|
Content: contentFormated,
|
|
|
|
}}
|
|
|
|
s.Ts.AddDocuments(docs, fmt.Sprintf("%d", posts[i].ID))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("redis: set JOB_PUSH_TO_SEARCH error: %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) PushPostToSearch(post *ms.Post) {
|
|
|
|
events.OnEvent(&pushPostToSearchEvent{
|
|
|
|
fn: s.pushPostToSearch,
|
|
|
|
post: post,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) pushPostToSearch(post *ms.Post) {
|
|
|
|
postFormated := post.Format()
|
|
|
|
postFormated.User = &ms.UserFormated{
|
|
|
|
ID: post.UserID,
|
|
|
|
}
|
|
|
|
contents, _ := s.Ds.GetPostContentsByIDs([]int64{post.ID})
|
|
|
|
for _, content := range contents {
|
|
|
|
postFormated.Contents = append(postFormated.Contents, content.Format())
|
|
|
|
}
|
|
|
|
contentFormated := ""
|
|
|
|
for _, content := range postFormated.Contents {
|
|
|
|
if content.Type == ms.ContentTypeText || content.Type == ms.ContentTypeTitle {
|
|
|
|
contentFormated = contentFormated + content.Content + "\n"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
docs := []core.TsDocItem{{
|
|
|
|
Post: post,
|
|
|
|
Content: contentFormated,
|
|
|
|
}}
|
|
|
|
s.Ts.AddDocuments(docs, fmt.Sprintf("%d", post.ID))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) DeleteSearchPost(post *ms.Post) error {
|
|
|
|
return s.Ts.DeleteDocuments([]string{fmt.Sprintf("%d", post.ID)})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) GetTweetList(conditions ms.ConditionsT, offset, limit int) ([]*ms.Post, []*ms.PostFormated, error) {
|
|
|
|
posts, err := s.Ds.GetPosts(conditions, offset, limit)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
postFormated, err := s.Ds.MergePosts(posts)
|
|
|
|
return posts, postFormated, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *DaoServant) RelationTypFrom(me *ms.User, username string) (res *cs.VistUser, err error) {
|
|
|
|
res = &cs.VistUser{
|
|
|
|
RelTyp: cs.RelationSelf,
|
|
|
|
Username: username,
|
|
|
|
}
|
|
|
|
// visit by self
|
|
|
|
if me != nil && me.Username == username {
|
|
|
|
res.UserId = me.ID
|
|
|
|
return
|
|
|
|
}
|
|
|
|
he, xerr := s.Ds.GetUserByUsername(username)
|
|
|
|
if xerr != nil || (he.Model != nil && he.ID <= 0) {
|
|
|
|
return nil, errors.New("get user failed with username: " + username)
|
|
|
|
}
|
|
|
|
res.UserId = he.ID
|
|
|
|
// visit by guest
|
|
|
|
if me == nil {
|
|
|
|
res.RelTyp = cs.RelationGuest
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// visit by admin/friend/other
|
|
|
|
if me.IsAdmin {
|
|
|
|
res.RelTyp = cs.RelationAdmin
|
|
|
|
} else if s.Ds.IsFriend(me.ID, he.ID) {
|
|
|
|
res.RelTyp = cs.RelationFriend
|
|
|
|
} else {
|
|
|
|
res.RelTyp = cs.RelationGuest
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBindAnyFn() func(c *gin.Context, obj any) mir.Error {
|
|
|
|
if conf.UseSentryGin() {
|
|
|
|
return bindAnySentry
|
|
|
|
}
|
|
|
|
return bindAny
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBaseServant() *BaseServant {
|
|
|
|
return &BaseServant{
|
|
|
|
bindAny: NewBindAnyFn(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewDaoServant() *DaoServant {
|
|
|
|
return &DaoServant{
|
|
|
|
BaseServant: NewBaseServant(),
|
|
|
|
Redis: cache.NewRedisCache(),
|
|
|
|
Dsa: dao.WebDataServantA(),
|
|
|
|
Ds: dao.DataService(),
|
|
|
|
Ts: dao.TweetSearchService(),
|
|
|
|
}
|
|
|
|
}
|