@@ -0,0 +1,49 @@
|
||||
package calories
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
"strings"
|
||||
)
|
||||
|
||||
var (
|
||||
productsMap = map[string]struct {
|
||||
caloriesIn100G int
|
||||
awgWeightG int
|
||||
}{
|
||||
"чай": {caloriesIn100G: 65, awgWeightG: 200},
|
||||
"яблоко": {caloriesIn100G: 52, awgWeightG: 242},
|
||||
"хлеб": {caloriesIn100G: 245, awgWeightG: 35},
|
||||
}
|
||||
)
|
||||
|
||||
func CalcCalories(text string) (int, error) {
|
||||
product, ok := productsMap[strings.ToLower(text)]
|
||||
if ok {
|
||||
return product.caloriesIn100G * product.awgWeightG / 100, nil
|
||||
}
|
||||
arr := strings.Split(text, " ")
|
||||
if len(arr) == 2 {
|
||||
product, ok := productsMap[strings.ToLower(arr[0])]
|
||||
if ok {
|
||||
count, err := strconv.Atoi(arr[1])
|
||||
if err != nil {
|
||||
return 0, nil
|
||||
}
|
||||
return product.caloriesIn100G * count / 100, nil
|
||||
}
|
||||
count1, err := strconv.Atoi(arr[0])
|
||||
if err != nil {
|
||||
return 0, nil
|
||||
}
|
||||
count2, err := strconv.Atoi(arr[1])
|
||||
if err != nil {
|
||||
return 0, nil
|
||||
}
|
||||
return count1 * count2 / 100, nil
|
||||
}
|
||||
count, err := strconv.Atoi(text)
|
||||
if err != nil {
|
||||
return 0, nil
|
||||
}
|
||||
return count, nil
|
||||
}
|
||||
@@ -0,0 +1,65 @@
|
||||
package calories
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestCalcCalories(t *testing.T) {
|
||||
t.Parallel()
|
||||
tests := []struct {
|
||||
name string
|
||||
text string
|
||||
want int
|
||||
wantErr bool
|
||||
}{
|
||||
{
|
||||
"empty text",
|
||||
"",
|
||||
0,
|
||||
false,
|
||||
},
|
||||
{
|
||||
"1 text",
|
||||
"1",
|
||||
1,
|
||||
false,
|
||||
},
|
||||
{
|
||||
"2 5 text",
|
||||
"20 50",
|
||||
10,
|
||||
false,
|
||||
},
|
||||
{
|
||||
"чай text",
|
||||
"чай",
|
||||
130,
|
||||
false,
|
||||
},
|
||||
{
|
||||
"чай 250 text",
|
||||
"чай 250",
|
||||
162,
|
||||
false,
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
tt := tt
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
t.Parallel()
|
||||
count, err := CalcCalories(tt.text)
|
||||
if (err != nil) != tt.wantErr {
|
||||
t.Errorf("error calc calories: %v", err)
|
||||
return
|
||||
}
|
||||
if count != tt.want {
|
||||
fmt.Println(count, tt.want)
|
||||
fmt.Println(count, tt.want)
|
||||
t.Errorf("error count: %v", err)
|
||||
return
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,13 @@
|
||||
package commands
|
||||
|
||||
type Command string
|
||||
|
||||
const (
|
||||
Start = Command("/start")
|
||||
Help = Command("/help")
|
||||
Ping = Command("/ping")
|
||||
Go = Command("/go")
|
||||
Stat = Command("/stat")
|
||||
Eat = Command("/eat")
|
||||
Pause = Command("/pause")
|
||||
)
|
||||
@@ -0,0 +1,19 @@
|
||||
package config
|
||||
|
||||
type Config struct {
|
||||
MongoURL string
|
||||
DBName string
|
||||
ChatsCollectionName string
|
||||
WorkoutsCollectionName string
|
||||
CaloriesCollectionName string
|
||||
}
|
||||
|
||||
func NewConfig(mongoURl string, dbName string) *Config {
|
||||
return &Config{
|
||||
MongoURL: mongoURl,
|
||||
DBName: dbName,
|
||||
ChatsCollectionName: "chats",
|
||||
WorkoutsCollectionName: "workouts",
|
||||
CaloriesCollectionName: "calories",
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,22 @@
|
||||
package db
|
||||
|
||||
import "time"
|
||||
|
||||
type Calories struct {
|
||||
ChatID int64 `bson:"chat_id"`
|
||||
Count int `bson:"count"`
|
||||
CreatedAt time.Time `bson:"created_at"`
|
||||
Username string `bson:"username"`
|
||||
}
|
||||
|
||||
func NewCalories(
|
||||
count int,
|
||||
username string,
|
||||
) *Calories {
|
||||
loc, _ := time.LoadLocation("Asia/Novosibirsk")
|
||||
return &Calories{
|
||||
Count: count,
|
||||
Username: username,
|
||||
CreatedAt: time.Now().In(loc),
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,23 @@
|
||||
package db
|
||||
|
||||
const (
|
||||
UserStateNone = UserState("")
|
||||
UserStateGo = UserState("Go")
|
||||
UserStateEat = UserState("Eat")
|
||||
UserStatePause = UserState("Pause")
|
||||
)
|
||||
|
||||
type UserState string
|
||||
|
||||
type Chat struct {
|
||||
ChatID int64 `bson:"chat_id"`
|
||||
}
|
||||
|
||||
type ChatInfo struct {
|
||||
ChatID int64 `bson:"chat_id"`
|
||||
Status string `bson:"status"`
|
||||
}
|
||||
|
||||
func (c *ChatInfo) GetStatus() UserState {
|
||||
return UserState(c.Status)
|
||||
}
|
||||
@@ -0,0 +1,225 @@
|
||||
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
|
||||
}
|
||||
@@ -0,0 +1,25 @@
|
||||
package db
|
||||
|
||||
import "time"
|
||||
|
||||
type Workout struct {
|
||||
ChatID int64 `bson:"chat_id"`
|
||||
Name string `bson:"name"`
|
||||
Count int `bson:"count"`
|
||||
CreatedAt time.Time `bson:"created_at"`
|
||||
Username string `bson:"username"`
|
||||
}
|
||||
|
||||
func NewWorkout(
|
||||
name string,
|
||||
count int,
|
||||
username string,
|
||||
) *Workout {
|
||||
loc, _ := time.LoadLocation("Asia/Novosibirsk")
|
||||
return &Workout{
|
||||
Name: name,
|
||||
Count: count,
|
||||
Username: username,
|
||||
CreatedAt: time.Now().In(loc),
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,21 @@
|
||||
package pause
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
func GetDuration(text string) (time.Duration, error) {
|
||||
d := strings.TrimSuffix(text, "ч")
|
||||
d = strings.TrimSuffix(d, "д")
|
||||
count, err := strconv.Atoi(d)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
res := time.Duration(count) * time.Hour
|
||||
if strings.HasSuffix(text, "д") {
|
||||
res *= 24
|
||||
}
|
||||
return res, nil
|
||||
}
|
||||
Reference in New Issue
Block a user