RLMNetworkTransport.mm 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2020 Realm Inc.
  4. //
  5. // Licensed under the Apache License, Version 2.0 (the "License");
  6. // you may not use this file except in compliance with the License.
  7. // You may obtain a copy of the License at
  8. //
  9. // http://www.apache.org/licenses/LICENSE-2.0
  10. //
  11. // Unless required by applicable law or agreed to in writing, software
  12. // distributed under the License is distributed on an "AS IS" BASIS,
  13. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. // See the License for the specific language governing permissions and
  15. // limitations under the License.
  16. //
  17. ////////////////////////////////////////////////////////////////////////////
  18. #import "RLMNetworkTransport_Private.hpp"
  19. #import "RLMApp.h"
  20. #import "RLMRealmConfiguration.h"
  21. #import "RLMSyncUtil_Private.hpp"
  22. #import "RLMSyncManager_Private.hpp"
  23. #import "RLMUtil.hpp"
  24. #import <realm/object-store/sync/generic_network_transport.hpp>
  25. #import <realm/util/scope_exit.hpp>
  26. using namespace realm;
  27. static_assert((int)RLMHTTPMethodGET == (int)app::HttpMethod::get);
  28. static_assert((int)RLMHTTPMethodPOST == (int)app::HttpMethod::post);
  29. static_assert((int)RLMHTTPMethodPUT == (int)app::HttpMethod::put);
  30. static_assert((int)RLMHTTPMethodPATCH == (int)app::HttpMethod::patch);
  31. static_assert((int)RLMHTTPMethodDELETE == (int)app::HttpMethod::del);
  32. #pragma mark RLMSessionDelegate
  33. @interface RLMSessionDelegate <NSURLSessionDelegate> : NSObject
  34. + (instancetype)delegateWithCompletion:(RLMNetworkTransportCompletionBlock)completion;
  35. @end
  36. NSString * const RLMHTTPMethodToNSString[] = {
  37. [RLMHTTPMethodGET] = @"GET",
  38. [RLMHTTPMethodPOST] = @"POST",
  39. [RLMHTTPMethodPUT] = @"PUT",
  40. [RLMHTTPMethodPATCH] = @"PATCH",
  41. [RLMHTTPMethodDELETE] = @"DELETE"
  42. };
  43. @implementation RLMRequest
  44. @end
  45. @implementation RLMResponse
  46. @end
  47. @interface RLMEventSessionDelegate <NSURLSessionDelegate> : NSObject
  48. + (instancetype)delegateWithEventSubscriber:(RLMEventSubscriber *)subscriber;
  49. @end;
  50. @implementation RLMNetworkTransport
  51. - (void)sendRequestToServer:(RLMRequest *) request
  52. completion:(RLMNetworkTransportCompletionBlock)completionBlock; {
  53. // Create the request
  54. NSURL *requestURL = [[NSURL alloc] initWithString: request.url];
  55. NSMutableURLRequest *urlRequest = [NSMutableURLRequest requestWithURL:requestURL];
  56. urlRequest.HTTPMethod = RLMHTTPMethodToNSString[request.method];
  57. if (![urlRequest.HTTPMethod isEqualToString:@"GET"]) {
  58. urlRequest.HTTPBody = [request.body dataUsingEncoding:NSUTF8StringEncoding];
  59. }
  60. urlRequest.timeoutInterval = request.timeout;
  61. for (NSString *key in request.headers) {
  62. [urlRequest addValue:request.headers[key] forHTTPHeaderField:key];
  63. }
  64. id delegate = [RLMSessionDelegate delegateWithCompletion:completionBlock];
  65. auto session = [NSURLSession sessionWithConfiguration:NSURLSessionConfiguration.defaultSessionConfiguration
  66. delegate:delegate delegateQueue:nil];
  67. // Add the request to a task and start it
  68. [[session dataTaskWithRequest:urlRequest] resume];
  69. // Tell the session to destroy itself once it's done with the request
  70. [session finishTasksAndInvalidate];
  71. }
  72. - (NSURLSession *)doStreamRequest:(nonnull RLMRequest *)request
  73. eventSubscriber:(nonnull id<RLMEventDelegate>)subscriber {
  74. NSURLSessionConfiguration *sessionConfig = [NSURLSessionConfiguration defaultSessionConfiguration];
  75. sessionConfig.timeoutIntervalForRequest = 30;
  76. sessionConfig.timeoutIntervalForResource = INT_MAX;
  77. sessionConfig.HTTPAdditionalHeaders = @{
  78. @"Content-Type": @"text/event-stream",
  79. @"Cache": @"no-cache",
  80. @"Accept": @"text/event-stream"
  81. };
  82. id delegate = [RLMEventSessionDelegate delegateWithEventSubscriber:subscriber];
  83. auto session = [NSURLSession sessionWithConfiguration:sessionConfig
  84. delegate:delegate
  85. delegateQueue:nil];
  86. NSURL *url = [[NSURL alloc] initWithString:request.url];
  87. [[session dataTaskWithURL:url] resume];
  88. return session;
  89. }
  90. - (RLMRequest *)RLMRequestFromRequest:(const realm::app::Request)request {
  91. RLMRequest *rlmRequest = [RLMRequest new];
  92. NSMutableDictionary<NSString *, NSString*> *headersDict = [NSMutableDictionary new];
  93. for(auto &[key, value] : request.headers) {
  94. [headersDict setValue:@(value.c_str()) forKey:@(key.c_str())];
  95. }
  96. rlmRequest.headers = headersDict;
  97. rlmRequest.method = static_cast<RLMHTTPMethod>(request.method);
  98. rlmRequest.timeout = request.timeout_ms;
  99. rlmRequest.url = @(request.url.c_str());
  100. rlmRequest.body = @(request.body.c_str());
  101. return rlmRequest;
  102. }
  103. @end
  104. #pragma mark RLMSessionDelegate
  105. @implementation RLMSessionDelegate {
  106. NSData *_data;
  107. RLMNetworkTransportCompletionBlock _completionBlock;
  108. }
  109. + (instancetype)delegateWithCompletion:(RLMNetworkTransportCompletionBlock)completion {
  110. RLMSessionDelegate *delegate = [RLMSessionDelegate new];
  111. delegate->_completionBlock = completion;
  112. return delegate;
  113. }
  114. - (void)URLSession:(__unused NSURLSession *)session
  115. dataTask:(__unused NSURLSessionDataTask *)dataTask
  116. didReceiveData:(NSData *)data {
  117. if (!_data) {
  118. _data = data;
  119. return;
  120. }
  121. if (![_data respondsToSelector:@selector(appendData:)]) {
  122. _data = [_data mutableCopy];
  123. }
  124. [(id)_data appendData:data];
  125. }
  126. - (void)URLSession:(__unused NSURLSession *)session
  127. task:(NSURLSessionTask *)task
  128. didCompleteWithError:(NSError *)error
  129. {
  130. RLMResponse *response = [RLMResponse new];
  131. NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *) task.response;
  132. response.headers = httpResponse.allHeaderFields;
  133. response.httpStatusCode = httpResponse.statusCode;
  134. if (error) {
  135. response.body = error.localizedDescription;
  136. response.customStatusCode = error.code;
  137. return _completionBlock(response);
  138. }
  139. response.body = [[NSString alloc] initWithData:_data encoding:NSUTF8StringEncoding];
  140. _completionBlock(response);
  141. }
  142. @end
  143. @implementation RLMEventSessionDelegate {
  144. RLMEventSubscriber *_subscriber;
  145. bool _hasOpened;
  146. }
  147. + (instancetype)delegateWithEventSubscriber:(RLMEventSubscriber *)subscriber {
  148. RLMEventSessionDelegate *delegate = [RLMEventSessionDelegate new];
  149. delegate->_subscriber = subscriber;
  150. return delegate;
  151. }
  152. - (void)URLSession:(__unused NSURLSession *)session
  153. dataTask:(__unused NSURLSessionDataTask *)dataTask
  154. didReceiveData:(NSData *)data {
  155. if (!_hasOpened) {
  156. _hasOpened = true;
  157. [_subscriber didOpen];
  158. }
  159. NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *) dataTask.response;
  160. if (httpResponse.statusCode != 200) {
  161. NSString *errorStatus = [NSString stringWithFormat:@"URLSession HTTP error code: %ld",
  162. (long)httpResponse.statusCode];
  163. NSError *error = [NSError errorWithDomain:RLMErrorDomain
  164. code:0
  165. userInfo:@{NSLocalizedDescriptionKey: errorStatus}];
  166. return [_subscriber didCloseWithError:error];
  167. }
  168. [_subscriber didReceiveEvent:data];
  169. }
  170. - (void)URLSession:(__unused NSURLSession *)session
  171. task:(NSURLSessionTask *)task
  172. didCompleteWithError:(NSError *)error
  173. {
  174. RLMResponse *response = [RLMResponse new];
  175. NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *) task.response;
  176. response.headers = httpResponse.allHeaderFields;
  177. response.httpStatusCode = httpResponse.statusCode;
  178. // -999 indicates that the session was cancelled.
  179. if (error && (error.code != -999)) {
  180. response.body = [error localizedDescription];
  181. return [_subscriber didCloseWithError:error];
  182. } else if (error && (error.code == -999)) {
  183. return [_subscriber didCloseWithError:nil];
  184. }
  185. if (response.httpStatusCode != 200) {
  186. NSString *errorStatus = [NSString stringWithFormat:@"URLSession HTTP error code: %ld",
  187. (long)httpResponse.statusCode];
  188. NSError *error = [NSError errorWithDomain:RLMErrorDomain
  189. code:0
  190. userInfo:@{NSLocalizedDescriptionKey: errorStatus}];
  191. return [_subscriber didCloseWithError:error];
  192. }
  193. }
  194. @end