7a04f298d2
- update to latest telegram layer - remove some references to fields in tg.Entities that don't exist in the schema - originally added here: https://github.com/beeper/td/commit/820929062a2ba0104397bc01235ab58a9cff780e - referenced here - https://github.com/mautrix/telegramgo/commit/124f0967ed195b5a380c9bd02e170ada9710dde3 - https://github.com/mautrix/telegramgo/commit/4205047aab2e0639217148b5d125bfaab668bd8e
130 lines
2.6 KiB
Go
130 lines
2.6 KiB
Go
package uploader
|
|
|
|
import (
|
|
"context"
|
|
"io"
|
|
|
|
"github.com/go-faster/errors"
|
|
|
|
"go.mau.fi/mautrix-telegram/pkg/gotd/bin"
|
|
"go.mau.fi/mautrix-telegram/pkg/gotd/syncio"
|
|
"go.mau.fi/mautrix-telegram/pkg/gotd/tdsync"
|
|
"go.mau.fi/mautrix-telegram/pkg/gotd/tg"
|
|
"go.mau.fi/mautrix-telegram/pkg/gotd/tgerr"
|
|
)
|
|
|
|
type part struct {
|
|
id int
|
|
buf *bin.Buffer
|
|
upload *Upload
|
|
}
|
|
|
|
func (u *Uploader) uploadBigFilePart(ctx context.Context, p part) (int, error) {
|
|
defer u.pool.Put(p.buf)
|
|
|
|
// Upload loop.
|
|
for {
|
|
r, err := u.rpc.UploadSaveBigFilePart(ctx, &tg.UploadSaveBigFilePartRequest{
|
|
FileID: p.upload.id,
|
|
FilePart: p.id,
|
|
FileTotalParts: p.upload.totalParts,
|
|
Bytes: p.buf.Buf,
|
|
})
|
|
|
|
if flood, err := tgerr.FloodWait(ctx, err); err != nil {
|
|
if flood {
|
|
continue
|
|
}
|
|
return 0, errors.Wrapf(err, "send upload part %d RPC", p.id)
|
|
}
|
|
|
|
// If Telegram returned false, it seems save is not successful, so we retry to send.
|
|
if r {
|
|
return p.buf.Len(), nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func (u *Uploader) bigLoop(ctx context.Context, threads int, upload *Upload) error { // nolint:gocognit
|
|
g := tdsync.NewCancellableGroup(ctx)
|
|
toSend := make(chan part, threads)
|
|
|
|
// Run read loop
|
|
r := syncio.NewReader(upload.from)
|
|
g.Go(func(ctx context.Context) error {
|
|
last := false
|
|
totalStreamSize := 0
|
|
|
|
for {
|
|
buf := u.pool.GetSize(u.partSize)
|
|
|
|
n, err := io.ReadFull(r, buf.Buf)
|
|
if n > 0 {
|
|
totalStreamSize += n
|
|
}
|
|
switch {
|
|
case errors.Is(err, io.ErrUnexpectedEOF):
|
|
last = true
|
|
if upload.totalParts == -1 {
|
|
totalParts := (totalStreamSize + u.partSize - 1) / u.partSize
|
|
upload.totalParts = int(totalParts)
|
|
}
|
|
case errors.Is(err, io.EOF):
|
|
u.pool.Put(buf)
|
|
|
|
close(toSend)
|
|
return nil
|
|
case err != nil:
|
|
u.pool.Put(buf)
|
|
|
|
return errors.Wrap(err, "read source")
|
|
}
|
|
|
|
buf.Buf = buf.Buf[:n]
|
|
nextPart := part{
|
|
id: int(upload.sentParts.Load()),
|
|
buf: buf,
|
|
upload: upload,
|
|
}
|
|
select {
|
|
case toSend <- nextPart:
|
|
upload.sentParts.Inc()
|
|
if last {
|
|
close(toSend)
|
|
return nil
|
|
}
|
|
case <-ctx.Done():
|
|
u.pool.Put(buf)
|
|
|
|
return ctx.Err()
|
|
}
|
|
}
|
|
})
|
|
|
|
for i := 0; i < threads; i++ {
|
|
g.Go(func(ctx context.Context) error {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
case part, ok := <-toSend:
|
|
if !ok {
|
|
return nil
|
|
}
|
|
|
|
n, err := u.uploadBigFilePart(ctx, part)
|
|
if err != nil {
|
|
return errors.Wrap(err, "upload part")
|
|
}
|
|
|
|
if err := u.callback(ctx, upload.confirm(part.id, n)); err != nil {
|
|
return errors.Wrap(err, "progress callback")
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
return g.Wait()
|
|
}
|