From 8158687dd3b8626bce74a70929361808b362e5f5 Mon Sep 17 00:00:00 2001 From: Cristian Lupu Date: Thu, 16 Apr 2020 00:50:09 +0300 Subject: [PATCH] Implement maximum batch request size (#874) * Implement maximum request size * Add test when batch exceeds the size * Execute check before gzip * Remove stub request --- Analytics/Classes/Internal/SEGHTTPClient.h | 2 +- Analytics/Classes/Internal/SEGHTTPClient.m | 8 +++++++- AnalyticsTests/HTTPClientTest.swift | 22 +++++++++++++++++----- 3 files changed, 25 insertions(+), 7 deletions(-) diff --git a/Analytics/Classes/Internal/SEGHTTPClient.h b/Analytics/Classes/Internal/SEGHTTPClient.h index 284133f88..59a6897cd 100644 --- a/Analytics/Classes/Internal/SEGHTTPClient.h +++ b/Analytics/Classes/Internal/SEGHTTPClient.h @@ -33,7 +33,7 @@ NS_ASSUME_NONNULL_BEGIN * NOTE: You need to re-dispatch within the completionHandler onto a desired queue to avoid threading issues. * Completion handlers are called on a dispatch queue internal to SEGHTTPClient. */ -- (NSURLSessionUploadTask *)upload:(JSON_DICT)batch forWriteKey:(NSString *)writeKey completionHandler:(void (^)(BOOL retry))completionHandler; +- (nullable NSURLSessionUploadTask *)upload:(JSON_DICT)batch forWriteKey:(NSString *)writeKey completionHandler:(void (^)(BOOL retry))completionHandler; - (NSURLSessionDataTask *)settingsForWriteKey:(NSString *)writeKey completionHandler:(void (^)(BOOL success, JSON_DICT _Nullable settings))completionHandler; diff --git a/Analytics/Classes/Internal/SEGHTTPClient.m b/Analytics/Classes/Internal/SEGHTTPClient.m index 21c24bd20..d89b2fd27 100644 --- a/Analytics/Classes/Internal/SEGHTTPClient.m +++ b/Analytics/Classes/Internal/SEGHTTPClient.m @@ -2,6 +2,7 @@ #import "NSData+SEGGZIP.h" #import "SEGAnalyticsUtils.h" +static const NSUInteger kMaxBatchSize = 475000; // 475KB @implementation SEGHTTPClient @@ -66,7 +67,7 @@ - (void)dealloc } -- (NSURLSessionUploadTask *)upload:(NSDictionary *)batch forWriteKey:(NSString *)writeKey completionHandler:(void (^)(BOOL retry))completionHandler +- (nullable NSURLSessionUploadTask *)upload:(NSDictionary *)batch forWriteKey:(NSString *)writeKey completionHandler:(void (^)(BOOL retry))completionHandler { // batch = SEGCoerceDictionary(batch); NSURLSession *session = [self sessionForWriteKey:writeKey]; @@ -93,6 +94,11 @@ - (NSURLSessionUploadTask *)upload:(NSDictionary *)batch forWriteKey:(NSString * completionHandler(NO); // Don't retry this batch. return nil; } + if (payload.length >= kMaxBatchSize) { + SEGLog(@"Payload exceeded the limit of %luKB per batch", kMaxBatchSize / 1000); + completionHandler(NO); + return nil; + } NSData *gzippedPayload = [payload seg_gzippedData]; NSURLSessionUploadTask *task = [session uploadTaskWithRequest:request fromData:gzippedPayload completionHandler:^(NSData *_Nullable data, NSURLResponse *_Nullable response, NSError *_Nullable error) { diff --git a/AnalyticsTests/HTTPClientTest.swift b/AnalyticsTests/HTTPClientTest.swift index c57350627..d25c2b696 100644 --- a/AnalyticsTests/HTTPClientTest.swift +++ b/AnalyticsTests/HTTPClientTest.swift @@ -127,7 +127,7 @@ class HTTPClientTest: QuickSpec { done = true } expect(done).toEventually(beTrue()) - expect(task.state).toEventually(equal(URLSessionTask.State.completed)) + expect(task?.state).toEventually(equal(URLSessionTask.State.completed)) } it("asks to retry for 3xx response") { @@ -142,7 +142,7 @@ class HTTPClientTest: QuickSpec { done = true } expect(done).toEventually(beTrue()) - expect(task.state).toEventually(equal(URLSessionTask.State.completed)) + expect(task?.state).toEventually(equal(URLSessionTask.State.completed)) } it("does not ask to retry for 4xx response") { @@ -157,7 +157,7 @@ class HTTPClientTest: QuickSpec { done = true } expect(done).toEventually(beTrue()) - expect(task.state).toEventually(equal(URLSessionTask.State.completed)) + expect(task?.state).toEventually(equal(URLSessionTask.State.completed)) } it("asks to retry for 429 response") { @@ -172,7 +172,7 @@ class HTTPClientTest: QuickSpec { done = true } expect(done).toEventually(beTrue()) - expect(task.state).toEventually(equal(URLSessionTask.State.completed)) + expect(task?.state).toEventually(equal(URLSessionTask.State.completed)) } it("asks to retry for 5xx response") { @@ -187,7 +187,19 @@ class HTTPClientTest: QuickSpec { done = true } expect(done).toEventually(beTrue()) - expect(task.state).toEventually(equal(URLSessionTask.State.completed)) + expect(task?.state).toEventually(equal(URLSessionTask.State.completed)) + } + + it("fails when batch size exceeds the max limit size") { + let oversizedBatch: [String: Any] = ["sentAt":"2016-07-19'T'19:25:06Z", + "batch": Array(repeating: ["type":"track", "event":"foo"], count: 16000)] + var done = false + let task = client.upload(oversizedBatch, forWriteKey: "bar") { retry in + expect(retry) == false + done = true + } + expect(done).toEventually(beTrue()) + expect(task).toEventually(beNil()) } }