Swiftgram/Classes/BITTelemetryManager.m
2015-09-09 18:47:13 -07:00

186 lines
6.2 KiB
Objective-C

#import "HockeySDK.h"
#if HOCKEYSDK_FEATURE_TELEMETRY
#import "BITTelemetryManager.h"
#import "BITTelemetryContext.h"
#import "BITTelemetryManagerPrivate.h"
#import "BITHockeyHelper.h"
#import "HockeySDKPrivate.h"
#import "BITChannel.h"
#import "BITSession.h"
#import "BITSessionState.h"
#import "BITSessionStateData.h"
#import "BITPersistence.h"
#import "BITHockeyBaseManagerPrivate.h"
#import "BITSender.h"
static char *const kBITTelemetryEventQueue =
"net.hockeyapp.telemetryEventQueue";
NSString *const kBITSessionFileType = @"plist";
NSString *const kBITApplicationDidEnterBackgroundTime = @"BITApplicationDidEnterBackgroundTime";
NSString *const kBITApplicationWasLaunched = @"BITApplicationWasLaunched";
NSString *const BITTelemetryEndpoint = @"https://dc-int.services.visualstudio.com/v2/track";
@implementation BITTelemetryManager {
id _appWillEnterForegroundObserver;
id _appDidEnterBackgroundObserver;
}
@synthesize channel = _channel;
@synthesize telemetryContext = _telemetryContext;
@synthesize persistence = _persistence;
@synthesize userDefaults = _userDefaults;
#pragma mark - Create & start instance
- (instancetype)init {
if((self = [super init])) {
_telemetryEventQueue = dispatch_queue_create(kBITTelemetryEventQueue, DISPATCH_QUEUE_CONCURRENT);
_appBackgroundTimeBeforeSessionExpires = 20;
}
self.serverURL = nil;
return self;
}
- (instancetype)initWithChannel:(BITChannel *)channel telemetryContext:(BITTelemetryContext *)telemetryContext persistence:(BITPersistence *)persistence userDefaults:(NSUserDefaults *)userDefaults {
if((self = [self init])) {
_channel = channel;
_telemetryContext = telemetryContext;
_persistence = persistence;
_userDefaults = userDefaults;
}
return self;
}
- (void)startManager {
if(!self.serverURL){
self.serverURL = BITTelemetryEndpoint;
}
_sender = [[BITSender alloc]initWithPersistence:self.persistence serverURL:[NSURL URLWithString:self.serverURL]];
[self startNewSessionWithId:bit_UUID()];
[self registerObservers];
}
#pragma mark - Sessions
- (void)registerObservers {
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
__weak typeof(self) weakSelf = self;
if(nil == _appDidEnterBackgroundObserver) {
_appDidEnterBackgroundObserver = [nc addObserverForName:UIApplicationDidEnterBackgroundNotification
object:nil
queue:NSOperationQueue.mainQueue
usingBlock:^(NSNotification *note) {
typeof(self) strongSelf = weakSelf;
[strongSelf updateDidEnterBackgroundTime];
}];
}
if(nil == _appWillEnterForegroundObserver) {
_appWillEnterForegroundObserver = [nc addObserverForName:UIApplicationWillEnterForegroundNotification
object:nil
queue:NSOperationQueue.mainQueue
usingBlock:^(NSNotification *note) {
typeof(self) strongSelf = weakSelf;
[strongSelf startNewSessionIfNeeded];
}];
}
}
- (void)unregisterObservers {
[[NSNotificationCenter defaultCenter] removeObserver:self];
_appDidEnterBackgroundObserver = nil;
_appWillEnterForegroundObserver = nil;
}
- (void)updateDidEnterBackgroundTime {
[self.userDefaults setDouble:[[NSDate date] timeIntervalSince1970] forKey:kBITApplicationDidEnterBackgroundTime];
[self.userDefaults synchronize];
}
- (void)startNewSessionIfNeeded {
if(self.appBackgroundTimeBeforeSessionExpires == 0) {
__weak typeof(self) weakSelf = self;
dispatch_async(_telemetryEventQueue, ^{
typeof(self) strongSelf = weakSelf;
[strongSelf startNewSessionWithId:bit_UUID()];
});
}
double appDidEnterBackgroundTime = [self.userDefaults doubleForKey:kBITApplicationDidEnterBackgroundTime];
double timeSinceLastBackground = [[NSDate date] timeIntervalSince1970] - appDidEnterBackgroundTime;
if(timeSinceLastBackground > self.appBackgroundTimeBeforeSessionExpires) {
[self startNewSessionWithId:bit_UUID()];
}
}
- (void)startNewSessionWithId:(NSString *)sessionId {
BITSession *newSession = [self createNewSessionWithId:sessionId];
[self.telemetryContext setSessionId:newSession.sessionId];
[self.telemetryContext setIsFirstSession:newSession.isFirst];
[self.telemetryContext setIsNewSession:newSession.isNew];
[self trackSessionWithState:BITSessionState_start];
}
// iOS 8 Sim Bug: iOS Simulator -> Reset Content and Settings
- (BITSession *)createNewSessionWithId:(NSString *)sessionId {
BITSession *session = [BITSession new];
session.sessionId = sessionId;
session.isNew = @"true";
if([self.userDefaults boolForKey:kBITApplicationWasLaunched] == NO) {
session.isFirst = @"true";
[self.userDefaults setBool:YES forKey:kBITApplicationWasLaunched];
[self.userDefaults synchronize];
} else {
session.isFirst = @"false";
}
return session;
}
#pragma mark - Track telemetry
- (void)trackSessionWithState:(BITSessionState) state {
BITSessionStateData *sessionStateData = [BITSessionStateData new];
sessionStateData.state = state;
[self.channel enqueueTelemetryItem:sessionStateData];
}
#pragma mark - Custom getter
- (BITChannel *)channel {
if(!_channel){
_channel = [[BITChannel alloc]initWithTelemetryContext:self.telemetryContext persistence:self.persistence];
}
return _channel;
}
- (BITTelemetryContext *)telemetryContext {
if(!_telemetryContext){
_telemetryContext = [[BITTelemetryContext alloc] initWithAppIdentifier:self.appIdentifier persistence:self.persistence];
}
return _telemetryContext;
}
- (BITPersistence *)persistence {
if(!_persistence){
_persistence = [BITPersistence new];
}
return _persistence;
}
- (NSUserDefaults *)userDefaults {
if(!_userDefaults){
_userDefaults = [NSUserDefaults standardUserDefaults];
}
return _userDefaults;
}
@end
#endif /* HOCKEYSDK_FEATURE_TELEMETRY */