teldrive/pkg/cron/cron.go
2024-08-31 15:50:17 +00:00

149 lines
4 KiB
Go

package cron
import (
"context"
"time"
"github.com/go-co-op/gocron"
"github.com/jackc/pgx/v5/pgtype"
"github.com/tgdrive/teldrive/internal/config"
"github.com/tgdrive/teldrive/internal/logging"
"github.com/tgdrive/teldrive/internal/tgc"
"github.com/tgdrive/teldrive/pkg/models"
"github.com/tgdrive/teldrive/pkg/schemas"
"go.uber.org/zap"
"gorm.io/datatypes"
"gorm.io/gorm"
)
type File struct {
ID string `json:"id"`
Parts []schemas.Part `json:"parts"`
}
type Result struct {
Files datatypes.JSONSlice[File]
Session string
UserId int64
ChannelId int64
}
type UploadResult struct {
Parts datatypes.JSONSlice[int]
Session string
UserId int64
ChannelId int64
}
type CronService struct {
db *gorm.DB
cnf *config.Config
logger *zap.SugaredLogger
}
func StartCronJobs(scheduler *gocron.Scheduler, db *gorm.DB, cnf *config.Config) {
if !cnf.CronJobs.Enable {
return
}
ctx := context.Background()
cron := CronService{db: db, cnf: cnf, logger: logging.DefaultLogger()}
scheduler.Every(cnf.CronJobs.CleanFilesInterval).Do(cron.CleanFiles, ctx)
scheduler.Every(cnf.CronJobs.FolderSizeInterval).Do(cron.UpdateFolderSize)
scheduler.Every(cnf.CronJobs.CleanUploadsInterval).Do(cron.CleanUploads, ctx)
scheduler.StartAsync()
}
func (c *CronService) CleanFiles(ctx context.Context) {
var results []Result
if err := c.db.Model(&models.File{}).
Select("JSONB_AGG(jsonb_build_object('id',files.id, 'parts',files.parts)) as files", "files.channel_id", "files.user_id", "s.session").
Joins("left join teldrive.users as u on u.user_id = files.user_id").
Joins("left join (select * from teldrive.sessions order by created_at desc limit 1) as s on u.user_id = s.user_id").
Where("type = ?", "file").
Where("status = ?", "pending_deletion").
Group("files.channel_id").Group("files.user_id").Group("s.session").
Scan(&results).Error; err != nil {
return
}
for _, row := range results {
if row.Session == "" {
break
}
ids := []int{}
fileIds := []string{}
for _, file := range row.Files {
fileIds = append(fileIds, file.ID)
for _, part := range file.Parts {
ids = append(ids, int(part.ID))
}
}
client, _ := tgc.AuthClient(ctx, &c.cnf.TG, row.Session)
err := tgc.DeleteMessages(ctx, client, row.ChannelId, ids)
if err != nil {
c.logger.Errorw("failed to delete messages", err)
return
}
items := pgtype.Array[string]{
Elements: fileIds,
Valid: true,
Dims: []pgtype.ArrayDimension{{Length: int32(len(fileIds)), LowerBound: 1}},
}
c.db.Where("id = any($1)", items).Delete(&models.File{})
c.logger.Infow("cleaned files", "user", row.UserId, "channel", row.ChannelId)
}
}
func (c *CronService) CleanUploads(ctx context.Context) {
var upResults []UploadResult
if err := c.db.Model(&models.Upload{}).
Select("JSONB_AGG(uploads.part_id) as parts", "uploads.channel_id", "uploads.user_id", "s.session").
Joins("left join teldrive.users as u on u.user_id = uploads.user_id").
Joins("left join (select * from teldrive.sessions order by created_at desc limit 1) as s on s.user_id = uploads.user_id").
Where("uploads.created_at < ?", time.Now().UTC().Add(-c.cnf.TG.Uploads.Retention)).
Group("uploads.channel_id").Group("uploads.user_id").Group("s.session").
Scan(&upResults).Error; err != nil {
return
}
for _, result := range upResults {
if result.Session != "" && len(result.Parts) > 0 {
client, _ := tgc.AuthClient(ctx, &c.cnf.TG, result.Session)
err := tgc.DeleteMessages(ctx, client, result.ChannelId, result.Parts)
if err != nil {
c.logger.Errorw("failed to delete messages", err)
return
}
}
items := pgtype.Array[int]{
Elements: result.Parts,
Valid: true,
Dims: []pgtype.ArrayDimension{{Length: int32(len(result.Parts)), LowerBound: 1}},
}
c.db.Where("part_id = any(?)", items).Where("channel_id = ?", result.ChannelId).
Where("user_id = ?", result.UserId).Delete(&models.Upload{}).Delete(&models.Upload{})
}
}
func (c *CronService) UpdateFolderSize() {
c.db.Exec("call teldrive.update_size();")
}