teldrive/internal/tgc/helpers.go
2024-06-23 15:08:50 +05:30

241 lines
5.7 KiB
Go

package tgc
import (
"bytes"
"context"
"errors"
"fmt"
"math"
"runtime"
"sync"
"github.com/divyam234/teldrive/internal/cache"
"github.com/divyam234/teldrive/internal/config"
"github.com/divyam234/teldrive/internal/kv"
"github.com/divyam234/teldrive/pkg/types"
"github.com/gotd/td/telegram"
"github.com/gotd/td/tg"
"golang.org/x/sync/errgroup"
)
var (
ErrInValidChannelID = errors.New("invalid channel id")
ErrInvalidChannelMessages = errors.New("invalid channel messages")
)
func GetChannelById(ctx context.Context, client *tg.Client, channelId int64) (*tg.InputChannel, error) {
inputChannel := &tg.InputChannel{
ChannelID: channelId,
}
channels, err := client.ChannelsGetChannels(ctx, []tg.InputChannelClass{inputChannel})
if err != nil {
return nil, err
}
if len(channels.GetChats()) == 0 {
return nil, ErrInValidChannelID
}
return channels.GetChats()[0].(*tg.Channel).AsInput(), nil
}
func DeleteMessages(ctx context.Context, client *telegram.Client, channelId int64, ids []int) error {
return RunWithAuth(ctx, client, "", func(ctx context.Context) error {
channel, err := GetChannelById(ctx, client.API(), channelId)
if err != nil {
return err
}
batchSize := 100
batchCount := int(math.Ceil(float64(len(ids)) / float64(batchSize)))
g, _ := errgroup.WithContext(ctx)
g.SetLimit(runtime.NumCPU())
for i := 0; i < batchCount; i++ {
start := i * batchSize
end := min((i+1)*batchSize, len(ids))
batchIds := ids[start:end]
g.Go(func() error {
messageDeleteRequest := tg.ChannelsDeleteMessagesRequest{Channel: channel, ID: batchIds}
_, err = client.API().ChannelsDeleteMessages(ctx, &messageDeleteRequest)
return err
})
}
return g.Wait()
})
}
func getTGMessagesBatch(ctx context.Context, client *tg.Client, channel *tg.InputChannel, ids []int) (tg.MessagesMessagesClass, error) {
msgIds := []tg.InputMessageClass{}
for _, id := range ids {
msgIds = append(msgIds, &tg.InputMessageID{ID: id})
}
messageRequest := tg.ChannelsGetMessagesRequest{
Channel: channel,
ID: msgIds,
}
res, err := client.ChannelsGetMessages(ctx, &messageRequest)
if err != nil {
return nil, err
}
return res, nil
}
func GetMessages(ctx context.Context, client *tg.Client, ids []int, channelId int64) ([]tg.MessageClass, error) {
channel, err := GetChannelById(ctx, client, channelId)
if err != nil {
return nil, err
}
batchSize := 200
batchCount := int(math.Ceil(float64(len(ids)) / float64(batchSize)))
g, _ := errgroup.WithContext(ctx)
g.SetLimit(runtime.NumCPU())
messageMap := make(map[int]*tg.MessagesChannelMessages)
var mapMu sync.Mutex
for i := range batchCount {
g.Go(func() error {
splitIds := ids[i*batchSize : min((i+1)*batchSize, len(ids))]
res, err := getTGMessagesBatch(ctx, client, channel, splitIds)
if err != nil {
return err
}
messages, ok := res.(*tg.MessagesChannelMessages)
if !ok {
return ErrInvalidChannelMessages
}
mapMu.Lock()
messageMap[i] = messages
mapMu.Unlock()
return nil
})
}
if err = g.Wait(); err != nil {
return nil, err
}
allMessages := []tg.MessageClass{}
for i := range batchCount {
allMessages = append(allMessages, messageMap[i].Messages...)
}
return allMessages, nil
}
func GetChunk(ctx context.Context, client *tg.Client, location tg.InputFileLocationClass, offset int64, limit int64) ([]byte, error) {
req := &tg.UploadGetFileRequest{
Offset: offset,
Limit: int(limit),
Location: location,
Precise: true,
}
r, err := client.UploadGetFile(ctx, req)
if err != nil {
return nil, err
}
switch result := r.(type) {
case *tg.UploadFile:
return result.Bytes, nil
default:
return nil, fmt.Errorf("unexpected type %T", r)
}
}
func GetMediaContent(ctx context.Context, client *tg.Client, location tg.InputFileLocationClass) (*bytes.Buffer, error) {
offset := int64(0)
limit := int64(1024 * 1024)
buff := &bytes.Buffer{}
for {
r, err := GetChunk(ctx, client, location, offset, limit)
if err != nil {
return buff, err
}
if len(r) == 0 {
break
}
buff.Write(r)
offset += int64(limit)
}
return buff, nil
}
func GetBotInfo(ctx context.Context, KV kv.KV, config *config.TGConfig, token string) (*types.BotInfo, error) {
var user *tg.User
client, _ := BotClient(ctx, KV, config, token, Middlewares(config, 5)...)
err := RunWithAuth(ctx, client, token, func(ctx context.Context) error {
user, _ = client.Self(ctx)
return nil
})
if err != nil {
return nil, err
}
return &types.BotInfo{Id: user.ID, UserName: user.Username, Token: token}, nil
}
func GetLocation(ctx context.Context, client *Client, fileId string, channelId int64, partId int64) (location *tg.InputDocumentFileLocation, err error) {
cache := cache.FromContext(ctx)
key := fmt.Sprintf("location:%s:%s:%d", client.UserId, fileId, partId)
err = cache.Get(key, location)
if err != nil {
channel, err := GetChannelById(ctx, client.Tg.API(), channelId)
if err != nil {
return nil, err
}
messageRequest := tg.ChannelsGetMessagesRequest{
Channel: channel,
ID: []tg.InputMessageClass{&tg.InputMessageID{ID: int(partId)}},
}
res, err := client.Tg.API().ChannelsGetMessages(ctx, &messageRequest)
if err != nil {
return nil, err
}
messages, _ := res.(*tg.MessagesChannelMessages)
item := messages.Messages[0].(*tg.Message)
media := item.Media.(*tg.MessageMediaDocument)
document := media.Document.(*tg.Document)
location = document.AsInputDocumentFileLocation()
cache.Set(key, location, 3600)
}
return location, nil
}
func CalculateChunkSize(start, end int64) int64 {
chunkSize := int64(1024 * 1024)
for chunkSize > 1024 && chunkSize > (end-start) {
chunkSize /= 2
}
return chunkSize
}