/****************************************************************************** * Copyright (c) 2008-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 "BlocklistDownloader.h" #import "BlocklistDownloaderViewController.h" #import "BlocklistScheduler.h" #import "Controller.h" @interface BlocklistDownloader (Private) - (void) startDownload; - (void) decompressBlocklist; @end @implementation BlocklistDownloader BlocklistDownloader * fBLDownloader = nil; + (BlocklistDownloader *) downloader { if (!fBLDownloader) { fBLDownloader = [[BlocklistDownloader alloc] init]; [fBLDownloader startDownload]; } return fBLDownloader; } + (BOOL) isRunning { return fBLDownloader != nil; } - (void) setViewController: (BlocklistDownloaderViewController *) viewController { fViewController = viewController; if (fViewController) { switch (fState) { case BLOCKLIST_DL_START: [fViewController setStatusStarting]; break; case BLOCKLIST_DL_DOWNLOADING: [fViewController setStatusProgressForCurrentSize: fCurrentSize expectedSize: fExpectedSize]; break; case BLOCKLIST_DL_PROCESSING: [fViewController setStatusProcessing]; break; } } } - (void) cancelDownload { [fViewController setFinished]; [fDownload cancel]; [BlocklistScheduler.scheduler updateSchedule]; fBLDownloader = nil; } //using the actual filename is the best bet - (void) download: (NSURLDownload *) download decideDestinationWithSuggestedFilename: (NSString *) filename { [fDownload setDestination: [NSTemporaryDirectory() stringByAppendingPathComponent: filename] allowOverwrite: NO]; } - (void) download: (NSURLDownload *) download didCreateDestination: (NSString *) path { fDestination = path; } - (void) download: (NSURLDownload *) download didReceiveResponse: (NSURLResponse *) response { fState = BLOCKLIST_DL_DOWNLOADING; fCurrentSize = 0; fExpectedSize = response.expectedContentLength; [fViewController setStatusProgressForCurrentSize: fCurrentSize expectedSize: fExpectedSize]; } - (void) download: (NSURLDownload *) download didReceiveDataOfLength: (NSUInteger) length { fCurrentSize += length; [fViewController setStatusProgressForCurrentSize: fCurrentSize expectedSize: fExpectedSize]; } - (void) download: (NSURLDownload *) download didFailWithError: (NSError *) error { [fViewController setFailed: error.localizedDescription]; [NSUserDefaults.standardUserDefaults setObject: [NSDate date] forKey: @"BlocklistNewLastUpdate"]; [BlocklistScheduler.scheduler updateSchedule]; fBLDownloader = nil; } - (void) downloadDidFinish: (NSURLDownload *) download { fState = BLOCKLIST_DL_PROCESSING; [fViewController setStatusProcessing]; NSAssert(fDestination != nil, @"the blocklist file destination has not been specified"); dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ [self decompressBlocklist]; dispatch_async(dispatch_get_main_queue(), ^{ const int count = tr_blocklistSetContent(((Controller *)NSApp.delegate).sessionHandle, fDestination.UTF8String); //delete downloaded file [NSFileManager.defaultManager removeItemAtPath: fDestination error: NULL]; if (count > 0) [fViewController setFinished]; else [fViewController setFailed: NSLocalizedString(@"The specified blocklist file did not contain any valid rules.", "blocklist fail message")]; //update last updated date for schedule NSDate * date = [NSDate date]; [NSUserDefaults.standardUserDefaults setObject: date forKey: @"BlocklistNewLastUpdate"]; [NSUserDefaults.standardUserDefaults setObject: date forKey: @"BlocklistNewLastUpdateSuccess"]; [BlocklistScheduler.scheduler updateSchedule]; [NSNotificationCenter.defaultCenter postNotificationName: @"BlocklistUpdated" object: nil]; fBLDownloader = nil; }); }); } - (BOOL) download: (NSURLDownload *) download shouldDecodeSourceDataOfMIMEType: (NSString *) encodingType { return YES; } @end @implementation BlocklistDownloader (Private) - (void) startDownload { fState = BLOCKLIST_DL_START; [BlocklistScheduler.scheduler cancelSchedule]; NSString * urlString = [NSUserDefaults.standardUserDefaults stringForKey: @"BlocklistURL"]; if (!urlString) urlString = @""; else if (![urlString isEqualToString: @""] && [urlString rangeOfString: @"://"].location == NSNotFound) urlString = [@"http://" stringByAppendingString: urlString]; NSURLRequest * request = [NSURLRequest requestWithURL: [NSURL URLWithString: urlString]]; fDownload = [[NSURLDownload alloc] initWithRequest: request delegate: self]; } //.gz, .tar.gz, .tgz, and .bgz will be decompressed by NSURLDownload for us. However, we have to do .zip files manually. - (void) decompressBlocklist { if ([fDestination.pathExtension.lowercaseString isEqualToString: @"zip"]) { BOOL success = NO; NSString * workingDirectory = fDestination.stringByDeletingLastPathComponent; //First, perform the actual unzipping NSTask * unzip = [[NSTask alloc] init]; unzip.launchPath = @"/usr/bin/unzip"; unzip.currentDirectoryPath = workingDirectory; unzip.arguments = @[ @"-o", /* overwrite */ @"-q", /* quiet! */ fDestination, /* source zip file */ @"-d", workingDirectory /*destination*/ ]; @try { [unzip launch]; [unzip waitUntilExit]; if (unzip.terminationStatus == 0) success = YES; } @catch(id exc) { success = NO; } if (success) { //Now find out what file we actually extracted; don't just assume it matches the zipfile's name NSTask *zipinfo; zipinfo = [[NSTask alloc] init]; zipinfo.launchPath = @"/usr/bin/zipinfo"; zipinfo.arguments = @[ @"-1", /* just the filename */ fDestination /* source zip file */ ]; zipinfo.standardOutput = [NSPipe pipe]; @try { NSFileHandle * zipinfoOutput = [zipinfo.standardOutput fileHandleForReading]; [zipinfo launch]; [zipinfo waitUntilExit]; NSString * actualFilename = [[NSString alloc] initWithData: [zipinfoOutput readDataToEndOfFile] encoding: NSUTF8StringEncoding]; actualFilename = [actualFilename stringByTrimmingCharactersInSet: NSCharacterSet.whitespaceAndNewlineCharacterSet]; NSString * newBlocklistPath = [workingDirectory stringByAppendingPathComponent: actualFilename]; //Finally, delete the ZIP file; we're done with it, and we'll return the unzipped blocklist [NSFileManager.defaultManager removeItemAtPath: fDestination error: NULL]; fDestination = newBlocklistPath; } @catch(id exc) {} } } } @end