| // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "net/base/platform_mime_util.h" | 
 |  | 
 | #import <Foundation/Foundation.h> | 
 |  | 
 | #include <string> | 
 |  | 
 | #include "base/mac/foundation_util.h" | 
 | #include "base/mac/scoped_cftyperef.h" | 
 | #include "base/strings/sys_string_conversions.h" | 
 |  | 
 | #if defined(OS_IOS) | 
 | #include <MobileCoreServices/MobileCoreServices.h> | 
 | #else | 
 | #include <CoreServices/CoreServices.h> | 
 | #endif  // defined(OS_IOS) | 
 |  | 
 | #if !defined(OS_IOS) | 
 | // SPI declaration; see the commentary in GetPlatformExtensionsForMimeType. | 
 | // iOS must not use any private API, per Apple guideline. | 
 |  | 
 | @interface NSURLFileTypeMappings : NSObject | 
 | + (NSURLFileTypeMappings*)sharedMappings; | 
 | - (NSArray*)extensionsForMIMEType:(NSString*)mimeType; | 
 | @end | 
 | #endif  // !defined(OS_IOS) | 
 |  | 
 | namespace net { | 
 |  | 
 | bool PlatformMimeUtil::GetPlatformMimeTypeFromExtension( | 
 |     const base::FilePath::StringType& ext, std::string* result) const { | 
 |   std::string ext_nodot = ext; | 
 |   if (ext_nodot.length() >= 1 && ext_nodot[0] == L'.') | 
 |     ext_nodot.erase(ext_nodot.begin()); | 
 |   base::ScopedCFTypeRef<CFStringRef> ext_ref( | 
 |       base::SysUTF8ToCFStringRef(ext_nodot)); | 
 |   if (!ext_ref) | 
 |     return false; | 
 |   base::ScopedCFTypeRef<CFStringRef> uti(UTTypeCreatePreferredIdentifierForTag( | 
 |       kUTTagClassFilenameExtension, ext_ref, NULL)); | 
 |   if (!uti) | 
 |     return false; | 
 |   base::ScopedCFTypeRef<CFStringRef> mime_ref( | 
 |       UTTypeCopyPreferredTagWithClass(uti, kUTTagClassMIMEType)); | 
 |   if (!mime_ref) | 
 |     return false; | 
 |  | 
 |   *result = base::SysCFStringRefToUTF8(mime_ref); | 
 |   return true; | 
 | } | 
 |  | 
 | bool PlatformMimeUtil::GetPlatformPreferredExtensionForMimeType( | 
 |     const std::string& mime_type, | 
 |     base::FilePath::StringType* ext) const { | 
 |   base::ScopedCFTypeRef<CFStringRef> mime_ref( | 
 |       base::SysUTF8ToCFStringRef(mime_type)); | 
 |   if (!mime_ref) | 
 |     return false; | 
 |   base::ScopedCFTypeRef<CFStringRef> uti(UTTypeCreatePreferredIdentifierForTag( | 
 |       kUTTagClassMIMEType, mime_ref, NULL)); | 
 |   if (!uti) | 
 |     return false; | 
 |   base::ScopedCFTypeRef<CFStringRef> ext_ref( | 
 |       UTTypeCopyPreferredTagWithClass(uti, kUTTagClassFilenameExtension)); | 
 |   if (!ext_ref) | 
 |     return false; | 
 |  | 
 |   *ext = base::SysCFStringRefToUTF8(ext_ref); | 
 |   return true; | 
 | } | 
 |  | 
 | void PlatformMimeUtil::GetPlatformExtensionsForMimeType( | 
 |     const std::string& mime_type, | 
 |     std::unordered_set<base::FilePath::StringType>* extensions) const { | 
 | #if defined(OS_IOS) | 
 |   NSArray* extensions_list = nil; | 
 | #else | 
 |   // There is no API for this that uses UTIs. The WebKitSystemInterface call | 
 |   // WKGetExtensionsForMIMEType() is a thin wrapper around | 
 |   // [[NSURLFileTypeMappings sharedMappings] extensionsForMIMEType:], which is | 
 |   // used by Firefox as well. | 
 |   // | 
 |   // See: | 
 |   // http://mxr.mozilla.org/mozilla-central/search?string=extensionsForMIMEType | 
 |   // http://www.openradar.me/11384153 | 
 |   // rdar://11384153 | 
 |   NSArray* extensions_list = | 
 |       [[NSURLFileTypeMappings sharedMappings] | 
 |           extensionsForMIMEType:base::SysUTF8ToNSString(mime_type)]; | 
 | #endif  // defined(OS_IOS) | 
 |  | 
 |   if (extensions_list) { | 
 |     for (NSString* extension in extensions_list) | 
 |       extensions->insert(base::SysNSStringToUTF8(extension)); | 
 |   } else { | 
 |     // Huh? Give up. | 
 |     base::FilePath::StringType ext; | 
 |     if (GetPlatformPreferredExtensionForMimeType(mime_type, &ext)) | 
 |       extensions->insert(ext); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace net |