Remove Cocoa CFURLConnection loading code
[WebKit-https.git] / Source / WebCore / platform / network / mac / ResourceErrorMac.mm
index 275ca41..709b699 100644 (file)
  *    notice, this list of conditions and the following disclaimer in the
  *    documentation and/or other materials provided with the distribution.
  *
- * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 COMPUTER, INC. OR
+ * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
  * 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
 #import "config.h"
 #import "ResourceError.h"
 
-#import "BlockExceptions.h"
-#import "KURL.h"
+#import "URL.h"
+#import <CoreFoundation/CFError.h>
 #import <Foundation/Foundation.h>
+#import <wtf/BlockObjCExceptions.h>
 
 @interface NSError (WebExtras)
 - (NSString *)_web_localizedDescription;
 @end
 
+#if PLATFORM(IOS)
+
+// This workaround code exists here because we can't call translateToCFError in Foundation. Once we
+// have that, we can remove this code. <rdar://problem/9837415> Need SPI for translateCFError
+// The code is mostly identical to Foundation - I changed the class name and fixed minor compile errors.
+// We need this because client code (Safari) wants an NSError with NSURLErrorDomain as its domain.
+// The Foundation code below does that and sets up appropriate certificate keys in the NSError.
+
+@interface WebCustomNSURLError : NSError
+
+@end
+
+@implementation WebCustomNSURLError
+
+static NSDictionary* dictionaryThatCanCode(NSDictionary* src)
+{
+    // This function makes a copy of input dictionary, modifies it such that it "should" (as much as we can help it)
+    // not contain any objects that do not conform to NSCoding protocol, and returns it autoreleased.
+
+    NSMutableDictionary* dst = [src mutableCopy];
+
+    // Kill the known problem entries.
+    [dst removeObjectForKey:@"NSErrorPeerCertificateChainKey"]; // NSArray with SecCertificateRef objects
+    [dst removeObjectForKey:@"NSErrorClientCertificateChainKey"]; // NSArray with SecCertificateRef objects
+    [dst removeObjectForKey:NSURLErrorFailingURLPeerTrustErrorKey]; // SecTrustRef object
+    [dst removeObjectForKey:NSUnderlyingErrorKey]; // (Immutable) CFError containing kCF equivalent of the above
+    // We could reconstitute this but it's more trouble than it's worth
+
+    // Non-comprehensive safety check:  Kill top-level dictionary entries that don't conform to NSCoding.
+    // We may hit ones we just removed, but that's fine.
+    // We don't handle arbitrary objects that clients have stuffed into the dictionary, since we may not know how to
+    // get at its conents (e.g., a CFError object -- you'd have to know it had a userInfo dictionary and kill things
+    // inside it).
+    [src enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL*) {
+        if (! [obj conformsToProtocol:@protocol(NSCoding)]) {
+            [dst removeObjectForKey:key];
+        }
+        // FIXME: We could drill down into subdictionaries, but it seems more trouble than it's worth
+    }];
+
+    return [dst autorelease];
+}
+
+- (void)encodeWithCoder:(NSCoder *)coder
+{
+    NSDictionary* newUserInfo = dictionaryThatCanCode([self userInfo]);
+
+    [[NSError errorWithDomain:[self domain] code:[self code] userInfo:newUserInfo] encodeWithCoder:coder];
+}
+
+@end
+
+#endif // PLATFORM(IOS)
+
 namespace WebCore {
 
+static RetainPtr<NSError> createNSErrorFromResourceErrorBase(const ResourceErrorBase& resourceError)
+{
+    RetainPtr<NSMutableDictionary> userInfo = adoptNS([[NSMutableDictionary alloc] init]);
+
+    if (!resourceError.localizedDescription().isEmpty())
+        [userInfo.get() setValue:resourceError.localizedDescription() forKey:NSLocalizedDescriptionKey];
+
+    if (!resourceError.failingURL().isEmpty()) {
+        [userInfo.get() setValue:(NSString *)resourceError.failingURL().string() forKey:@"NSErrorFailingURLStringKey"];
+        if (NSURL *cocoaURL = (NSURL *)resourceError.failingURL())
+            [userInfo.get() setValue:cocoaURL forKey:@"NSErrorFailingURLKey"];
+    }
+
+    return adoptNS([[NSError alloc] initWithDomain:resourceError.domain() code:resourceError.errorCode() userInfo:userInfo.get()]);
+}
+
+ResourceError::ResourceError(NSError *nsError)
+    : ResourceErrorBase(Type::Null)
+    , m_dataIsUpToDate(false)
+    , m_platformError(nsError)
+{
+    if (nsError)
+        setType(([m_platformError.get() code] == NSURLErrorTimedOut) ? Type::Timeout : Type::General);
+}
+
+ResourceError::ResourceError(CFErrorRef cfError)
+    : ResourceError((NSError *)cfError)
+{
+}
+
 void ResourceError::platformLazyInit()
 {
     if (m_dataIsUpToDate)
@@ -44,10 +129,10 @@ void ResourceError::platformLazyInit()
     m_domain = [m_platformError.get() domain];
     m_errorCode = [m_platformError.get() code];
 
-    NSString* failingURLString = [[m_platformError.get() userInfo] valueForKey:@"NSErrorFailingURLStringKey"];
-    if (!failingURLString)
-        failingURLString = [[[m_platformError.get() userInfo] valueForKey:@"NSErrorFailingURLKey"] absoluteString];
-    m_failingURL = failingURLString; 
+    if (NSString* failingURLString = [[m_platformError.get() userInfo] valueForKey:@"NSErrorFailingURLStringKey"])
+        m_failingURL = URL(URL(), failingURLString);
+    else
+        m_failingURL = URL((NSURL *)[[m_platformError.get() userInfo] valueForKey:@"NSErrorFailingURLKey"]);
     // Workaround for <rdar://problem/6554067>
     m_localizedDescription = m_failingURL;
     BEGIN_BLOCK_OBJC_EXCEPTIONS;
@@ -59,32 +144,39 @@ void ResourceError::platformLazyInit()
 
 bool ResourceError::platformCompare(const ResourceError& a, const ResourceError& b)
 {
-    return (NSError*)a == (NSError*)b;
+    return a.nsError() == b.nsError();
+}
+
+void ResourceError::doPlatformIsolatedCopy(const ResourceError&)
+{
 }
 
-ResourceError::operator NSError*() const
+NSError *ResourceError::nsError() const
 {
-    if (m_isNull) {
+    if (isNull()) {
         ASSERT(!m_platformError);
         return nil;
     }
-    
-    if (!m_platformError) {
-        RetainPtr<NSMutableDictionary> userInfo(AdoptNS, [[NSMutableDictionary alloc] init]);
 
-        if (!m_localizedDescription.isEmpty())
-            [userInfo.get() setValue:m_localizedDescription forKey:NSLocalizedDescriptionKey];
+    if (!m_platformError)
+        m_platformError = createNSErrorFromResourceErrorBase(*this);
 
-        if (!m_failingURL.isEmpty()) {
-            NSURL *cocoaURL = KURL(ParsedURLString, m_failingURL);
-            [userInfo.get() setValue:m_failingURL forKey:@"NSErrorFailingURLStringKey"];
-            [userInfo.get() setValue:cocoaURL forKey:@"NSErrorFailingURLKey"];
-        }
+    return m_platformError.get();
+}
 
-        m_platformError.adoptNS([[NSError alloc] initWithDomain:m_domain code:m_errorCode userInfo:userInfo.get()]);
-    }
+ResourceError::operator NSError *() const
+{
+    return nsError();
+}
 
-    return m_platformError.get();
+CFErrorRef ResourceError::cfError() const
+{
+    return (CFErrorRef)nsError();
+}
+
+ResourceError::operator CFErrorRef() const
+{
+    return cfError();
 }
 
 } // namespace WebCore