blob: 9dc504fcc49722dc7d7f08c40c0b168d987d8d2f [file] [log] [blame]
// 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 "content/browser/download/file_metadata_mac.h"
#include <ApplicationServices/ApplicationServices.h>
#include <Foundation/Foundation.h>
#include "base/files/file_path.h"
#include "base/logging.h"
#include "base/mac/mac_logging.h"
#include "base/mac/mac_util.h"
#include "base/mac/scoped_cftyperef.h"
#include "url/gurl.h"
namespace content {
// As of Mac OS X 10.4 ("Tiger"), files can be tagged with metadata describing
// various attributes. Metadata is integrated with the system's Spotlight
// feature and is searchable. Ordinarily, metadata can only be set by
// Spotlight importers, which requires that the importer own the target file.
// However, there's an attribute intended to describe the origin of a
// file, that can store the source URL and referrer of a downloaded file.
// It's stored as a "com.apple.metadata:kMDItemWhereFroms" extended attribute,
// structured as a binary1-format plist containing a list of sources. This
// attribute can only be populated by the downloader, not a Spotlight importer.
// Safari on 10.4 and later populates this attribute.
//
// With this metadata set, you can locate downloads by performing a Spotlight
// search for their source or referrer URLs, either from within the Spotlight
// UI or from the command line:
// mdfind 'kMDItemWhereFroms == "http://releases.mozilla.org/*"'
//
// There is no documented API to set metadata on a file directly as of the
// 10.5 SDK. The MDSetItemAttribute function does exist to perform this task,
// but it's undocumented.
void AddOriginMetadataToFile(const base::FilePath& file, const GURL& source,
const GURL& referrer) {
// There's no declaration for MDItemSetAttribute in any known public SDK.
// It exists in the 10.4 and 10.5 runtimes. To play it safe, do the lookup
// at runtime instead of declaring it ourselves and linking against what's
// provided. This has two benefits:
// - If Apple relents and declares the function in a future SDK (it's
// happened before), our build won't break.
// - If Apple removes or renames the function in a future runtime, the
// loader won't refuse to let the application launch. Instead, we'll
// silently fail to set any metadata.
typedef OSStatus (*MDItemSetAttribute_type)(MDItemRef, CFStringRef,
CFTypeRef);
static MDItemSetAttribute_type md_item_set_attribute_func = NULL;
static bool did_symbol_lookup = false;
if (!did_symbol_lookup) {
did_symbol_lookup = true;
CFBundleRef metadata_bundle =
CFBundleGetBundleWithIdentifier(CFSTR("com.apple.Metadata"));
if (!metadata_bundle)
return;
md_item_set_attribute_func = (MDItemSetAttribute_type)
CFBundleGetFunctionPointerForName(metadata_bundle,
CFSTR("MDItemSetAttribute"));
}
if (!md_item_set_attribute_func)
return;
NSString* file_path =
[NSString stringWithUTF8String:file.value().c_str()];
if (!file_path)
return;
base::ScopedCFTypeRef<MDItemRef> md_item(
MDItemCreate(NULL, base::mac::NSToCFCast(file_path)));
if (!md_item)
return;
// We won't put any more than 2 items into the attribute.
NSMutableArray* list = [NSMutableArray arrayWithCapacity:2];
// Follow Safari's lead: the first item in the list is the source URL of
// the downloaded file. If the referrer is known, store that, too.
NSString* origin_url = [NSString stringWithUTF8String:source.spec().c_str()];
if (origin_url)
[list addObject:origin_url];
NSString* referrer_url =
[NSString stringWithUTF8String:referrer.spec().c_str()];
if (referrer_url)
[list addObject:referrer_url];
md_item_set_attribute_func(md_item, kMDItemWhereFroms,
base::mac::NSToCFCast(list));
}
// The OS will automatically quarantine files due to the
// LSFileQuarantineEnabled entry in our Info.plist, but it knows relatively
// little about the files. We add more information about the download to
// improve the UI shown by the OS when the users tries to open the file.
void AddQuarantineMetadataToFile(const base::FilePath& file, const GURL& source,
const GURL& referrer) {
FSRef file_ref;
if (!base::mac::FSRefFromPath(file.value(), &file_ref))
return;
NSMutableDictionary* quarantine_properties = nil;
CFTypeRef quarantine_properties_base = NULL;
if (LSCopyItemAttribute(&file_ref, kLSRolesAll, kLSItemQuarantineProperties,
&quarantine_properties_base) == noErr) {
if (CFGetTypeID(quarantine_properties_base) ==
CFDictionaryGetTypeID()) {
// Quarantine properties will already exist if LSFileQuarantineEnabled
// is on and the file doesn't match an exclusion.
quarantine_properties =
[[(NSDictionary*)quarantine_properties_base mutableCopy] autorelease];
} else {
LOG(WARNING) << "kLSItemQuarantineProperties is not a dictionary on file "
<< file.value();
}
CFRelease(quarantine_properties_base);
}
if (!quarantine_properties) {
// If there are no quarantine properties, then the file isn't quarantined
// (e.g., because the user has set up exclusions for certain file types).
// We don't want to add any metadata, because that will cause the file to
// be quarantined against the user's wishes.
return;
}
// kLSQuarantineAgentNameKey, kLSQuarantineAgentBundleIdentifierKey, and
// kLSQuarantineTimeStampKey are set for us (see LSQuarantine.h), so we only
// need to set the values that the OS can't infer.
if (![quarantine_properties valueForKey:(NSString*)kLSQuarantineTypeKey]) {
CFStringRef type = source.SchemeIsHTTPOrHTTPS()
? kLSQuarantineTypeWebDownload
: kLSQuarantineTypeOtherDownload;
[quarantine_properties setValue:(NSString*)type
forKey:(NSString*)kLSQuarantineTypeKey];
}
if (![quarantine_properties
valueForKey:(NSString*)kLSQuarantineOriginURLKey] &&
referrer.is_valid()) {
NSString* referrer_url =
[NSString stringWithUTF8String:referrer.spec().c_str()];
[quarantine_properties setValue:referrer_url
forKey:(NSString*)kLSQuarantineOriginURLKey];
}
if (![quarantine_properties valueForKey:(NSString*)kLSQuarantineDataURLKey] &&
source.is_valid()) {
NSString* origin_url =
[NSString stringWithUTF8String:source.spec().c_str()];
[quarantine_properties setValue:origin_url
forKey:(NSString*)kLSQuarantineDataURLKey];
}
OSStatus os_error = LSSetItemAttribute(&file_ref, kLSRolesAll,
kLSItemQuarantineProperties,
quarantine_properties);
if (os_error != noErr) {
OSSTATUS_LOG(WARNING, os_error)
<< "Unable to set quarantine attributes on file " << file.value();
}
}
} // namespace content