You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
178 lines
4.1 KiB
178 lines
4.1 KiB
3 years ago
|
package tdesktop
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"crypto/md5" // #nosec G501
|
||
|
"encoding/binary"
|
||
|
"io"
|
||
|
"io/fs"
|
||
|
"math"
|
||
|
|
||
|
"github.com/go-faster/errors"
|
||
|
"go.uber.org/multierr"
|
||
|
)
|
||
|
|
||
|
type tdesktopFile struct {
|
||
|
data []byte
|
||
|
n int
|
||
|
version uint32
|
||
|
}
|
||
|
|
||
|
func open(filesystem fs.FS, fileName string) (*tdesktopFile, error) {
|
||
|
suffixes := []string{"0", "1", "s"}
|
||
|
|
||
|
tryRead := func(p string) (_ *tdesktopFile, rErr error) {
|
||
|
f, err := filesystem.Open(p)
|
||
|
if err != nil {
|
||
|
return nil, errors.Wrap(err, "open")
|
||
|
}
|
||
|
defer multierr.AppendInvoke(&rErr, multierr.Close(f))
|
||
|
|
||
|
return fromFile(f)
|
||
|
}
|
||
|
|
||
|
for _, suffix := range suffixes {
|
||
|
p := fileName + suffix
|
||
|
if _, err := fs.Stat(filesystem, p); err != nil {
|
||
|
if errors.Is(err, fs.ErrNotExist) ||
|
||
|
errors.Is(err, fs.ErrPermission) {
|
||
|
continue
|
||
|
}
|
||
|
return nil, errors.Wrap(err, "stat")
|
||
|
}
|
||
|
|
||
|
f, err := tryRead(p)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, io.ErrUnexpectedEOF) {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
var magicErr *WrongMagicError
|
||
|
if errors.As(err, &magicErr) {
|
||
|
continue
|
||
|
}
|
||
|
return nil, errors.Wrap(err, "read tdesktop file")
|
||
|
}
|
||
|
|
||
|
return f, nil
|
||
|
}
|
||
|
|
||
|
return nil, errors.Errorf("file %q not found", fileName)
|
||
|
}
|
||
|
|
||
|
var tdesktopFileMagic = [4]byte{'T', 'D', 'F', '$'}
|
||
|
|
||
|
// fromFile creates new Telegram Desktop storage file.
|
||
|
// Based on https://github.com/telegramdesktop/tdesktop/blob/v2.9.8/Telegram/SourceFiles/storage/details/storage_file_utilities.cpp#L473.
|
||
|
func fromFile(r io.Reader) (*tdesktopFile, error) {
|
||
|
buf := make([]byte, 16)
|
||
|
if _, err := io.ReadFull(r, buf[:8]); err != nil {
|
||
|
return nil, errors.Wrap(err, "read magic and version")
|
||
|
}
|
||
|
|
||
|
var magic, version [4]byte
|
||
|
copy(magic[:], buf[:4])
|
||
|
// TODO(tdakkota): check version
|
||
|
copy(version[:], buf[4:8])
|
||
|
if magic != tdesktopFileMagic {
|
||
|
return nil, &WrongMagicError{
|
||
|
Magic: magic,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
data, err := io.ReadAll(r)
|
||
|
if err != nil {
|
||
|
return nil, errors.Wrap(err, "read data")
|
||
|
}
|
||
|
if l := len(data); l < 16 {
|
||
|
return nil, errors.Errorf("invalid data length %d", l)
|
||
|
}
|
||
|
hash := data[len(data)-16:]
|
||
|
data = data[:len(data)-16]
|
||
|
|
||
|
computedHash := telegramFileHash(data, version)
|
||
|
if !bytes.Equal(computedHash[:], hash) {
|
||
|
return nil, errors.New("hash mismatch")
|
||
|
}
|
||
|
|
||
|
v := binary.LittleEndian.Uint32(version[:])
|
||
|
return &tdesktopFile{
|
||
|
data: data,
|
||
|
version: v,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
func writeFile(w io.Writer, data []byte, version [4]byte) error {
|
||
|
if _, err := w.Write(tdesktopFileMagic[:]); err != nil {
|
||
|
return errors.Wrap(err, "write magic")
|
||
|
}
|
||
|
if _, err := w.Write(version[:]); err != nil {
|
||
|
return errors.Wrap(err, "write version")
|
||
|
}
|
||
|
if _, err := w.Write(data); err != nil {
|
||
|
return errors.Wrap(err, "write data")
|
||
|
}
|
||
|
hash := telegramFileHash(data, version)
|
||
|
if _, err := w.Write(hash[:]); err != nil {
|
||
|
return errors.Wrap(err, "write hash")
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func telegramFileHash(data []byte, version [4]byte) (r [md5.Size]byte) {
|
||
|
h := md5.New() // #nosec G401
|
||
|
_, _ = h.Write(data)
|
||
|
var packedLength [4]byte
|
||
|
binary.LittleEndian.PutUint32(packedLength[:], uint32(len(data)))
|
||
|
_, _ = h.Write(packedLength[:])
|
||
|
_, _ = h.Write(version[:])
|
||
|
_, _ = h.Write(tdesktopFileMagic[:])
|
||
|
h.Sum(r[:0])
|
||
|
return r
|
||
|
}
|
||
|
|
||
|
func (f *tdesktopFile) readArray() ([]byte, error) {
|
||
|
data, skip, err := readArray(f.data[f.n:], binary.BigEndian)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
f.n += skip
|
||
|
return data, nil
|
||
|
}
|
||
|
|
||
|
func readArray(data []byte, order binary.ByteOrder) (array []byte, n int, _ error) {
|
||
|
if len(data) < 4 {
|
||
|
return nil, 0, io.ErrUnexpectedEOF
|
||
|
}
|
||
|
// See https://github.com/qt/qtbase/blob/5.15.2/src/corelib/text/qbytearray.cpp#L3314.
|
||
|
length := order.Uint32(data)
|
||
|
if length == 0xffffffff {
|
||
|
return nil, 4, nil
|
||
|
}
|
||
|
|
||
|
if uint64(length) >= uint64(len(data)) {
|
||
|
return nil, 0, io.ErrUnexpectedEOF
|
||
|
}
|
||
|
r := data[4 : 4+length]
|
||
|
return r, len(r) + 4, nil
|
||
|
}
|
||
|
|
||
|
func writeArray(writer io.Writer, data []byte, order binary.ByteOrder) error {
|
||
|
length := len(data)
|
||
|
if uint64(length) > uint64(math.MaxUint32) {
|
||
|
return errors.Errorf("data length too big (%d)", length)
|
||
|
}
|
||
|
|
||
|
r := make([]byte, 4)
|
||
|
order.PutUint32(r, uint32(length))
|
||
|
if _, err := writer.Write(r); err != nil {
|
||
|
return errors.Wrap(err, "write length")
|
||
|
}
|
||
|
|
||
|
if _, err := writer.Write(data); err != nil {
|
||
|
return errors.Wrap(err, "write data")
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|