mirror of
https://github.com/transmission/transmission
synced 2025-01-04 05:56:02 +00:00
266 lines
8.9 KiB
Objective-C
266 lines
8.9 KiB
Objective-C
/******************************************************************************
|
|
* $Id$
|
|
*
|
|
* 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) dealloc
|
|
{
|
|
[fDownload release];
|
|
[fDestination release];
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void) cancelDownload
|
|
{
|
|
[fViewController setFinished];
|
|
|
|
[fDownload cancel];
|
|
|
|
[[BlocklistScheduler scheduler] updateSchedule];
|
|
|
|
fBLDownloader = nil;
|
|
[self release];
|
|
}
|
|
|
|
//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 release];
|
|
fDestination = [path retain];
|
|
}
|
|
|
|
- (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;
|
|
[self release];
|
|
}
|
|
|
|
- (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;
|
|
[self release];
|
|
});
|
|
});
|
|
}
|
|
|
|
- (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 setLaunchPath: @"/usr/bin/unzip"];
|
|
[unzip setCurrentDirectoryPath: workingDirectory];
|
|
[unzip setArguments: [NSArray arrayWithObjects:
|
|
@"-o", /* overwrite */
|
|
@"-q", /* quiet! */
|
|
fDestination, /* source zip file */
|
|
@"-d", workingDirectory, /*destination*/
|
|
nil]];
|
|
|
|
@try
|
|
{
|
|
[unzip launch];
|
|
[unzip waitUntilExit];
|
|
|
|
if ([unzip terminationStatus] == 0)
|
|
success = YES;
|
|
}
|
|
@catch(id exc)
|
|
{
|
|
success = NO;
|
|
}
|
|
[unzip release];
|
|
|
|
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 setLaunchPath: @"/usr/bin/zipinfo"];
|
|
[zipinfo setArguments: [NSArray arrayWithObjects:
|
|
@"-1", /* just the filename */
|
|
fDestination, /* source zip file */
|
|
nil]];
|
|
[zipinfo setStandardOutput: [NSPipe pipe]];
|
|
|
|
@try
|
|
{
|
|
NSFileHandle * zipinfoOutput = [[zipinfo standardOutput] fileHandleForReading];
|
|
|
|
[zipinfo launch];
|
|
[zipinfo waitUntilExit];
|
|
|
|
NSString * actualFilename = [[[NSString alloc] initWithData: [zipinfoOutput readDataToEndOfFile]
|
|
encoding: NSUTF8StringEncoding] autorelease];
|
|
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 release];
|
|
fDestination = [newBlocklistPath retain];
|
|
}
|
|
@catch(id exc) {}
|
|
[zipinfo release];
|
|
}
|
|
}
|
|
}
|
|
|
|
@end
|