RLMAnalytics.mm 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2015 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. // Asynchronously submits build information to Realm if running in an iOS
  19. // simulator or on OS X if a debugger is attached. Does nothing if running on an
  20. // iOS / watchOS device or if a debugger is *not* attached.
  21. //
  22. // To be clear: this does *not* run when your app is in production or on
  23. // your end-user’s devices; it will only run in the simulator or when a debugger
  24. // is attached.
  25. //
  26. // Why are we doing this? In short, because it helps us build a better product
  27. // for you. None of the data personally identifies you, your employer or your
  28. // app, but it *will* help us understand what language you use, what iOS
  29. // versions you target, etc. Having this info will help prioritizing our time,
  30. // adding new features and deprecating old features. Collecting an anonymized
  31. // bundle & anonymized MAC is the only way for us to count actual usage of the
  32. // other metrics accurately. If we don’t have a way to deduplicate the info
  33. // reported, it will be useless, as a single developer building their Swift app
  34. // 10 times would report 10 times more than a single Objective-C developer that
  35. // only builds once, making the data all but useless.
  36. // No one likes sharing data unless it’s necessary, we get it, and we’ve
  37. // debated adding this for a long long time. Since Realm is a free product
  38. // without an email signup, we feel this is a necessary step so we can collect
  39. // relevant data to build a better product for you. If you truly, absolutely
  40. // feel compelled to not send this data back to Realm, then you can set an env
  41. // variable named REALM_DISABLE_ANALYTICS. Since Realm is free we believe
  42. // letting these analytics run is a small price to pay for the product & support
  43. // we give you.
  44. //
  45. // Currently the following information is reported:
  46. // - What version of Realm is being used, and from which language (obj-c or Swift).
  47. // - What version of OS X it's running on (in case Xcode aggressively drops
  48. // support for older versions again, we need to know what we need to support).
  49. // - The minimum iOS/OS X version that the application is targeting (again, to
  50. // help us decide what versions we need to support).
  51. // - An anonymous MAC address and bundle ID to aggregate the other information on.
  52. // - What version of Swift is being used (if applicable).
  53. #import "RLMAnalytics.hpp"
  54. #import <Foundation/Foundation.h>
  55. #if TARGET_IPHONE_SIMULATOR || TARGET_OS_MAC || (TARGET_OS_WATCH && TARGET_OS_SIMULATOR) || (TARGET_OS_TV && TARGET_OS_SIMULATOR)
  56. #import "RLMRealm.h"
  57. #import "RLMUtil.hpp"
  58. #import <array>
  59. #import <sys/socket.h>
  60. #import <sys/sysctl.h>
  61. #import <net/if.h>
  62. #import <net/if_dl.h>
  63. #import <CommonCrypto/CommonDigest.h>
  64. #ifndef REALM_COCOA_VERSION
  65. #import "RLMVersion.h"
  66. #endif
  67. #if REALM_ENABLE_SYNC
  68. #import <realm/sync/version.hpp>
  69. #endif
  70. // Declared for RealmSwiftObjectUtil
  71. @interface NSObject (SwiftVersion)
  72. + (NSString *)swiftVersion;
  73. @end
  74. // Wrapper for sysctl() that handles the memory management stuff
  75. static auto RLMSysCtl(int *mib, u_int mibSize, size_t *bufferSize) {
  76. std::unique_ptr<void, decltype(&free)> buffer(nullptr, &free);
  77. int ret = sysctl(mib, mibSize, nullptr, bufferSize, nullptr, 0);
  78. if (ret != 0) {
  79. return buffer;
  80. }
  81. buffer.reset(malloc(*bufferSize));
  82. if (!buffer) {
  83. return buffer;
  84. }
  85. ret = sysctl(mib, mibSize, buffer.get(), bufferSize, nullptr, 0);
  86. if (ret != 0) {
  87. buffer.reset();
  88. }
  89. return buffer;
  90. }
  91. // Get the version of OS X we're running on (even in the simulator this gives
  92. // the OS X version and not the simulated iOS version)
  93. static NSString *RLMOSVersion() {
  94. std::array<int, 2> mib = {{CTL_KERN, KERN_OSRELEASE}};
  95. size_t bufferSize;
  96. auto buffer = RLMSysCtl(&mib[0], mib.size(), &bufferSize);
  97. if (!buffer) {
  98. return nil;
  99. }
  100. return [[NSString alloc] initWithBytesNoCopy:buffer.release()
  101. length:bufferSize - 1
  102. encoding:NSUTF8StringEncoding
  103. freeWhenDone:YES];
  104. }
  105. // Hash the data in the given buffer and convert it to a hex-format string
  106. static NSString *RLMHashData(const void *bytes, size_t length) {
  107. unsigned char buffer[CC_SHA256_DIGEST_LENGTH];
  108. CC_SHA256(bytes, static_cast<CC_LONG>(length), buffer);
  109. char formatted[CC_SHA256_DIGEST_LENGTH * 2 + 1];
  110. for (int i = 0; i < CC_SHA256_DIGEST_LENGTH; ++i) {
  111. sprintf(formatted + i * 2, "%02x", buffer[i]);
  112. }
  113. return [[NSString alloc] initWithBytes:formatted
  114. length:CC_SHA256_DIGEST_LENGTH * 2
  115. encoding:NSUTF8StringEncoding];
  116. }
  117. // Returns the hash of the MAC address of the first network adaptor since the
  118. // vendorIdentifier isn't constant between iOS simulators.
  119. static NSString *RLMMACAddress() {
  120. int en0 = static_cast<int>(if_nametoindex("en0"));
  121. if (!en0) {
  122. return nil;
  123. }
  124. std::array<int, 6> mib = {{CTL_NET, PF_ROUTE, 0, AF_LINK, NET_RT_IFLIST, en0}};
  125. size_t bufferSize;
  126. auto buffer = RLMSysCtl(&mib[0], mib.size(), &bufferSize);
  127. if (!buffer) {
  128. return nil;
  129. }
  130. // sockaddr_dl struct is immediately after the if_msghdr struct in the buffer
  131. auto sockaddr = reinterpret_cast<sockaddr_dl *>(static_cast<if_msghdr *>(buffer.get()) + 1);
  132. auto mac = reinterpret_cast<const unsigned char *>(sockaddr->sdl_data + sockaddr->sdl_nlen);
  133. return RLMHashData(mac, 6);
  134. }
  135. static NSDictionary *RLMAnalyticsPayload() {
  136. NSBundle *appBundle = NSBundle.mainBundle;
  137. NSString *hashedBundleID = appBundle.bundleIdentifier;
  138. // Main bundle isn't always the one of interest (e.g. when running tests
  139. // it's xctest rather than the app's bundle), so look for one with a bundle ID
  140. if (!hashedBundleID) {
  141. for (NSBundle *bundle in NSBundle.allBundles) {
  142. if ((hashedBundleID = bundle.bundleIdentifier)) {
  143. appBundle = bundle;
  144. break;
  145. }
  146. }
  147. }
  148. // If we found a bundle ID anywhere, hash it as it could contain sensitive
  149. // information (e.g. the name of an unnanounced product)
  150. if (hashedBundleID) {
  151. NSData *data = [hashedBundleID dataUsingEncoding:NSUTF8StringEncoding];
  152. hashedBundleID = RLMHashData(data.bytes, data.length);
  153. }
  154. NSString *osVersionString = [[NSProcessInfo processInfo] operatingSystemVersionString];
  155. Class swiftObjectUtilClass = NSClassFromString(@"RealmSwiftObjectUtil");
  156. BOOL isSwift = swiftObjectUtilClass != nil;
  157. NSString *swiftVersion = isSwift ? [swiftObjectUtilClass swiftVersion] : @"N/A";
  158. static NSString *kUnknownString = @"unknown";
  159. NSString *hashedMACAddress = RLMMACAddress() ?: kUnknownString;
  160. return @{
  161. @"event": @"Run",
  162. @"properties": @{
  163. // MixPanel properties
  164. @"token": @"ce0fac19508f6c8f20066d345d360fd0",
  165. // Anonymous identifiers to deduplicate events
  166. @"distinct_id": hashedMACAddress,
  167. @"Anonymized MAC Address": hashedMACAddress,
  168. @"Anonymized Bundle ID": hashedBundleID ?: kUnknownString,
  169. // Which version of Realm is being used
  170. @"Binding": @"cocoa",
  171. @"Language": isSwift ? @"swift" : @"objc",
  172. @"Realm Version": REALM_COCOA_VERSION,
  173. #if REALM_ENABLE_SYNC
  174. @"Sync Version": @(REALM_SYNC_VER_STRING),
  175. #endif
  176. #if TARGET_OS_WATCH
  177. @"Target OS Type": @"watchos",
  178. #elif TARGET_OS_TV
  179. @"Target OS Type": @"tvos",
  180. #elif TARGET_OS_IPHONE
  181. @"Target OS Type": @"ios",
  182. #else
  183. @"Target OS Type": @"osx",
  184. #endif
  185. @"Swift Version": swiftVersion,
  186. // Current OS version the app is targetting
  187. @"Target OS Version": osVersionString,
  188. // Minimum OS version the app is targetting
  189. @"Target OS Minimum Version": appBundle.infoDictionary[@"MinimumOSVersion"] ?: kUnknownString,
  190. // Host OS version being built on
  191. @"Host OS Type": @"osx",
  192. @"Host OS Version": RLMOSVersion() ?: kUnknownString,
  193. }
  194. };
  195. }
  196. void RLMSendAnalytics() {
  197. if (getenv("REALM_DISABLE_ANALYTICS") || !RLMIsDebuggerAttached() || RLMIsRunningInPlayground()) {
  198. return;
  199. }
  200. NSData *payload = [NSJSONSerialization dataWithJSONObject:RLMAnalyticsPayload() options:0 error:nil];
  201. NSString *url = [NSString stringWithFormat:@"https://api.mixpanel.com/track/?data=%@&ip=1", [payload base64EncodedStringWithOptions:0]];
  202. // No error handling or anything because logging errors annoyed people for no
  203. // real benefit, and it's not clear what else we could do
  204. [[NSURLSession.sharedSession dataTaskWithURL:[NSURL URLWithString:url]] resume];
  205. }
  206. #else
  207. void RLMSendAnalytics() {}
  208. #endif