/****************************************************************************** * Copyright (c) 2006-2012 Transmission authors and contributors * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. *****************************************************************************/ #import "PortChecker.h" #define CHECKER_URL(port) [NSString stringWithFormat:@"https://portcheck.transmissionbt.com/%ld", port] #define CHECK_FIRE 3.0 @interface PortChecker (Private) - (void)startProbe:(NSTimer*)timer; - (void)callBackWithStatus:(port_status_t)status; @end @implementation PortChecker - (instancetype)initForPort:(NSInteger)portNumber delay:(BOOL)delay withDelegate:(id)delegate { if ((self = [super init])) { fDelegate = delegate; fStatus = PORT_STATUS_CHECKING; fTimer = [NSTimer scheduledTimerWithTimeInterval:CHECK_FIRE target:self selector:@selector(startProbe:) userInfo:@(portNumber) repeats:NO]; if (!delay) { [fTimer fire]; } } return self; } - (void)dealloc { [fTimer invalidate]; } - (port_status_t)status { return fStatus; } - (void)cancelProbe { [fTimer invalidate]; fTimer = nil; [fConnection cancel]; } - (void)connection:(NSURLConnection*)connection didReceiveResponse:(NSURLResponse*)response { fPortProbeData.length = 0; } - (void)connection:(NSURLConnection*)connection didReceiveData:(NSData*)data { [fPortProbeData appendData:data]; } - (void)connection:(NSURLConnection*)connection didFailWithError:(NSError*)error { NSLog(@"Unable to get port status: connection failed (%@)", error.localizedDescription); [self callBackWithStatus:PORT_STATUS_ERROR]; } - (void)connectionDidFinishLoading:(NSURLConnection*)connection { NSString* probeString = [[NSString alloc] initWithData:fPortProbeData encoding:NSUTF8StringEncoding]; fPortProbeData = nil; if (probeString) { if ([probeString isEqualToString:@"1"]) { [self callBackWithStatus:PORT_STATUS_OPEN]; } else if ([probeString isEqualToString:@"0"]) { [self callBackWithStatus:PORT_STATUS_CLOSED]; } else { NSLog(@"Unable to get port status: invalid response (%@)", probeString); [self callBackWithStatus:PORT_STATUS_ERROR]; } } else { NSLog(@"Unable to get port status: invalid data received"); [self callBackWithStatus:PORT_STATUS_ERROR]; } } @end @implementation PortChecker (Private) - (void)startProbe:(NSTimer*)timer { fTimer = nil; NSURLRequest* portProbeRequest = [NSURLRequest requestWithURL:[NSURL URLWithString:CHECKER_URL([[timer userInfo] integerValue])] cachePolicy:NSURLRequestReloadIgnoringLocalAndRemoteCacheData timeoutInterval:15.0]; if ((fConnection = [[NSURLConnection alloc] initWithRequest:portProbeRequest delegate:self])) { fPortProbeData = [[NSMutableData alloc] init]; } else { NSLog(@"Unable to get port status: failed to initiate connection"); [self callBackWithStatus:PORT_STATUS_ERROR]; } } - (void)callBackWithStatus:(port_status_t)status { fStatus = status; if (fDelegate && [fDelegate respondsToSelector:@selector(portCheckerDidFinishProbing:)]) { [fDelegate performSelectorOnMainThread:@selector(portCheckerDidFinishProbing:) withObject:self waitUntilDone:NO]; } } @end