mirror of https://gitee.com/answerdev/answer.git
159 lines
4.0 KiB
Go
159 lines
4.0 KiB
Go
package activity
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/answerdev/answer/internal/service/activity_common"
|
|
"github.com/answerdev/answer/internal/service/follow"
|
|
"github.com/answerdev/answer/pkg/obj"
|
|
"github.com/segmentfault/pacman/log"
|
|
"xorm.io/builder"
|
|
|
|
"github.com/answerdev/answer/internal/base/data"
|
|
"github.com/answerdev/answer/internal/base/reason"
|
|
"github.com/answerdev/answer/internal/entity"
|
|
"github.com/answerdev/answer/internal/service/unique"
|
|
"github.com/segmentfault/pacman/errors"
|
|
"xorm.io/xorm"
|
|
)
|
|
|
|
// FollowRepo activity repository
|
|
type FollowRepo struct {
|
|
data *data.Data
|
|
uniqueIDRepo unique.UniqueIDRepo
|
|
activityRepo activity_common.ActivityRepo
|
|
}
|
|
|
|
// NewFollowRepo new repository
|
|
func NewFollowRepo(
|
|
data *data.Data,
|
|
uniqueIDRepo unique.UniqueIDRepo,
|
|
activityRepo activity_common.ActivityRepo,
|
|
) follow.FollowRepo {
|
|
return &FollowRepo{
|
|
data: data,
|
|
uniqueIDRepo: uniqueIDRepo,
|
|
activityRepo: activityRepo,
|
|
}
|
|
}
|
|
|
|
func (ar *FollowRepo) Follow(ctx context.Context, objectID, userID string) error {
|
|
activityType, _, _, err := ar.activityRepo.GetActivityTypeByObjID(ctx, objectID, "follow")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = ar.data.DB.Transaction(func(session *xorm.Session) (result any, err error) {
|
|
var (
|
|
existsActivity entity.Activity
|
|
has bool
|
|
)
|
|
result = nil
|
|
|
|
has, err = session.Where(builder.Eq{"activity_type": activityType}).
|
|
And(builder.Eq{"user_id": userID}).
|
|
And(builder.Eq{"object_id": objectID}).
|
|
Get(&existsActivity)
|
|
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
if has && existsActivity.Cancelled == entity.ActivityAvailable {
|
|
return
|
|
}
|
|
|
|
if has {
|
|
_, err = session.Where(builder.Eq{"id": existsActivity.ID}).
|
|
Cols(`cancelled`).
|
|
Update(&entity.Activity{
|
|
Cancelled: entity.ActivityAvailable,
|
|
})
|
|
} else {
|
|
// update existing activity with new user id and u object id
|
|
_, err = session.Insert(&entity.Activity{
|
|
UserID: userID,
|
|
ObjectID: objectID,
|
|
OriginalObjectID: objectID,
|
|
ActivityType: activityType,
|
|
Cancelled: entity.ActivityAvailable,
|
|
Rank: 0,
|
|
HasRank: 0,
|
|
})
|
|
}
|
|
|
|
if err != nil {
|
|
log.Error(err)
|
|
return
|
|
}
|
|
|
|
// start update followers when everything is fine
|
|
err = ar.updateFollows(ctx, session, objectID, 1)
|
|
if err != nil {
|
|
log.Error(err)
|
|
}
|
|
|
|
return
|
|
})
|
|
|
|
return err
|
|
}
|
|
|
|
func (ar *FollowRepo) FollowCancel(ctx context.Context, objectID, userID string) error {
|
|
activityType, _, _, err := ar.activityRepo.GetActivityTypeByObjID(ctx, objectID, "follow")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = ar.data.DB.Transaction(func(session *xorm.Session) (result any, err error) {
|
|
var (
|
|
existsActivity entity.Activity
|
|
has bool
|
|
)
|
|
result = nil
|
|
|
|
has, err = session.Where(builder.Eq{"activity_type": activityType}).
|
|
And(builder.Eq{"user_id": userID}).
|
|
And(builder.Eq{"object_id": objectID}).
|
|
Get(&existsActivity)
|
|
|
|
if err != nil || !has {
|
|
return
|
|
}
|
|
|
|
if has && existsActivity.Cancelled == entity.ActivityCancelled {
|
|
return
|
|
}
|
|
if _, err = session.Where("id = ?", existsActivity.ID).
|
|
Cols("cancelled").
|
|
Update(&entity.Activity{
|
|
Cancelled: entity.ActivityCancelled,
|
|
CancelledAt: time.Now(),
|
|
}); err != nil {
|
|
return
|
|
}
|
|
err = ar.updateFollows(ctx, session, objectID, -1)
|
|
return
|
|
})
|
|
return err
|
|
}
|
|
|
|
func (ar *FollowRepo) updateFollows(ctx context.Context, session *xorm.Session, objectID string, follows int) error {
|
|
objectType, err := obj.GetObjectTypeStrByObjectID(objectID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
switch objectType {
|
|
case "question":
|
|
_, err = session.Where("id = ?", objectID).Incr("follow_count", follows).Update(&entity.Question{})
|
|
case "user":
|
|
_, err = session.Where("id = ?", objectID).Incr("follow_count", follows).Update(&entity.User{})
|
|
case "tag":
|
|
_, err = session.Where("id = ?", objectID).Incr("follow_count", follows).Update(&entity.Tag{})
|
|
default:
|
|
err = errors.InternalServer(reason.DisallowFollow).WithMsg("this object can't be followed")
|
|
}
|
|
return err
|
|
}
|