261 lines
7.9 KiB
Go
261 lines
7.9 KiB
Go
|
|
package repository
|
||
|
|
|
||
|
|
import (
|
||
|
|
"context"
|
||
|
|
"strings"
|
||
|
|
"time"
|
||
|
|
|
||
|
|
app "haixun-backend/internal/library/errors"
|
||
|
|
"haixun-backend/internal/library/errors/code"
|
||
|
|
libmongo "haixun-backend/internal/library/mongo"
|
||
|
|
"haixun-backend/internal/model/scan_post/domain/entity"
|
||
|
|
domrepo "haixun-backend/internal/model/scan_post/domain/repository"
|
||
|
|
|
||
|
|
"go.mongodb.org/mongo-driver/bson"
|
||
|
|
"go.mongodb.org/mongo-driver/mongo"
|
||
|
|
"go.mongodb.org/mongo-driver/mongo/options"
|
||
|
|
)
|
||
|
|
|
||
|
|
type mongoRepository struct {
|
||
|
|
collection *mongo.Collection
|
||
|
|
}
|
||
|
|
|
||
|
|
func NewMongoRepository(db *mongo.Database) domrepo.Repository {
|
||
|
|
if db == nil {
|
||
|
|
return &mongoRepository{}
|
||
|
|
}
|
||
|
|
return &mongoRepository{collection: db.Collection(entity.CollectionName)}
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) EnsureIndexes(ctx context.Context) error {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
return libmongo.EnsureIndexes(ctx, r.collection, []mongo.IndexModel{
|
||
|
|
{
|
||
|
|
Keys: bson.D{{Key: "tenant_id", Value: 1}, {Key: "owner_uid", Value: 1}, {Key: "brand_id", Value: 1}, {Key: "permalink", Value: 1}},
|
||
|
|
Options: options.Index().SetUnique(true).SetPartialFilterExpression(bson.M{"brand_id": bson.M{"$gt": ""}})},
|
||
|
|
{
|
||
|
|
Keys: bson.D{{Key: "tenant_id", Value: 1}, {Key: "owner_uid", Value: 1}, {Key: "persona_id", Value: 1}, {Key: "permalink", Value: 1}},
|
||
|
|
Options: options.Index().SetUnique(true).SetPartialFilterExpression(bson.M{"persona_id": bson.M{"$gt": ""}})},
|
||
|
|
{
|
||
|
|
Keys: bson.D{{Key: "tenant_id", Value: 1}, {Key: "owner_uid", Value: 1}, {Key: "brand_id", Value: 1}, {Key: "priority", Value: 1}},
|
||
|
|
},
|
||
|
|
{
|
||
|
|
Keys: bson.D{{Key: "tenant_id", Value: 1}, {Key: "owner_uid", Value: 1}, {Key: "persona_id", Value: 1}, {Key: "priority", Value: 1}},
|
||
|
|
},
|
||
|
|
{
|
||
|
|
Keys: bson.D{{Key: "scan_job_id", Value: 1}},
|
||
|
|
},
|
||
|
|
})
|
||
|
|
}
|
||
|
|
|
||
|
|
func brandOwnerFilter(tenantID, ownerUID, brandID string) bson.M {
|
||
|
|
filter := bson.M{
|
||
|
|
"tenant_id": tenantID,
|
||
|
|
"owner_uid": ownerUID,
|
||
|
|
}
|
||
|
|
for k, v := range libmongo.BrandScopeFilter(brandID) {
|
||
|
|
filter[k] = v
|
||
|
|
}
|
||
|
|
return filter
|
||
|
|
}
|
||
|
|
|
||
|
|
func personaViralFilter(tenantID, ownerUID, personaID string) bson.M {
|
||
|
|
return bson.M{
|
||
|
|
"tenant_id": tenantID,
|
||
|
|
"owner_uid": ownerUID,
|
||
|
|
"persona_id": strings.TrimSpace(personaID),
|
||
|
|
"flow": entity.FlowViral,
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) ReplaceForViralScan(ctx context.Context, tenantID, ownerUID, personaID, scanJobID string, posts []entity.ScanPost) error {
|
||
|
|
if r.collection == nil {
|
||
|
|
return app.For(code.Persona).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
_, err := r.collection.DeleteMany(ctx, personaViralFilter(tenantID, ownerUID, personaID))
|
||
|
|
if err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
if len(posts) == 0 {
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
docs := make([]any, 0, len(posts))
|
||
|
|
for _, post := range posts {
|
||
|
|
docs = append(docs, post)
|
||
|
|
}
|
||
|
|
_, err = r.collection.InsertMany(ctx, docs)
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) ReplaceForScan(ctx context.Context, tenantID, ownerUID, brandID, scanJobID string, posts []entity.ScanPost) error {
|
||
|
|
if r.collection == nil {
|
||
|
|
return app.For(code.Brand).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
_, err := r.collection.DeleteMany(ctx, brandOwnerFilter(tenantID, ownerUID, brandID))
|
||
|
|
if err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
if len(posts) == 0 {
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
docs := make([]any, 0, len(posts))
|
||
|
|
for _, post := range posts {
|
||
|
|
docs = append(docs, post)
|
||
|
|
}
|
||
|
|
_, err = r.collection.InsertMany(ctx, docs)
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) Get(ctx context.Context, tenantID, ownerUID, brandID, postID string) (*entity.ScanPost, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil, app.For(code.Brand).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
filter := brandOwnerFilter(tenantID, ownerUID, brandID)
|
||
|
|
filter["_id"] = strings.TrimSpace(postID)
|
||
|
|
var out entity.ScanPost
|
||
|
|
err := r.collection.FindOne(ctx, filter).Decode(&out)
|
||
|
|
if err == mongo.ErrNoDocuments {
|
||
|
|
return nil, app.For(code.Brand).ResNotFound("scan post not found")
|
||
|
|
}
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return &out, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) UpdateOutreach(
|
||
|
|
ctx context.Context,
|
||
|
|
tenantID, ownerUID, brandID, postID string,
|
||
|
|
patch entity.OutreachPatch,
|
||
|
|
) (*entity.ScanPost, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil, app.For(code.Brand).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
set := bson.M{}
|
||
|
|
if strings.TrimSpace(patch.Status) != "" {
|
||
|
|
set["outreach_status"] = strings.TrimSpace(patch.Status)
|
||
|
|
}
|
||
|
|
if strings.TrimSpace(patch.PublishedReplyID) != "" {
|
||
|
|
set["published_reply_id"] = strings.TrimSpace(patch.PublishedReplyID)
|
||
|
|
}
|
||
|
|
if strings.TrimSpace(patch.PublishedPermalink) != "" {
|
||
|
|
set["published_permalink"] = strings.TrimSpace(patch.PublishedPermalink)
|
||
|
|
}
|
||
|
|
if len(set) == 0 {
|
||
|
|
return r.Get(ctx, tenantID, ownerUID, brandID, postID)
|
||
|
|
}
|
||
|
|
set["outreach_update_at"] = time.Now().UnixNano()
|
||
|
|
|
||
|
|
filter := brandOwnerFilter(tenantID, ownerUID, brandID)
|
||
|
|
filter["_id"] = strings.TrimSpace(postID)
|
||
|
|
opts := options.FindOneAndUpdate().SetReturnDocument(options.After)
|
||
|
|
var out entity.ScanPost
|
||
|
|
err := r.collection.FindOneAndUpdate(ctx, filter, bson.M{"$set": set}, opts).Decode(&out)
|
||
|
|
if err == mongo.ErrNoDocuments {
|
||
|
|
return nil, app.For(code.Brand).ResNotFound("scan post not found")
|
||
|
|
}
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return &out, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) GetForPersona(ctx context.Context, tenantID, ownerUID, personaID, postID string) (*entity.ScanPost, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil, app.For(code.Persona).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
filter := personaViralFilter(tenantID, ownerUID, personaID)
|
||
|
|
filter["_id"] = strings.TrimSpace(postID)
|
||
|
|
var out entity.ScanPost
|
||
|
|
err := r.collection.FindOne(ctx, filter).Decode(&out)
|
||
|
|
if err == mongo.ErrNoDocuments {
|
||
|
|
return nil, app.For(code.Persona).ResNotFound("viral scan post not found")
|
||
|
|
}
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return &out, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) ListForPersona(ctx context.Context, tenantID, ownerUID string, filter domrepo.PersonaListFilter) ([]entity.ScanPost, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil, app.For(code.Persona).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
query := personaViralFilter(tenantID, ownerUID, filter.PersonaID)
|
||
|
|
if flow := strings.TrimSpace(filter.Flow); flow != "" {
|
||
|
|
query["flow"] = flow
|
||
|
|
}
|
||
|
|
limit := filter.Limit
|
||
|
|
if limit <= 0 {
|
||
|
|
limit = 100
|
||
|
|
}
|
||
|
|
if limit > 500 {
|
||
|
|
limit = 500
|
||
|
|
}
|
||
|
|
opts := options.Find().
|
||
|
|
SetSort(bson.D{{Key: "engagement_score", Value: -1}, {Key: "create_at", Value: -1}}).
|
||
|
|
SetLimit(int64(limit))
|
||
|
|
cur, err := r.collection.Find(ctx, query, opts)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
defer cur.Close(ctx)
|
||
|
|
var out []entity.ScanPost
|
||
|
|
if err := cur.All(ctx, &out); err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return out, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) List(ctx context.Context, tenantID, ownerUID string, filter domrepo.ListFilter) ([]entity.ScanPost, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return nil, app.For(code.Brand).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
query := brandOwnerFilter(tenantID, ownerUID, filter.BrandID)
|
||
|
|
if strings.TrimSpace(filter.Priority) != "" {
|
||
|
|
query["priority"] = strings.TrimSpace(filter.Priority)
|
||
|
|
}
|
||
|
|
if filter.ProductFitMin > 0 {
|
||
|
|
query["product_fit_score"] = bson.M{"$gte": filter.ProductFitMin}
|
||
|
|
}
|
||
|
|
limit := filter.Limit
|
||
|
|
if limit <= 0 {
|
||
|
|
limit = 100
|
||
|
|
}
|
||
|
|
if limit > 500 {
|
||
|
|
limit = 500
|
||
|
|
}
|
||
|
|
opts := options.Find().
|
||
|
|
SetSort(bson.D{{Key: "placement_score", Value: -1}, {Key: "create_at", Value: -1}}).
|
||
|
|
SetLimit(int64(limit))
|
||
|
|
cur, err := r.collection.Find(ctx, query, opts)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
defer cur.Close(ctx)
|
||
|
|
var out []entity.ScanPost
|
||
|
|
if err := cur.All(ctx, &out); err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
if filter.Recent7dOnly {
|
||
|
|
filtered := make([]entity.ScanPost, 0, len(out))
|
||
|
|
for _, item := range out {
|
||
|
|
if item.Priority == "gold" || item.Priority == "recent" {
|
||
|
|
filtered = append(filtered, item)
|
||
|
|
}
|
||
|
|
}
|
||
|
|
return filtered, nil
|
||
|
|
}
|
||
|
|
return out, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *mongoRepository) CountByBrand(ctx context.Context, tenantID, ownerUID, brandID string) (int, error) {
|
||
|
|
if r.collection == nil {
|
||
|
|
return 0, app.For(code.Brand).DBUnavailable("Mongo is not configured")
|
||
|
|
}
|
||
|
|
count, err := r.collection.CountDocuments(ctx, brandOwnerFilter(tenantID, ownerUID, brandID))
|
||
|
|
return int(count), err
|
||
|
|
}
|