1
- // Copyright (c) 2021 Tulir Asokan
1
+ // Copyright (c) 2024 Tulir Asokan
2
2
//
3
3
// This Source Code Form is subject to the terms of the Mozilla Public
4
4
// License, v. 2.0. If a copy of the MPL was not distributed with this
@@ -14,8 +14,10 @@ import (
14
14
"encoding/base64"
15
15
"encoding/json"
16
16
"fmt"
17
+ "io"
17
18
"net/http"
18
19
"net/url"
20
+ "os"
19
21
20
22
"go.mau.fi/util/random"
21
23
@@ -87,7 +89,43 @@ func (cli *Client) Upload(ctx context.Context, plaintext []byte, appInfo MediaTy
87
89
dataHash := sha256 .Sum256 (dataToUpload )
88
90
resp .FileEncSHA256 = dataHash [:]
89
91
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 )
91
129
return
92
130
}
93
131
@@ -125,11 +163,31 @@ func (cli *Client) UploadNewsletter(ctx context.Context, data []byte, appInfo Me
125
163
resp .FileLength = uint64 (len (data ))
126
164
hash := sha256 .Sum256 (data )
127
165
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
+ }
128
186
err = cli .rawUpload (ctx , data , resp .FileSHA256 , appInfo , true , & resp )
129
187
return
130
188
}
131
189
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 {
133
191
mediaConn , err := cli .refreshMediaConn (false )
134
192
if err != nil {
135
193
return fmt .Errorf ("failed to refresh media connections: %w" , err )
@@ -168,7 +226,7 @@ func (cli *Client) rawUpload(ctx context.Context, dataToUpload, fileHash []byte,
168
226
RawQuery : q .Encode (),
169
227
}
170
228
171
- req , err := http .NewRequestWithContext (ctx , http .MethodPost , uploadURL .String (), bytes . NewReader ( dataToUpload ) )
229
+ req , err := http .NewRequestWithContext (ctx , http .MethodPost , uploadURL .String (), dataToUpload )
172
230
if err != nil {
173
231
return fmt .Errorf ("failed to prepare request: %w" , err )
174
232
}
0 commit comments