@@ -3,6 +3,7 @@ package sync
3
3
import (
4
4
"context"
5
5
"fmt"
6
+ "io"
6
7
7
8
"github.com/docker/go-units"
8
9
"github.com/go-viper/mapstructure/v2"
@@ -25,34 +26,32 @@ var MaxUnaryFileSize = int64(units.MB)
25
26
// uses StreamingDataCaptureUpload API so as to not exceed the unary response size.
26
27
// Otherwise, uploads data over DataCaptureUpload API.
27
28
// Note: the bytes size returned is the size of the input file. It only returns a non 0 value in the success case.
28
- func uploadDataCaptureFile (ctx context.Context , f * data.CaptureFile , conn cloudConn , logger logging.Logger ) (uint64 , error ) {
29
+ func uploadDataCaptureFile (ctx context.Context , f * data.CaptureFile , conn cloudConn , flag bool , logger logging.Logger ) (uint64 , error ) {
29
30
logger .Debugf ("preparing to upload data capture file: %s, size: %d" , f .GetPath (), f .Size ())
30
31
31
32
md := f .ReadMetadata ()
33
+
34
+ // camera.GetImages is a special case. For that API we make 2 binary data upload requests
35
+ if md .GetType () == v1 .DataType_DATA_TYPE_BINARY_SENSOR && md .GetMethodName () == data .GetImages {
36
+ return uint64 (f .Size ()), uploadGetImages (ctx , conn , md , f , logger )
37
+ }
38
+
39
+ metaData := uploadMetadata (conn .partID , md , md .GetFileExtension ())
40
+ if md .GetType () == v1 .DataType_DATA_TYPE_BINARY_SENSOR && flag {
41
+ return uint64 (f .Size ()), uploadChunkedBinaryData (ctx , conn .client , metaData , f , logger )
42
+ }
43
+
32
44
sensorData , err := data .SensorDataFromCaptureFile (f )
33
45
if err != nil {
34
46
return 0 , errors .Wrap (err , "error reading sensor data" )
35
47
}
36
48
37
- // Do not attempt to upload a file without any sensor readings.
38
49
if len (sensorData ) == 0 {
39
50
logger .Warnf ("ignoring and deleting empty capture file without syncing it: %s" , f .GetPath ())
40
51
// log here as this will delete a .capture file without uploading it and without moving it to the failed directory
41
52
return 0 , nil
42
53
}
43
54
44
- if md .GetType () == v1 .DataType_DATA_TYPE_BINARY_SENSOR && len (sensorData ) > 1 {
45
- return 0 , fmt .Errorf ("binary sensor data file with more than one sensor reading is not supported: %s" , f .GetPath ())
46
- }
47
-
48
- // camera.GetImages is a special case. For that API we make 2 binary data upload requests
49
- if md .GetType () == v1 .DataType_DATA_TYPE_BINARY_SENSOR && md .GetMethodName () == data .GetImages {
50
- logger .Debugf ("attemping to upload camera.GetImages data: %s" , f .GetPath ())
51
-
52
- return uint64 (f .Size ()), uploadGetImages (ctx , conn , md , sensorData [0 ], f .Size (), f .GetPath (), logger )
53
- }
54
-
55
- metaData := uploadMetadata (conn .partID , md , md .GetFileExtension ())
56
55
return uint64 (f .Size ()), uploadSensorData (ctx , conn .client , metaData , sensorData , f .Size (), f .GetPath (), logger )
57
56
}
58
57
@@ -73,11 +72,26 @@ func uploadGetImages(
73
72
ctx context.Context ,
74
73
conn cloudConn ,
75
74
md * v1.DataCaptureMetadata ,
76
- sd * v1.SensorData ,
77
- size int64 ,
78
- path string ,
75
+ f * data.CaptureFile ,
79
76
logger logging.Logger ,
80
77
) error {
78
+ logger .Debugf ("attemping to upload camera.GetImages data: %s" , f .GetPath ())
79
+
80
+ sensorData , err := data .SensorDataFromCaptureFile (f )
81
+ if err != nil {
82
+ return errors .Wrap (err , "error reading sensor data" )
83
+ }
84
+
85
+ if len (sensorData ) == 0 {
86
+ logger .Warnf ("ignoring and deleting empty capture file without syncing it: %s" , f .GetPath ())
87
+ // log here as this will delete a .capture file without uploading it and without moving it to the failed directory
88
+ return nil
89
+ }
90
+
91
+ if len (sensorData ) > 1 {
92
+ return fmt .Errorf ("binary sensor data file with more than one sensor reading is not supported: %s" , f .GetPath ())
93
+ }
94
+ sd := sensorData [0 ]
81
95
var res pb.GetImagesResponse
82
96
if err := mapstructure .Decode (sd .GetStruct ().AsMap (), & res ); err != nil {
83
97
return errors .Wrap (err , "failed to decode camera.GetImagesResponse" )
@@ -100,7 +114,7 @@ func uploadGetImages(
100
114
metadata := uploadMetadata (conn .partID , md , getFileExtFromImageFormat (img .GetFormat ()))
101
115
// TODO: This is wrong as the size describes the size of the entire GetImages response, but we are only
102
116
// uploading one of the 2 images in that response here.
103
- if err := uploadSensorData (ctx , conn .client , metadata , newSensorData , size , path , logger ); err != nil {
117
+ if err := uploadSensorData (ctx , conn .client , metadata , newSensorData , f . Size (), f . GetPath () , logger ); err != nil {
104
118
return errors .Wrapf (err , "failed uploading GetImages image index: %d" , i )
105
119
}
106
120
}
@@ -123,6 +137,45 @@ func getImagesTimestamps(res *pb.GetImagesResponse, sensorData *v1.SensorData) (
123
137
return timeRequested , timeReceived
124
138
}
125
139
140
+ func uploadChunkedBinaryData (
141
+ ctx context.Context ,
142
+ client v1.DataSyncServiceClient ,
143
+ uploadMD * v1.UploadMetadata ,
144
+ f * data.CaptureFile ,
145
+ logger logging.Logger ,
146
+ ) error {
147
+ // If it's a large binary file, we need to upload it in chunks.
148
+ logger .Debugf ("attempting to upload large binary file using StreamingDataCaptureUpload, file: %s" , f .GetPath ())
149
+ var smd v1.SensorMetadata
150
+ r , err := f .BinaryReader (& smd )
151
+ if err != nil {
152
+ return err
153
+ }
154
+ c , err := client .StreamingDataCaptureUpload (ctx )
155
+ if err != nil {
156
+ return errors .Wrap (err , "error creating StreamingDataCaptureUpload client" )
157
+ }
158
+
159
+ // First send metadata.
160
+ streamMD := & v1.StreamingDataCaptureUploadRequest_Metadata {
161
+ Metadata : & v1.DataCaptureUploadMetadata {
162
+ UploadMetadata : uploadMD ,
163
+ SensorMetadata : & smd ,
164
+ },
165
+ }
166
+ if err := c .Send (& v1.StreamingDataCaptureUploadRequest {UploadPacket : streamMD }); err != nil {
167
+ return errors .Wrap (err , "StreamingDataCaptureUpload failed sending metadata" )
168
+ }
169
+
170
+ // Then call the function to send the rest.
171
+ if err := sendChunkedStreamingDCRequests (ctx , c , r , f .GetPath (), logger ); err != nil {
172
+ return errors .Wrap (err , "StreamingDataCaptureUpload failed to sync" )
173
+ }
174
+
175
+ _ , err = c .CloseAndRecv ()
176
+ return errors .Wrap (err , "StreamingDataCaptureUpload CloseAndRecv failed" )
177
+ }
178
+
126
179
func uploadSensorData (
127
180
ctx context.Context ,
128
181
client v1.DataSyncServiceClient ,
@@ -171,6 +224,52 @@ func uploadSensorData(
171
224
return errors .Wrap (err , "DataCaptureUpload failed" )
172
225
}
173
226
227
+ func sendChunkedStreamingDCRequests (
228
+ ctx context.Context ,
229
+ stream v1.DataSyncService_StreamingDataCaptureUploadClient ,
230
+ r io.Reader ,
231
+ path string ,
232
+ logger logging.Logger ,
233
+ ) error {
234
+ chunk := make ([]byte , UploadChunkSize )
235
+ // Loop until there is no more content to send.
236
+ chunkCount := 0
237
+ for {
238
+ select {
239
+ case <- ctx .Done ():
240
+ return ctx .Err ()
241
+ default :
242
+ n , errRead := r .Read (chunk )
243
+ if n > 0 {
244
+ // if there is data, send it
245
+ // Build request with contents.
246
+ uploadReq := & v1.StreamingDataCaptureUploadRequest {
247
+ UploadPacket : & v1.StreamingDataCaptureUploadRequest_Data {
248
+ Data : chunk [:n ],
249
+ },
250
+ }
251
+
252
+ // Send request
253
+ logger .Debugf ("datasync.StreamingDataCaptureUpload sending chunk %d for file: %s" , chunkCount , path )
254
+ if errSend := stream .Send (uploadReq ); errSend != nil {
255
+ return errSend
256
+ }
257
+ }
258
+
259
+ // if we reached the end of the file return nil err (success)
260
+ if errors .Is (errRead , io .EOF ) {
261
+ return nil
262
+ }
263
+
264
+ // if Read hit an unexpected error, return the error
265
+ if errRead != nil {
266
+ return errRead
267
+ }
268
+ chunkCount ++
269
+ }
270
+ }
271
+ }
272
+
174
273
func sendStreamingDCRequests (
175
274
ctx context.Context ,
176
275
stream v1.DataSyncService_StreamingDataCaptureUploadClient ,
0 commit comments