226 lines
4.5 KiB
Go
226 lines
4.5 KiB
Go
package db
|
|
|
|
import (
|
|
"context"
|
|
"log"
|
|
"time"
|
|
"valera/internal/config"
|
|
|
|
"go.mongodb.org/mongo-driver/bson"
|
|
"go.mongodb.org/mongo-driver/mongo"
|
|
"go.mongodb.org/mongo-driver/mongo/options"
|
|
)
|
|
|
|
type DB struct {
|
|
chatsColection *mongo.Collection
|
|
workoutsColection *mongo.Collection
|
|
caloriesColection *mongo.Collection
|
|
}
|
|
|
|
func NewDB(
|
|
mongoURL string,
|
|
dbName string,
|
|
) (*DB, error) {
|
|
ctx := context.Background()
|
|
opt := options.Client().
|
|
ApplyURI(mongoURL).
|
|
SetMinPoolSize(10)
|
|
cfg := config.NewConfig(mongoURL, dbName)
|
|
|
|
client, err := mongo.Connect(ctx, opt)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &DB{
|
|
chatsColection: client.Database(dbName).Collection(cfg.ChatsCollectionName),
|
|
workoutsColection: client.Database(dbName).Collection(cfg.WorkoutsCollectionName),
|
|
caloriesColection: client.Database(dbName).Collection(cfg.CaloriesCollectionName),
|
|
}, nil
|
|
}
|
|
|
|
func (db *DB) AddChat(chatID int64) error {
|
|
ctx := context.Background()
|
|
|
|
result := db.chatsColection.FindOne(ctx, bson.M{"chat_id": chatID})
|
|
if result.Err() == mongo.ErrNoDocuments {
|
|
if _, err := db.chatsColection.InsertOne(ctx, &Chat{ChatID: chatID}); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
return result.Err()
|
|
}
|
|
|
|
func (db *DB) AddWorkout(chatID int64, workout *Workout) error {
|
|
ctx := context.Background()
|
|
|
|
if workout.Count <= 0 {
|
|
return nil
|
|
}
|
|
if err := db.AddChat(chatID); err != nil {
|
|
return err
|
|
}
|
|
workout.ChatID = chatID
|
|
|
|
_, err := db.workoutsColection.InsertOne(
|
|
ctx,
|
|
workout,
|
|
)
|
|
return err
|
|
}
|
|
|
|
func (db *DB) AddCalories(chatID int64, calories *Calories) error {
|
|
ctx := context.Background()
|
|
|
|
if calories.Count <= 0 {
|
|
return nil
|
|
}
|
|
if err := db.AddChat(chatID); err != nil {
|
|
return err
|
|
}
|
|
calories.ChatID = chatID
|
|
|
|
_, err := db.caloriesColection.InsertOne(
|
|
ctx,
|
|
calories,
|
|
)
|
|
return err
|
|
}
|
|
|
|
func (db *DB) GetChatInfo(chatID int64) (*ChatInfo, error) {
|
|
ctx := context.Background()
|
|
|
|
if err := db.AddChat(chatID); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
chatInfoDTO := &ChatInfo{}
|
|
if err := db.chatsColection.FindOne(ctx, bson.M{"chat_id": chatID}).Decode(chatInfoDTO); err != nil {
|
|
return nil, err
|
|
}
|
|
return chatInfoDTO, nil
|
|
}
|
|
|
|
func (db *DB) SetStatusToChat(chatID int64, status UserState) error {
|
|
ctx := context.Background()
|
|
|
|
_, err := db.chatsColection.UpdateOne(
|
|
ctx,
|
|
bson.M{"chat_id": chatID},
|
|
bson.M{"$set": bson.M{"status": status}},
|
|
)
|
|
return err
|
|
}
|
|
|
|
func (db *DB) SetPause(chatID int64, duration time.Duration) error {
|
|
ctx := context.Background()
|
|
|
|
loc, _ := time.LoadLocation("Asia/Novosibirsk")
|
|
t := time.Now().In(loc).Add(+duration)
|
|
_, err := db.chatsColection.UpdateOne(
|
|
ctx,
|
|
bson.M{"chat_id": chatID},
|
|
bson.M{"$set": bson.M{"pause_until": t}},
|
|
)
|
|
return err
|
|
}
|
|
|
|
func (db *DB) GetStat(chatID int64, cron bool) (map[string]int, error) {
|
|
ctx := context.Background()
|
|
|
|
if err := db.AddChat(chatID); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
loc, _ := time.LoadLocation("Asia/Novosibirsk")
|
|
t := time.Now().In(loc)
|
|
if cron {
|
|
t = t.Add(-24 * time.Hour)
|
|
}
|
|
if !cron {
|
|
t = time.Date(t.Year(), t.Month(), t.Day(), 0, 0, 0, 0, loc)
|
|
}
|
|
|
|
res := map[string]int{}
|
|
|
|
cursor, err := db.workoutsColection.Find(
|
|
ctx,
|
|
bson.M{
|
|
"chat_id": chatID,
|
|
"created_at": bson.M{"$gt": t},
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for cursor.Next(context.Background()) {
|
|
var result Workout
|
|
if err := cursor.Decode(&result); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
res[result.Name] += result.Count
|
|
}
|
|
if err := cursor.Err(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
caloriesCursor, err := db.caloriesColection.Find(
|
|
ctx,
|
|
bson.M{
|
|
"chat_id": chatID,
|
|
"created_at": bson.M{"$gt": t},
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for caloriesCursor.Next(context.Background()) {
|
|
var result Calories
|
|
if err := caloriesCursor.Decode(&result); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
res["Калории"] += result.Count
|
|
}
|
|
if err := caloriesCursor.Err(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
return res, err
|
|
}
|
|
|
|
func (db *DB) GetAllChats() ([]int64, error) {
|
|
ctx := context.Background()
|
|
|
|
loc, _ := time.LoadLocation("Asia/Novosibirsk")
|
|
t := time.Now().In(loc)
|
|
cursor, err := db.chatsColection.Find(
|
|
ctx,
|
|
bson.M{
|
|
"$or": bson.A{
|
|
bson.M{
|
|
"pause_until": bson.M{"$lt": t},
|
|
},
|
|
bson.M{
|
|
"pause_until": bson.M{"$exists": false},
|
|
},
|
|
},
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var res []int64
|
|
for cursor.Next(context.Background()) {
|
|
var result ChatInfo
|
|
if err := cursor.Decode(&result); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
res = append(res, result.ChatID)
|
|
}
|
|
if err := cursor.Err(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
return res, err
|
|
}
|