package repository import ( "context" "fmt" "log" "go.mongodb.org/mongo-driver/bson" "go.mongodb.org/mongo-driver/bson/primitive" "go.mongodb.org/mongo-driver/mongo" "go.mongodb.org/mongo-driver/mongo/options" "go.uber.org/zap" "penahub.gitlab.yandexcloud.net/pena-services/customer/internal/dto" "penahub.gitlab.yandexcloud.net/pena-services/customer/internal/errors" "penahub.gitlab.yandexcloud.net/pena-services/customer/internal/fields" "penahub.gitlab.yandexcloud.net/pena-services/customer/internal/models" mongoWrapper "penahub.gitlab.yandexcloud.net/pena-services/customer/pkg/mongo" ) type HistoryRepositoryDeps struct { Logger *zap.Logger MongoDB *mongo.Collection } type HistoryRepository struct { logger *zap.Logger mongoDB *mongo.Collection } func NewHistoryRepository(deps HistoryRepositoryDeps) *HistoryRepository { if deps.Logger == nil { log.Panicln("logger is nil on ") } if deps.MongoDB == nil { log.Panicln("mongodb is nil on ") } return &HistoryRepository{ logger: deps.Logger, mongoDB: deps.MongoDB, } } func (receiver *HistoryRepository) Insert(ctx context.Context, history *models.History) (*models.History, errors.Error) { result, err := receiver.mongoDB.InsertOne(ctx, history.Sanitize()) if err != nil { receiver.logger.Error("failed to insert history on of ", zap.Any("history", history), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to insert history on of : %w", err), errors.ErrInternalError, ) } insertedID := result.InsertedID.(primitive.ObjectID).Hex() history.ID = insertedID return history, nil } func (receiver *HistoryRepository) FindMany(ctx context.Context, dto *dto.GetHistories) ([]models.History, errors.Error) { findOptions := options.Find() findOptions.SetSkip((dto.Pagination.Page - 1) * dto.Pagination.Limit) findOptions.SetLimit(dto.Pagination.Limit) findOptions.SetSort(bson.D{{ Key: "createdAt", Value: -1, }}) histories, err := mongoWrapper.Find[models.History](ctx, &mongoWrapper.RequestSettings{ Driver: receiver.mongoDB, Options: findOptions, Filter: dto.BSON(), }) if err != nil { receiver.logger.Error("failed to find many histories on of ", zap.Int64("page", dto.Pagination.Page), zap.Int64("limit", dto.Pagination.Limit), zap.Int64("skip", (dto.Pagination.Page-1)*dto.Pagination.Limit), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to find many histories on of : %w", err), errors.ErrInternalError, ) } return histories, nil } func (receiver *HistoryRepository) CountAll(ctx context.Context, dto *dto.GetHistories) (int64, errors.Error) { count, err := receiver.mongoDB.CountDocuments(ctx, dto.BSON()) if err != nil { receiver.logger.Error("failed to count all documents on of ", zap.Error(err), ) return 0, errors.New( fmt.Errorf("failed to count all documents on of : %w", err), errors.ErrInternalError, ) } return count, nil } // TODO:tests // GetRecentTariffs method for processing a user request with data aggregation with a limit of 100 sorted in descending order. func (receiver *HistoryRepository) GetRecentTariffs(ctx context.Context, userID string) ([]models.TariffID, errors.Error) { matchStage := bson.D{ {Key: "$match", Value: bson.D{ {Key: fields.History.UserID, Value: userID}, {Key: fields.History.IsDeleted, Value: false}, {Key: fields.History.Type, Value: models.CustomerHistoryKeyPayCart}, }}, } sortStage := bson.D{ {Key: "$sort", Value: bson.D{ {Key: "createdAt", Value: -1}, }}, } limitStage := bson.D{ {Key: "$limit", Value: 100}, } cursor, err := receiver.mongoDB.Aggregate(ctx, mongo.Pipeline{matchStage, sortStage, limitStage}) if err != nil { receiver.logger.Error("failed to get recent tariffs on of ", zap.String("userId", userID), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to get recent tariffs on of : %w", err), errors.ErrInternalError, ) } var result []models.TariffID if err := cursor.All(ctx, &result); err != nil { receiver.logger.Error("failed to decode recent tariffs on of ", zap.String("userId", userID), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to decode recent tariffs on of : %w", err), errors.ErrInternalError, ) } return result, nil } // TODO:tests. func (receiver *HistoryRepository) GetHistoryById(ctx context.Context, historyID string) (*models.ReportHistory, errors.Error) { history := &models.ReportHistory{} err := receiver.mongoDB.FindOne(ctx, bson.M{"_id": historyID}).Decode(history) if err != nil { receiver.logger.Error( "failed to find by id in of ", zap.String("historyID", historyID), zap.Error(err), ) if err == mongo.ErrNoDocuments { return nil, errors.New( fmt.Errorf("history not found with ID: %s", historyID), errors.ErrNotFound, ) } return nil, errors.New( fmt.Errorf("failed to find by id: %w", err), errors.ErrInternalError, ) } return history, nil } // TODO:tests. func (receiver *HistoryRepository) GetDocNumber(ctx context.Context, userID string) (map[string]int, errors.Error) { findOptions := options.Find() findOptions.SetSort(bson.D{{Key: "createdAt", Value: 1}}) filter := bson.M{ fields.History.UserID: userID, } cursor, err := receiver.mongoDB.Find(ctx, filter, findOptions) if err != nil { receiver.logger.Error("failed to get DocNumber list on of ", zap.String("userId", userID), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to get DocNumber list on of : %w", err), errors.ErrInternalError, ) } defer func() { if err := cursor.Close(ctx); err != nil { receiver.logger.Error("failed to close cursor on of ", zap.String("userId", userID), zap.Error(err), ) } }() result := make(map[string]int) var count int for cursor.Next(ctx) { var history models.History if err := cursor.Decode(&history); err != nil { receiver.logger.Error("failed to decode history on of ", zap.String("userId", userID), zap.Error(err), ) return nil, errors.New( fmt.Errorf("failed to decode history on of : %w", err), errors.ErrInternalError, ) } result[history.ID] = count count++ } if err := cursor.Err(); err != nil { receiver.logger.Error("cursor error on of ", zap.String("userId", userID), zap.Error(err), ) return nil, errors.New( fmt.Errorf("cursor error on of : %w", err), errors.ErrInternalError, ) } return result, nil }