blob: a4d919a64290b0861da4eb51b2868dec9ac1f388 [file] [log] [blame]
/*
* Copyright (C) 2005 Apple Computer, Inc. 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. Neither the name of Apple Computer, Inc. ("Apple") 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 APPLE AND ITS CONTRIBUTORS "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 APPLE OR ITS CONTRIBUTORS 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 <WebKit/WebDownload.h>
#import <Foundation/NSURLAuthenticationChallenge.h>
#import <Foundation/NSURLDownload.h>
#import <WebCore/AuthenticationMac.h>
#import <WebCore/Credential.h>
#import <WebCore/CredentialStorage.h>
#import <WebCore/ProtectionSpace.h>
#import <WebKit/WebPanelAuthenticationHandler.h>
#import <wtf/Assertions.h>
#import "WebTypesInternal.h"
using namespace WebCore;
@class NSURLConnectionDelegateProxy;
// FIXME: The following are NSURLDownload SPI - it would be nice to not have to override them at
// some point in the future
@interface NSURLDownload (WebDownloadCapability)
- (id)_initWithLoadingConnection:(NSURLConnection *)connection
request:(NSURLRequest *)request
response:(NSURLResponse *)response
delegate:(id)delegate
proxy:(NSURLConnectionDelegateProxy *)proxy;
- (id)_initWithRequest:(NSURLRequest *)request
delegate:(id)delegate
directory:(NSString *)directory;
@end
@interface WebDownloadInternal : NSObject <NSURLDownloadDelegate>
{
@public
id realDelegate;
}
- (void)setRealDelegate:(id)rd;
@end
@implementation WebDownloadInternal
- (void)dealloc
{
[realDelegate release];
[super dealloc];
}
- (void)setRealDelegate:(id)rd
{
[rd retain];
[realDelegate release];
realDelegate = rd;
}
- (BOOL)respondsToSelector:(SEL)selector
{
if (selector == @selector(downloadDidBegin:) ||
selector == @selector(download:willSendRequest:redirectResponse:) ||
selector == @selector(download:didReceiveResponse:) ||
selector == @selector(download:didReceiveDataOfLength:) ||
selector == @selector(download:shouldDecodeSourceDataOfMIMEType:) ||
selector == @selector(download:decideDestinationWithSuggestedFilename:) ||
selector == @selector(download:didCreateDestination:) ||
selector == @selector(downloadDidFinish:) ||
selector == @selector(download:didFailWithError:) ||
selector == @selector(download:shouldBeginChildDownloadOfSource:delegate:) ||
selector == @selector(download:didBeginChildDownload:)) {
return [realDelegate respondsToSelector:selector];
}
return [super respondsToSelector:selector];
}
- (void)downloadDidBegin:(NSURLDownload *)download
{
[realDelegate downloadDidBegin:download];
}
- (NSURLRequest *)download:(NSURLDownload *)download willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse
{
return [realDelegate download:download willSendRequest:request redirectResponse:redirectResponse];
}
- (void)download:(NSURLDownload *)download didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge
{
// Try previously stored credential first.
if (![challenge previousFailureCount]) {
NSURLCredential *credential = mac(CredentialStorage::get(core([challenge protectionSpace])));
if (credential) {
[[challenge sender] useCredential:credential forAuthenticationChallenge:challenge];
return;
}
}
if ([realDelegate respondsToSelector:@selector(download:didReceiveAuthenticationChallenge:)]) {
[realDelegate download:download didReceiveAuthenticationChallenge:challenge];
} else {
NSWindow *window = nil;
if ([realDelegate respondsToSelector:@selector(downloadWindowForAuthenticationSheet:)]) {
window = [realDelegate downloadWindowForAuthenticationSheet:(WebDownload *)download];
}
[[WebPanelAuthenticationHandler sharedHandler] startAuthentication:challenge window:window];
}
}
- (void)download:(NSURLDownload *)download didCancelAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge
{
if ([realDelegate respondsToSelector:@selector(download:didCancelAuthenticationChallenge:)]) {
[realDelegate download:download didCancelAuthenticationChallenge:challenge];
} else {
[[WebPanelAuthenticationHandler sharedHandler] cancelAuthentication:challenge];
}
}
- (void)download:(NSURLDownload *)download didReceiveResponse:(NSURLResponse *)response
{
[realDelegate download:download didReceiveResponse:response];
}
- (void)download:(NSURLDownload *)download didReceiveDataOfLength:(NSUInteger)length
{
[realDelegate download:download didReceiveDataOfLength:length];
}
- (BOOL)download:(NSURLDownload *)download shouldDecodeSourceDataOfMIMEType:(NSString *)encodingType
{
return [realDelegate download:download shouldDecodeSourceDataOfMIMEType:encodingType];
}
- (void)download:(NSURLDownload *)download decideDestinationWithSuggestedFilename:(NSString *)filename
{
[realDelegate download:download decideDestinationWithSuggestedFilename:filename];
}
- (void)download:(NSURLDownload *)download didCreateDestination:(NSString *)path
{
[realDelegate download:download didCreateDestination:path];
}
- (void)downloadDidFinish:(NSURLDownload *)download
{
[realDelegate downloadDidFinish:download];
}
- (void)download:(NSURLDownload *)download didFailWithError:(NSError *)error
{
[realDelegate download:download didFailWithError:error];
}
- (NSURLRequest *)download:(NSURLDownload *)download shouldBeginChildDownloadOfSource:(NSURLRequest *)child delegate:(id *)childDelegate
{
return [realDelegate download:download shouldBeginChildDownloadOfSource:child delegate:childDelegate];
}
- (void)download:(NSURLDownload *)parent didBeginChildDownload:(NSURLDownload *)child
{
[realDelegate download:parent didBeginChildDownload:child];
}
@end
@implementation WebDownload
- (void)_setRealDelegate:(id)delegate
{
if (_webInternal == nil) {
_webInternal = [[WebDownloadInternal alloc] init];
[_webInternal setRealDelegate:delegate];
} else {
ASSERT(_webInternal == delegate);
}
}
- (id)init
{
self = [super init];
if (self != nil) {
// _webInternal can be set up before init by _setRealDelegate
if (_webInternal == nil) {
_webInternal = [[WebDownloadInternal alloc] init];
}
}
return self;
}
- (void)dealloc
{
[_webInternal release];
[super dealloc];
}
- (id)initWithRequest:(NSURLRequest *)request delegate:(id<NSURLDownloadDelegate>)delegate
{
[self _setRealDelegate:delegate];
return [super initWithRequest:request delegate:_webInternal];
}
- (id)_initWithLoadingConnection:(NSURLConnection *)connection
request:(NSURLRequest *)request
response:(NSURLResponse *)response
delegate:(id)delegate
proxy:(NSURLConnectionDelegateProxy *)proxy
{
[self _setRealDelegate:delegate];
return [super _initWithLoadingConnection:connection request:request response:response delegate:_webInternal proxy:proxy];
}
- (id)_initWithRequest:(NSURLRequest *)request
delegate:(id)delegate
directory:(NSString *)directory
{
[self _setRealDelegate:delegate];
return [super _initWithRequest:request delegate:_webInternal directory:directory];
}
- (void)connection:(NSURLConnection *)connection willStopBufferingData:(NSData *)data
{
// NSURLConnection calls this method even if it is not implemented.
// This happens because NSURLConnection caches the results of respondsToSelector.
// Those results become invalid when the delegate of NSURLConnectionDelegateProxy is changed.
// This is a workaround since this problem needs to be fixed in NSURLConnectionDelegateProxy.
// <rdar://problem/3913270> NSURLConnection calls unimplemented delegate method in WebDownload
}
@end