/********************************************************************************** AudioPlayer.m Created by Thong Nguyen on 16/10/2012. https://github.com/tumtumtum/audjustable Copyright (c) 2012-2014 Thong Nguyen (tumtumtum@gmail.com). All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. All advertising materials mentioning features or use of this software must display the following acknowledgement: This product includes software developed by Thong Nguyen (tumtumtum@gmail.com) 4. Neither the name of Thong Nguyen nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY Thong Nguyen''AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. **********************************************************************************/ #import #import #import #import #import #import #import "mach/mach_time.h" #import #import #import "STKAutoRecoveringHTTPDataSource.h" #define DEFAULT_WATCHDOG_PERIOD_SECONDS (8) #define DEFAULT_INACTIVE_PERIOD_BEFORE_RECONNECT_SECONDS (15) static uint64_t GetTickCount(void) { static mach_timebase_info_data_t sTimebaseInfo; uint64_t machTime = mach_absolute_time(); if (sTimebaseInfo.denom == 0 ) { (void) mach_timebase_info(&sTimebaseInfo); } uint64_t millis = ((machTime / 1000000) * sTimebaseInfo.numer) / sTimebaseInfo.denom; return millis; } @interface STKAutoRecoveringHTTPDataSource() { int serial; int waitSeconds; NSTimer* timeoutTimer; BOOL waitingForNetwork; uint64_t ticksWhenLastDataReceived; SCNetworkReachabilityRef reachabilityRef; STKAutoRecoveringHTTPDataSourceOptions options; } -(void) reachabilityChanged; @end static void ReachabilityCallback(SCNetworkReachabilityRef target, SCNetworkReachabilityFlags flags, void* info) { @autoreleasepool { STKAutoRecoveringHTTPDataSource* dataSource = (__bridge STKAutoRecoveringHTTPDataSource*)info; [dataSource reachabilityChanged]; } } static void PopulateOptionsWithDefault(STKAutoRecoveringHTTPDataSourceOptions* options) { if (options->watchdogPeriodSeconds == 0) { options->watchdogPeriodSeconds = DEFAULT_WATCHDOG_PERIOD_SECONDS; } if (options->inactivePeriodBeforeReconnectSeconds == 0) { options->inactivePeriodBeforeReconnectSeconds = DEFAULT_INACTIVE_PERIOD_BEFORE_RECONNECT_SECONDS; } } @implementation STKAutoRecoveringHTTPDataSource -(STKHTTPDataSource*) innerHTTPDataSource { return (STKHTTPDataSource*)self.innerDataSource; } -(id) initWithDataSource:(STKDataSource *)innerDataSource { return [self initWithHTTPDataSource:(STKHTTPDataSource*)innerDataSource]; } -(id) initWithHTTPDataSource:(STKHTTPDataSource*)innerDataSourceIn { return [self initWithHTTPDataSource:innerDataSourceIn andOptions:(STKAutoRecoveringHTTPDataSourceOptions){}]; } -(id) initWithHTTPDataSource:(STKHTTPDataSource*)innerDataSourceIn andOptions:(STKAutoRecoveringHTTPDataSourceOptions)optionsIn { if (self = [super initWithDataSource:innerDataSourceIn]) { self.innerDataSource.delegate = self; struct sockaddr_in zeroAddress; bzero(&zeroAddress, sizeof(zeroAddress)); zeroAddress.sin_len = sizeof(zeroAddress); zeroAddress.sin_family = AF_INET; PopulateOptionsWithDefault(&optionsIn); self->options = optionsIn; reachabilityRef = SCNetworkReachabilityCreateWithAddress(kCFAllocatorDefault, (const struct sockaddr*)&zeroAddress); } return self; } -(BOOL) startNotifierOnRunLoop:(NSRunLoop*)runLoop { BOOL retVal = NO; SCNetworkReachabilityContext context = { 0, (__bridge void*)self, NULL, NULL, NULL }; if (SCNetworkReachabilitySetCallback(reachabilityRef, ReachabilityCallback, &context)) { if(SCNetworkReachabilityScheduleWithRunLoop(reachabilityRef, runLoop.getCFRunLoop, kCFRunLoopDefaultMode)) { retVal = YES; } } return retVal; } -(BOOL) registerForEvents:(NSRunLoop*)runLoop { [super registerForEvents:runLoop]; [self startNotifierOnRunLoop:runLoop]; if (timeoutTimer) { [timeoutTimer invalidate]; timeoutTimer = nil; } ticksWhenLastDataReceived = GetTickCount(); [self createTimeoutTimer]; return YES; } -(void) unregisterForEvents { [super unregisterForEvents]; [self stopNotifier]; [self destroyTimeoutTimer]; } -(void) timeoutTimerTick:(NSTimer*)timer { if (![self hasBytesAvailable]) { if ([self hasGotNetworkConnection]) { uint64_t currentTicks = GetTickCount(); if (((currentTicks - ticksWhenLastDataReceived) / 1000) >= options.inactivePeriodBeforeReconnectSeconds) { serial++; NSLog(@"timeoutTimerTick %lld/%lld", self.position, self.length); [self attemptReconnectWithSerial:@(serial)]; } } } } -(void) createTimeoutTimer { [self destroyTimeoutTimer]; NSRunLoop* runLoop = self.innerDataSource.eventsRunLoop; if (runLoop == nil) { return; } timeoutTimer = [NSTimer timerWithTimeInterval:options.watchdogPeriodSeconds target:self selector:@selector(timeoutTimerTick:) userInfo:@(serial) repeats:YES]; [runLoop addTimer:timeoutTimer forMode:NSRunLoopCommonModes]; } -(void) destroyTimeoutTimer { if (timeoutTimer) { [timeoutTimer invalidate]; timeoutTimer = nil; } } -(void) stopNotifier { if (reachabilityRef != NULL) { SCNetworkReachabilitySetCallback(reachabilityRef, NULL, NULL); SCNetworkReachabilityUnscheduleFromRunLoop(reachabilityRef, [self.innerDataSource.eventsRunLoop getCFRunLoop], kCFRunLoopDefaultMode); } } -(BOOL) hasGotNetworkConnection { SCNetworkReachabilityFlags flags; if (SCNetworkReachabilityGetFlags(reachabilityRef, &flags)) { return ((flags & kSCNetworkReachabilityFlagsReachable) != 0); } return NO; } -(void) seekToOffset:(int64_t)offset { ticksWhenLastDataReceived = GetTickCount(); [super seekToOffset:offset]; } -(void) close { [self destroyTimeoutTimer]; [super close]; } -(void) dealloc { NSLog(@"STKAutoRecoveringHTTPDataSource dealloc"); self.innerDataSource.delegate = nil; [self stopNotifier]; [self destroyTimeoutTimer]; [NSObject cancelPreviousPerformRequestsWithTarget:self]; if (reachabilityRef!= NULL) { CFRelease(reachabilityRef); } } -(void) reachabilityChanged { if (waitingForNetwork) { waitingForNetwork = NO; NSLog(@"reachabilityChanged %lld/%lld", self.position, self.length); serial++; [self attemptReconnectWithSerial:@(serial)]; } } -(void) dataSourceDataAvailable:(STKDataSource*)dataSource { if (![self.innerDataSource hasBytesAvailable]) { return; } serial++; waitSeconds = 1; ticksWhenLastDataReceived = GetTickCount(); [super dataSourceDataAvailable:dataSource]; } -(void) attemptReconnectWithSerial:(NSNumber*)serialIn { if (serialIn.intValue != self->serial) { return; } NSLog(@"attemptReconnect %lld/%lld", self.position, self.length); if (self.innerDataSource.eventsRunLoop) { [self.innerDataSource reconnect]; } } -(void) attemptReconnectWithTimer:(NSTimer*)timer { [self attemptReconnectWithSerial:(NSNumber*)timer.userInfo]; } -(void) processRetryOnError { if (![self hasGotNetworkConnection]) { waitingForNetwork = YES; return; } waitingForNetwork = NO; NSRunLoop* runLoop = self.innerDataSource.eventsRunLoop; if (runLoop == nil) { // DataSource no longer used return; } else { serial++; NSTimer* timer = [NSTimer timerWithTimeInterval:waitSeconds target:self selector:@selector(attemptReconnectWithTimer:) userInfo:@(serial) repeats:NO]; [runLoop addTimer:timer forMode:NSRunLoopCommonModes]; } waitSeconds = MIN(waitSeconds + 1, 5); } -(void) dataSourceEof:(STKDataSource*)dataSource { NSLog(@"dataSourceEof"); if ([self position] < [self length]) { [self processRetryOnError]; return; } [self.delegate dataSourceEof:self]; } -(void) dataSourceErrorOccured:(STKDataSource*)dataSource { NSLog(@"dataSourceErrorOccured"); if (self.innerDataSource.httpStatusCode == 416 /* Range out of bounds */) { [super dataSourceEof:dataSource]; } else { [self processRetryOnError]; } } -(NSString*) description { return [NSString stringWithFormat:@"HTTP data source with file length: %lld and position: %lld", self.length, self.position]; } @end