Remove Cocoa CFURLConnection loading code
[WebKit-https.git] / Source / WebCore / platform / network / mac / ResourceErrorMac.mm
index 2209afa..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
 
-namespace WebCore {
+#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
 
-#if USE(CFNETWORK)
+@implementation WebCustomNSURLError
 
-ResourceError::ResourceError(NSError *error)
-    : m_dataIsUpToDate(false)
-    , m_platformError(reinterpret_cast<CFErrorRef>(error))
+static NSDictionary* dictionaryThatCanCode(NSDictionary* src)
 {
-    m_isNull = !error;
+    // 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];
 }
 
-NSError *ResourceError::nsError() const
+- (void)encodeWithCoder:(NSCoder *)coder
 {
-    if (m_isNull) {
-        ASSERT(!m_platformError);
-        return nil;
-    }
-    if (!m_platformNSError) {
-        CFErrorRef error = m_platformError.get();
-        RetainPtr<NSDictionary> userInfo(AdoptCF, (NSDictionary *) CFErrorCopyUserInfo(error));
-        m_platformNSError.adoptNS([[NSError alloc] initWithDomain:(NSString *)CFErrorGetDomain(error) code:CFErrorGetCode(error) userInfo:userInfo.get()]);
-    }
-    return m_platformNSError.get();
+    NSDictionary* newUserInfo = dictionaryThatCanCode([self userInfo]);
+
+    [[NSError errorWithDomain:[self domain] code:[self code] userInfo:newUserInfo] encodeWithCoder:coder];
 }
 
-ResourceError::operator NSError *() const
+@end
+
+#endif // PLATFORM(IOS)
+
+namespace WebCore {
+
+static RetainPtr<NSError> createNSErrorFromResourceErrorBase(const ResourceErrorBase& resourceError)
 {
-    return nsError();
-}
+    RetainPtr<NSMutableDictionary> userInfo = adoptNS([[NSMutableDictionary alloc] init]);
 
-#else
+    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)
-    : m_dataIsUpToDate(false)
+    : ResourceErrorBase(Type::Null)
+    , m_dataIsUpToDate(false)
     , m_platformError(nsError)
 {
-    m_isNull = !nsError;
+    if (nsError)
+        setType(([m_platformError.get() code] == NSURLErrorTimedOut) ? Type::Timeout : Type::General);
 }
 
 ResourceError::ResourceError(CFErrorRef cfError)
-    : m_dataIsUpToDate(false)
-    , m_platformError((NSError *)cfError)
+    : ResourceError((NSError *)cfError)
 {
-    m_isNull = !cfError;
 }
 
 void ResourceError::platformLazyInit()
@@ -89,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;
@@ -107,27 +147,19 @@ bool ResourceError::platformCompare(const ResourceError& a, const ResourceError&
     return a.nsError() == b.nsError();
 }
 
+void ResourceError::doPlatformIsolatedCopy(const ResourceError&)
+{
+}
+
 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_failingURL.isEmpty()) {
-            NSURL *cocoaURL = KURL(ParsedURLString, m_failingURL);
-            [userInfo.get() setValue:m_failingURL forKey:@"NSErrorFailingURLStringKey"];
-            [userInfo.get() setValue:cocoaURL forKey:@"NSErrorFailingURLKey"];
-        }
 
-        m_platformError.adoptNS([[NSError alloc] initWithDomain:m_domain code:m_errorCode userInfo:userInfo.get()]);
-    }
+    if (!m_platformError)
+        m_platformError = createNSErrorFromResourceErrorBase(*this);
 
     return m_platformError.get();
 }
@@ -147,6 +179,4 @@ ResourceError::operator CFErrorRef() const
     return cfError();
 }
 
-#endif // USE(CFNETWORK)
-
 } // namespace WebCore