// Copyright © 2023 OpenIM open source community. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package chat import ( "context" "time" "github.com/openimsdk/tools/db/mongoutil" "github.com/openimsdk/tools/db/pagination" "github.com/openimsdk/tools/errs" "go.mongodb.org/mongo-driver/bson" "go.mongodb.org/mongo-driver/mongo" "go.mongodb.org/mongo-driver/mongo/options" "git.imall.cloud/openim/chat/pkg/common/db/table/chat" ) func NewAttribute(db *mongo.Database) (chat.AttributeInterface, error) { coll := db.Collection("attribute") _, err := coll.Indexes().CreateMany(context.Background(), []mongo.IndexModel{ { Keys: bson.D{ {Key: "user_id", Value: 1}, }, Options: options.Index().SetUnique(true), }, { Keys: bson.D{ {Key: "account", Value: 1}, }, }, { Keys: bson.D{ {Key: "email", Value: 1}, }, }, { Keys: bson.D{ {Key: "area_code", Value: 1}, {Key: "phone_number", Value: 1}, }, }, }) if err != nil { return nil, errs.Wrap(err) } return &Attribute{coll: coll}, nil } type Attribute struct { coll *mongo.Collection } func (o *Attribute) Create(ctx context.Context, attribute ...*chat.Attribute) error { return mongoutil.InsertMany(ctx, o.coll, attribute) } func (o *Attribute) Update(ctx context.Context, userID string, data map[string]any) error { if len(data) == 0 { return nil } return mongoutil.UpdateOne(ctx, o.coll, bson.M{"user_id": userID}, bson.M{"$set": data}, false) } func (o *Attribute) Find(ctx context.Context, userIds []string) ([]*chat.Attribute, error) { return mongoutil.Find[*chat.Attribute](ctx, o.coll, bson.M{"user_id": bson.M{"$in": userIds}}) } func (o *Attribute) FindAccount(ctx context.Context, accounts []string) ([]*chat.Attribute, error) { return mongoutil.Find[*chat.Attribute](ctx, o.coll, bson.M{"account": bson.M{"$in": accounts}}) } func (o *Attribute) FindPhone(ctx context.Context, phoneNumbers []string) ([]*chat.Attribute, error) { return mongoutil.Find[*chat.Attribute](ctx, o.coll, bson.M{"phone_number": bson.M{"$in": phoneNumbers}}) } func (o *Attribute) Search(ctx context.Context, keyword string, genders []int32, pagination pagination.Pagination) (int64, []*chat.Attribute, error) { filter := bson.M{} if len(genders) > 0 { filter["gender"] = bson.M{ "$in": genders, } } if keyword != "" { filter["$or"] = []bson.M{ {"user_id": bson.M{"$regex": keyword, "$options": "i"}}, {"account": bson.M{"$regex": keyword, "$options": "i"}}, {"nickname": bson.M{"$regex": keyword, "$options": "i"}}, {"phone_number": bson.M{"$regex": keyword, "$options": "i"}}, } } return mongoutil.FindPage[*chat.Attribute](ctx, o.coll, filter, pagination) } func (o *Attribute) TakePhone(ctx context.Context, areaCode string, phoneNumber string) (*chat.Attribute, error) { return mongoutil.FindOne[*chat.Attribute](ctx, o.coll, bson.M{"area_code": areaCode, "phone_number": phoneNumber}) } func (o *Attribute) TakeEmail(ctx context.Context, email string) (*chat.Attribute, error) { return mongoutil.FindOne[*chat.Attribute](ctx, o.coll, bson.M{"email": email}) } func (o *Attribute) TakeAccount(ctx context.Context, account string) (*chat.Attribute, error) { return mongoutil.FindOne[*chat.Attribute](ctx, o.coll, bson.M{"account": account}) } func (o *Attribute) Take(ctx context.Context, userID string) (*chat.Attribute, error) { return mongoutil.FindOne[*chat.Attribute](ctx, o.coll, bson.M{"user_id": userID}) } func (o *Attribute) SearchNormalUser(ctx context.Context, keyword string, forbiddenIDs []string, gender int32, startTime, endTime *time.Time, pagination pagination.Pagination) (int64, []*chat.Attribute, error) { filter := bson.M{} if gender == 0 { filter["gender"] = bson.M{ "$in": []int32{0, 1, 2}, } } else { filter["gender"] = gender } if len(forbiddenIDs) > 0 { filter["user_id"] = bson.M{ "$nin": forbiddenIDs, } } if keyword != "" { filter["$or"] = []bson.M{ {"user_id": bson.M{"$regex": keyword, "$options": "i"}}, {"account": bson.M{"$regex": keyword, "$options": "i"}}, {"nickname": bson.M{"$regex": keyword, "$options": "i"}}, {"phone_number": bson.M{"$regex": keyword, "$options": "i"}}, {"email": bson.M{"$regex": keyword, "$options": "i"}}, } } // 注册时间范围查询 if startTime != nil || endTime != nil { timeFilter := bson.M{} if startTime != nil { timeFilter["$gte"] = *startTime } if endTime != nil { // 使用 $lt(小于)而不是 $lte,确保不包含结束时间当天的数据 // 例如:endTime="2025-11-02 00:00:00" 应该查询到 2025-11-01 23:59:59,不包含 11月2日的数据 timeFilter["$lt"] = *endTime } if len(timeFilter) > 0 { filter["create_time"] = timeFilter } } return mongoutil.FindPage[*chat.Attribute](ctx, o.coll, filter, pagination, options.Find().SetSort(bson.D{{Key: "create_time", Value: -1}})) } // SearchNormalUserWithUserIDs 按条件搜索用户(支持额外的userIDs过滤) func (o *Attribute) SearchNormalUserWithUserIDs(ctx context.Context, keyword string, forbiddenIDs []string, gender int32, startTime, endTime *time.Time, userIDs []string, pagination pagination.Pagination) (int64, []*chat.Attribute, error) { filter := bson.M{} if gender == 0 { filter["gender"] = bson.M{ "$in": []int32{0, 1, 2}, } } else { filter["gender"] = gender } // 构建user_id过滤条件:需要同时满足在userIDs中,且不在forbiddenIDs中 userIDConditions := []bson.M{} if len(userIDs) > 0 { userIDConditions = append(userIDConditions, bson.M{"user_id": bson.M{"$in": userIDs}}) } if len(forbiddenIDs) > 0 { userIDConditions = append(userIDConditions, bson.M{"user_id": bson.M{"$nin": forbiddenIDs}}) } if len(userIDConditions) > 0 { if len(userIDConditions) == 1 { filter["user_id"] = userIDConditions[0]["user_id"] } else { // 需要同时满足多个条件,使用 $and filter["$and"] = userIDConditions } } if keyword != "" { filter["$or"] = []bson.M{ {"user_id": bson.M{"$regex": keyword, "$options": "i"}}, {"account": bson.M{"$regex": keyword, "$options": "i"}}, {"nickname": bson.M{"$regex": keyword, "$options": "i"}}, {"phone_number": bson.M{"$regex": keyword, "$options": "i"}}, {"email": bson.M{"$regex": keyword, "$options": "i"}}, } } // 注册时间范围查询 if startTime != nil || endTime != nil { timeFilter := bson.M{} if startTime != nil { timeFilter["$gte"] = *startTime } if endTime != nil { timeFilter["$lt"] = *endTime } if len(timeFilter) > 0 { filter["create_time"] = timeFilter } } return mongoutil.FindPage[*chat.Attribute](ctx, o.coll, filter, pagination, options.Find().SetSort(bson.D{{Key: "create_time", Value: -1}})) } func (o *Attribute) SearchUser(ctx context.Context, keyword string, userIDs []string, genders []int32, pagination pagination.Pagination) (int64, []*chat.Attribute, error) { filter := bson.M{} if len(genders) > 0 { filter["gender"] = bson.M{ "$in": genders, } } if len(userIDs) > 0 { filter["user_id"] = bson.M{ "$in": userIDs, } } if keyword != "" { filter["$or"] = []bson.M{ {"user_id": bson.M{"$regex": keyword, "$options": "i"}}, {"account": bson.M{"$regex": keyword, "$options": "i"}}, {"nickname": bson.M{"$regex": keyword, "$options": "i"}}, {"phone_number": bson.M{"$regex": keyword, "$options": "i"}}, {"email": bson.M{"$regex": keyword, "$options": "i"}}, } } return mongoutil.FindPage[*chat.Attribute](ctx, o.coll, filter, pagination) } func (o *Attribute) Delete(ctx context.Context, userIDs []string) error { if len(userIDs) == 0 { return nil } return mongoutil.DeleteMany(ctx, o.coll, bson.M{"user_id": bson.M{"$in": userIDs}}) }