2023-08-13 04:15:19 +08:00
|
|
|
package services
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
|
2023-08-16 05:53:02 +08:00
|
|
|
"github.com/divyam234/teldrive/cache"
|
|
|
|
"github.com/divyam234/teldrive/schemas"
|
|
|
|
"github.com/divyam234/teldrive/utils"
|
2023-08-13 04:15:19 +08:00
|
|
|
|
2023-08-16 05:53:02 +08:00
|
|
|
"github.com/divyam234/teldrive/types"
|
2023-08-13 04:15:19 +08:00
|
|
|
|
2023-08-16 05:53:02 +08:00
|
|
|
"github.com/divyam234/teldrive/models"
|
2023-08-13 04:15:19 +08:00
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/gotd/td/telegram/message"
|
|
|
|
"github.com/gotd/td/telegram/uploader"
|
|
|
|
"github.com/gotd/td/tg"
|
|
|
|
"gorm.io/gorm"
|
|
|
|
)
|
|
|
|
|
|
|
|
type UploadService struct {
|
|
|
|
Db *gorm.DB
|
|
|
|
ChannelID int64
|
|
|
|
}
|
|
|
|
|
|
|
|
func (us *UploadService) GetUploadFileById(c *gin.Context) (*schemas.UploadOut, *types.AppError) {
|
|
|
|
uploadId := c.Param("id")
|
|
|
|
parts := []schemas.UploadPartOut{}
|
|
|
|
if err := us.Db.Model(&models.Upload{}).Order("part_no").Where("upload_id = ?", uploadId).Find(&parts).Error; err != nil {
|
|
|
|
return nil, &types.AppError{Error: errors.New("failed to fetch from db"), Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &schemas.UploadOut{Parts: parts}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (us *UploadService) DeleteUploadFile(c *gin.Context) *types.AppError {
|
|
|
|
uploadId := c.Param("id")
|
|
|
|
if err := us.Db.Where("upload_id = ?", uploadId).Delete(&models.Upload{}).Error; err != nil {
|
|
|
|
return &types.AppError{Error: errors.New("failed to delete upload"), Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (us *UploadService) UploadFile(c *gin.Context) (*schemas.UploadPartOut, *types.AppError) {
|
|
|
|
|
|
|
|
var uploadQuery schemas.UploadQuery
|
|
|
|
|
|
|
|
uploadQuery.PartNo = 1
|
|
|
|
uploadQuery.TotalParts = 1
|
|
|
|
|
|
|
|
if err := c.ShouldBindQuery(&uploadQuery); err != nil {
|
|
|
|
return nil, &types.AppError{Error: err, Code: http.StatusBadRequest}
|
|
|
|
}
|
|
|
|
|
|
|
|
if uploadQuery.Filename == "" {
|
|
|
|
return nil, &types.AppError{Error: errors.New("filename missing"), Code: http.StatusBadRequest}
|
|
|
|
}
|
|
|
|
|
|
|
|
uploadId := c.Param("id")
|
|
|
|
|
2023-08-19 21:53:09 +08:00
|
|
|
var uploadPart []models.Upload
|
|
|
|
|
|
|
|
us.Db.Model(&models.Upload{}).Where("upload_id = ?", uploadId).Where("part_no = ?", uploadQuery.PartNo).Find(&uploadPart)
|
|
|
|
|
|
|
|
if len(uploadPart) == 1 {
|
|
|
|
out := mapSchema(&uploadPart[0])
|
|
|
|
return out, nil
|
|
|
|
}
|
2023-08-13 04:15:19 +08:00
|
|
|
config := utils.GetConfig()
|
|
|
|
|
|
|
|
var tgClient *utils.Client
|
|
|
|
|
|
|
|
var err error
|
|
|
|
if config.MultiClient {
|
|
|
|
tgClient = utils.GetBotClient()
|
|
|
|
tgClient.Workload++
|
|
|
|
|
|
|
|
} else {
|
|
|
|
val, _ := c.Get("jwtUser")
|
|
|
|
jwtUser := val.(*types.JWTClaims)
|
|
|
|
userId, _ := strconv.Atoi(jwtUser.Subject)
|
2023-08-16 05:02:08 +08:00
|
|
|
tgClient, _, err = utils.GetAuthClient(jwtUser.TgSession, userId)
|
2023-08-13 04:15:19 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, &types.AppError{Error: err, Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
file := c.Request.Body
|
|
|
|
|
|
|
|
fileSize := c.Request.ContentLength
|
|
|
|
|
|
|
|
api := tgClient.Tg.API()
|
|
|
|
|
|
|
|
u := uploader.NewUploader(api).WithThreads(16).WithPartSize(512 * 1024)
|
|
|
|
|
|
|
|
sender := message.NewSender(api).WithUploader(u)
|
|
|
|
|
|
|
|
fileName := uploadQuery.Filename
|
|
|
|
|
|
|
|
if uploadQuery.TotalParts > 1 {
|
|
|
|
fileName = fmt.Sprintf("%s.part.%03d", fileName, uploadQuery.PartNo)
|
|
|
|
}
|
|
|
|
|
|
|
|
upload, err := u.Upload(c, uploader.NewUpload(fileName, file, fileSize))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, &types.AppError{Error: err, Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
|
|
|
document := message.UploadedDocument(upload).Filename(fileName).ForceFile(true)
|
|
|
|
|
|
|
|
res, err := cache.CachedFunction(utils.GetChannelById, fmt.Sprintf("channels:%d", us.ChannelID))(c, api, us.ChannelID)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, &types.AppError{Error: err, Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
|
|
|
channel := res.(*tg.Channel)
|
|
|
|
|
|
|
|
target := sender.To(&tg.InputPeerChannel{ChannelID: channel.ID, AccessHash: channel.AccessHash})
|
|
|
|
|
|
|
|
res, err = target.Media(c, document)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, &types.AppError{Error: err, Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
|
|
|
updates := res.(*tg.Updates)
|
|
|
|
|
|
|
|
msgId := updates.Updates[0].(*tg.UpdateMessageID).ID
|
|
|
|
|
|
|
|
partUpload := &models.Upload{
|
|
|
|
Name: fileName,
|
|
|
|
UploadId: uploadId,
|
|
|
|
PartId: msgId,
|
|
|
|
ChannelID: us.ChannelID,
|
|
|
|
Size: fileSize,
|
|
|
|
PartNo: uploadQuery.PartNo,
|
|
|
|
TotalParts: uploadQuery.TotalParts,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := us.Db.Create(partUpload).Error; err != nil {
|
|
|
|
return nil, &types.AppError{Error: errors.New("failed to upload part"), Code: http.StatusInternalServerError}
|
|
|
|
}
|
|
|
|
|
2023-08-19 21:53:09 +08:00
|
|
|
out := mapSchema(partUpload)
|
2023-08-13 04:15:19 +08:00
|
|
|
|
|
|
|
return out, nil
|
|
|
|
}
|
2023-08-19 21:53:09 +08:00
|
|
|
|
|
|
|
func mapSchema(in *models.Upload) *schemas.UploadPartOut {
|
|
|
|
out := &schemas.UploadPartOut{
|
|
|
|
ID: in.ID,
|
|
|
|
Name: in.Name,
|
|
|
|
PartId: in.PartId,
|
|
|
|
ChannelID: in.ChannelID,
|
|
|
|
PartNo: in.PartNo,
|
|
|
|
TotalParts: in.TotalParts,
|
|
|
|
Size: in.Size,
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|