Skip to content

Commit 82d17aa

Browse files
committed
upload: add support for reading data from io.Reader
Fixes #94
1 parent 82a2975 commit 82d17aa

File tree

2 files changed

+109
-4
lines changed

2 files changed

+109
-4
lines changed

upload.go

+62-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright (c) 2021 Tulir Asokan
1+
// Copyright (c) 2024 Tulir Asokan
22
//
33
// This Source Code Form is subject to the terms of the Mozilla Public
44
// License, v. 2.0. If a copy of the MPL was not distributed with this
@@ -14,8 +14,10 @@ import (
1414
"encoding/base64"
1515
"encoding/json"
1616
"fmt"
17+
"io"
1718
"net/http"
1819
"net/url"
20+
"os"
1921

2022
"go.mau.fi/util/random"
2123

@@ -87,7 +89,43 @@ func (cli *Client) Upload(ctx context.Context, plaintext []byte, appInfo MediaTy
8789
dataHash := sha256.Sum256(dataToUpload)
8890
resp.FileEncSHA256 = dataHash[:]
8991

90-
err = cli.rawUpload(ctx, dataToUpload, resp.FileEncSHA256, appInfo, false, &resp)
92+
err = cli.rawUpload(ctx, bytes.NewReader(dataToUpload), resp.FileEncSHA256, appInfo, false, &resp)
93+
return
94+
}
95+
96+
// UploadReader uploads the given attachment to WhatsApp servers.
97+
//
98+
// This is otherwise identical to [Upload], but it reads the plaintext from an [io.Reader] instead of a byte slice.
99+
// A temporary file is required for the encryption process. If tempFile is nil, a temporary file will be created
100+
// and deleted after the upload.
101+
func (cli *Client) UploadReader(ctx context.Context, plaintext io.Reader, tempFile io.ReadWriteSeeker, appInfo MediaType) (resp UploadResponse, err error) {
102+
resp.MediaKey = random.Bytes(32)
103+
iv, cipherKey, macKey, _ := getMediaKeys(resp.MediaKey, appInfo)
104+
if tempFile == nil {
105+
tempFile, err = os.CreateTemp("", "whatsmeow-upload-*")
106+
if err != nil {
107+
err = fmt.Errorf("failed to create temporary file: %w", err)
108+
return
109+
}
110+
fmt.Println("OPENED TEMPFILE", tempFile.(*os.File).Name())
111+
defer func() {
112+
tempFileFile := tempFile.(*os.File)
113+
_ = tempFileFile.Close()
114+
_ = os.Remove(tempFileFile.Name())
115+
fmt.Println("REMOVED TEMPFILE", tempFile.(*os.File).Name())
116+
}()
117+
}
118+
resp.FileSHA256, resp.FileEncSHA256, resp.FileLength, err = cbcutil.EncryptStream(cipherKey, iv, macKey, plaintext, tempFile)
119+
if err != nil {
120+
err = fmt.Errorf("failed to encrypt file: %w", err)
121+
return
122+
}
123+
_, err = tempFile.Seek(0, io.SeekStart)
124+
if err != nil {
125+
err = fmt.Errorf("failed to seek to start of temporary file: %w", err)
126+
return
127+
}
128+
err = cli.rawUpload(ctx, tempFile, resp.FileEncSHA256, appInfo, false, &resp)
91129
return
92130
}
93131

@@ -125,11 +163,31 @@ func (cli *Client) UploadNewsletter(ctx context.Context, data []byte, appInfo Me
125163
resp.FileLength = uint64(len(data))
126164
hash := sha256.Sum256(data)
127165
resp.FileSHA256 = hash[:]
166+
err = cli.rawUpload(ctx, bytes.NewReader(data), resp.FileSHA256, appInfo, true, &resp)
167+
return
168+
}
169+
170+
// UploadNewsletterReader uploads the given attachment to WhatsApp servers without encrypting it first.
171+
//
172+
// This is otherwise identical to [Upload], but it reads the plaintext from an [io.Reader] instead of a byte slice.
173+
// Unlike [UploadReader], this does not require a temporary file. However, the data needs to be hashed first,
174+
// so an [io.ReadSeeker] is required to be able to read the data twice.
175+
func (cli *Client) UploadNewsletterReader(ctx context.Context, data io.ReadSeeker, appInfo MediaType) (resp UploadResponse, err error) {
176+
hasher := sha256.New()
177+
var fileLength int64
178+
fileLength, err = io.Copy(hasher, data)
179+
resp.FileLength = uint64(fileLength)
180+
resp.FileSHA256 = hasher.Sum(nil)
181+
_, err = data.Seek(0, io.SeekStart)
182+
if err != nil {
183+
err = fmt.Errorf("failed to seek to start of data: %w", err)
184+
return
185+
}
128186
err = cli.rawUpload(ctx, data, resp.FileSHA256, appInfo, true, &resp)
129187
return
130188
}
131189

132-
func (cli *Client) rawUpload(ctx context.Context, dataToUpload, fileHash []byte, appInfo MediaType, newsletter bool, resp *UploadResponse) error {
190+
func (cli *Client) rawUpload(ctx context.Context, dataToUpload io.Reader, fileHash []byte, appInfo MediaType, newsletter bool, resp *UploadResponse) error {
133191
mediaConn, err := cli.refreshMediaConn(false)
134192
if err != nil {
135193
return fmt.Errorf("failed to refresh media connections: %w", err)
@@ -168,7 +226,7 @@ func (cli *Client) rawUpload(ctx context.Context, dataToUpload, fileHash []byte,
168226
RawQuery: q.Encode(),
169227
}
170228

171-
req, err := http.NewRequestWithContext(ctx, http.MethodPost, uploadURL.String(), bytes.NewReader(dataToUpload))
229+
req, err := http.NewRequestWithContext(ctx, http.MethodPost, uploadURL.String(), dataToUpload)
172230
if err != nil {
173231
return fmt.Errorf("failed to prepare request: %w", err)
174232
}

util/cbcutil/cbc.go

+47
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,10 @@ import (
1818
"bytes"
1919
"crypto/aes"
2020
"crypto/cipher"
21+
"crypto/hmac"
2122
"crypto/rand"
23+
"crypto/sha256"
24+
"errors"
2225
"fmt"
2326
"io"
2427
)
@@ -99,3 +102,47 @@ func unpad(src []byte) ([]byte, error) {
99102

100103
return src[:(length - padLen)], nil
101104
}
105+
106+
func EncryptStream(key, iv, macKey []byte, plaintext io.Reader, ciphertext io.Writer) ([]byte, []byte, uint64, error) {
107+
block, err := aes.NewCipher(key)
108+
if err != nil {
109+
return nil, nil, 0, fmt.Errorf("failed to create cipher: %w", err)
110+
}
111+
cbc := cipher.NewCBCEncrypter(block, iv)
112+
113+
plainHasher := sha256.New()
114+
cipherHasher := sha256.New()
115+
cipherMAC := hmac.New(sha256.New, macKey)
116+
cipherMAC.Write(iv)
117+
118+
buf := make([]byte, 32*1024)
119+
var size int
120+
hasMore := true
121+
for hasMore {
122+
var n int
123+
n, err = io.ReadFull(plaintext, buf)
124+
plainHasher.Write(buf[:n])
125+
size += n
126+
if errors.Is(err, io.EOF) || errors.Is(err, io.ErrUnexpectedEOF) {
127+
padding := aes.BlockSize - size%aes.BlockSize
128+
buf = append(buf[:n], bytes.Repeat([]byte{byte(padding)}, padding)...)
129+
hasMore = false
130+
} else if err != nil {
131+
return nil, nil, 0, fmt.Errorf("failed to read file: %w", err)
132+
}
133+
cbc.CryptBlocks(buf, buf)
134+
cipherMAC.Write(buf)
135+
cipherHasher.Write(buf)
136+
_, err = ciphertext.Write(buf)
137+
if err != nil {
138+
return nil, nil, 0, fmt.Errorf("failed to write file: %w", err)
139+
}
140+
}
141+
mac := cipherMAC.Sum(nil)[:10]
142+
cipherHasher.Write(mac)
143+
_, err = ciphertext.Write(mac)
144+
if err != nil {
145+
return nil, nil, 0, fmt.Errorf("failed to write checksum to file: %w", err)
146+
}
147+
return plainHasher.Sum(nil), cipherHasher.Sum(nil), uint64(size), nil
148+
}

0 commit comments

Comments
 (0)