Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(612)

Side by Side Diff: chrome/browser/cocoa/file_metadata.mm

Issue 220037: Add kMDItemWhereFroms metadata attribute to downloaded files. Combine with qu... (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/
Patch Set: '' Created 11 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome/browser/cocoa/file_metadata.h ('k') | chrome/browser/download/download_file.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Property Changes:
Name: svn:eol-style
+ LF
OLDNEW
(Empty)
1 // Copyright (c) 2009 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "chrome/browser/cocoa/file_metadata.h"
6
7 #include <ApplicationServices/ApplicationServices.h>
8 #include <Foundation/Foundation.h>
9
10 #include "base/file_path.h"
11 #include "base/scoped_cftyperef.h"
12 #include "base/logging.h"
13 #include "base/mac_util.h"
14 #include "googleurl/src/gurl.h"
15
16 namespace file_metadata {
17
18 // As of Mac OS X 10.4 ("Tiger"), files can be tagged with metadata describing
19 // various attributes. Metadata is integrated with the system's Spotlight
20 // feature and is searchable. Ordinarily, metadata can only be set by
21 // Spotlight importers, which requires that the importer own the target file.
22 // However, there's an attribute intended to describe the origin of a
23 // file, that can store the source URL and referrer of a downloaded file.
24 // It's stored as a "com.apple.metadata:kMDItemWhereFroms" extended attribute,
25 // structured as a binary1-format plist containing a list of sources. This
26 // attribute can only be populated by the downloader, not a Spotlight importer.
27 // Safari on 10.4 and later populates this attribute.
28 //
29 // With this metadata set, you can locate downloads by performing a Spotlight
30 // search for their source or referrer URLs, either from within the Spotlight
31 // UI or from the command line:
32 // mdfind 'kMDItemWhereFroms == "http://releases.mozilla.org/*"'
33 //
34 // There is no documented API to set metadata on a file directly as of the
35 // 10.5 SDK. The MDSetItemAttribute function does exist to perform this task,
36 // but it's undocumented.
37 void AddOriginMetadataToFile(const FilePath& file, const GURL& source,
38 const GURL& referrer) {
39 // There's no declaration for MDItemSetAttribute in any known public SDK.
40 // It exists in the 10.4 and 10.5 runtimes. To play it safe, do the lookup
41 // at runtime instead of declaring it ourselves and linking against what's
42 // provided. This has two benefits:
43 // - If Apple relents and declares the function in a future SDK (it's
44 // happened before), our build won't break.
45 // - If Apple removes or renames the function in a future runtime, the
46 // loader won't refuse to let the application launch. Instead, we'll
47 // silently fail to set any metadata.
48 typedef OSStatus (*MDItemSetAttribute_type)(MDItemRef, CFStringRef,
49 CFTypeRef);
50 static MDItemSetAttribute_type md_item_set_attribute_func = NULL;
51
52 static bool did_symbol_lookup = false;
53 if (!did_symbol_lookup) {
54 did_symbol_lookup = true;
55 CFBundleRef metadata_bundle =
56 ::CFBundleGetBundleWithIdentifier(CFSTR("com.apple.Metadata"));
57 if (!metadata_bundle)
58 return;
59
60 md_item_set_attribute_func = (MDItemSetAttribute_type)
61 CFBundleGetFunctionPointerForName(metadata_bundle,
62 CFSTR("MDItemSetAttribute"));
63 }
64 if (!md_item_set_attribute_func)
65 return;
66
67 NSString* file_path =
68 [NSString stringWithUTF8String:file.value().c_str()];
69 if (!file_path)
70 return;
71
72 scoped_cftyperef<MDItemRef> md_item(
73 MDItemCreate(NULL, reinterpret_cast<CFStringRef>(file_path)));
74 if (!md_item)
75 return;
76
77 // We won't put any more than 2 items into the attribute.
78 NSMutableArray* list = [NSMutableArray arrayWithCapacity:2];
79
80 // Follow Safari's lead: the first item in the list is the source URL of
81 // the downloaded file. If the referrer is known, store that, too.
82 NSString* origin_url = [NSString stringWithUTF8String:source.spec().c_str()];
83 if (origin_url)
84 [list addObject:origin_url];
85 NSString* referrer_url =
86 [NSString stringWithUTF8String:referrer.spec().c_str()];
87 if (referrer_url)
88 [list addObject:referrer_url];
89
90 md_item_set_attribute_func(md_item, kMDItemWhereFroms,
91 reinterpret_cast<CFArrayRef>(list));
92 }
93
94 // The OS will automatically quarantine files due to the
95 // LSFileQuarantineEnabled entry in our Info.plist, but it knows relatively
96 // little about the files. We add more information about the download to
97 // improve the UI shown by the OS when the users tries to open the file.
98 void AddQuarantineMetadataToFile(const FilePath& file, const GURL& source,
99 const GURL& referrer) {
100 FSRef file_ref;
101 if (!mac_util::FSRefFromPath(file.value(), &file_ref))
102 return;
103
104 NSMutableDictionary* quarantine_properties = nil;
105 CFTypeRef quarantine_properties_base = NULL;
106 if (::LSCopyItemAttribute(&file_ref, kLSRolesAll, kLSItemQuarantineProperties,
107 &quarantine_properties_base) == noErr) {
108 if (::CFGetTypeID(quarantine_properties_base) ==
109 ::CFDictionaryGetTypeID()) {
110 // Quarantine properties will already exist if LSFileQuarantineEnabled
111 // is on and the file doesn't match an exclusion.
112 quarantine_properties =
113 [[(NSDictionary*)quarantine_properties_base mutableCopy] autorelease];
114 } else {
115 LOG(WARNING) << "kLSItemQuarantineProperties is not a dictionary on file "
116 << file.value();
117 }
118 ::CFRelease(quarantine_properties_base);
119 }
120
121 if (!quarantine_properties) {
122 // If there are no quarantine properties, then the file isn't quarantined
123 // (e.g., because the user has set up exclusions for certain file types).
124 // We don't want to add any metadata, because that will cause the file to
125 // be quarantined against the user's wishes.
126 return;
127 }
128
129 // kLSQuarantineAgentNameKey, kLSQuarantineAgentBundleIdentifierKey, and
130 // kLSQuarantineTimeStampKey are set for us (see LSQuarantine.h), so we only
131 // need to set the values that the OS can't infer.
132
133 if (![quarantine_properties valueForKey:(NSString*)kLSQuarantineTypeKey]) {
134 CFStringRef type = (source.SchemeIs("http") || source.SchemeIs("https"))
135 ? kLSQuarantineTypeWebDownload
136 : kLSQuarantineTypeOtherDownload;
137 [quarantine_properties setValue:(NSString*)type
138 forKey:(NSString*)kLSQuarantineTypeKey];
139 }
140
141 if (![quarantine_properties
142 valueForKey:(NSString*)kLSQuarantineOriginURLKey] &&
143 referrer.is_valid()) {
144 NSString* referrer_url =
145 [NSString stringWithUTF8String:referrer.spec().c_str()];
146 [quarantine_properties setValue:referrer_url
147 forKey:(NSString*)kLSQuarantineOriginURLKey];
148 }
149
150 if (![quarantine_properties valueForKey:(NSString*)kLSQuarantineDataURLKey] &&
151 source.is_valid()) {
152 NSString* origin_url =
153 [NSString stringWithUTF8String:source.spec().c_str()];
154 [quarantine_properties setValue:origin_url
155 forKey:(NSString*)kLSQuarantineDataURLKey];
156 }
157
158 OSStatus os_error = ::LSSetItemAttribute(&file_ref, kLSRolesAll,
159 kLSItemQuarantineProperties,
160 quarantine_properties);
161 if (os_error != noErr) {
162 LOG(WARNING) << "Unable to set quarantine attributes on file "
163 << file.value();
164 }
165 }
166
167 } // namespace file_metadata
OLDNEW
« no previous file with comments | « chrome/browser/cocoa/file_metadata.h ('k') | chrome/browser/download/download_file.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698