2010-07-22 07:25:24 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
// vim:set ts=2 sts=2 sw=2 et cin:
|
2012-05-21 15:12:37 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2010-07-22 07:25:24 +04:00
|
|
|
|
|
|
|
#include "CocoaFileUtils.h"
|
2013-10-09 21:11:13 +04:00
|
|
|
#include "nsCocoaUtils.h"
|
2010-07-22 07:25:24 +04:00
|
|
|
#include <Cocoa/Cocoa.h>
|
|
|
|
#include "nsObjCExceptions.h"
|
|
|
|
#include "nsDebug.h"
|
2018-07-23 17:01:26 +03:00
|
|
|
#include "nsString.h"
|
|
|
|
#include "mozilla/MacStringHelpers.h"
|
2016-10-03 15:52:41 +03:00
|
|
|
|
2010-07-22 07:25:24 +04:00
|
|
|
namespace CocoaFileUtils {
|
|
|
|
|
|
|
|
nsresult RevealFileInFinder(CFURLRef url) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoreleasePool localPool;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
|
|
|
BOOL success = [[NSWorkspace sharedWorkspace] selectFile:[(NSURL*)url path]
|
|
|
|
inFileViewerRootedAtPath:@""];
|
|
|
|
|
|
|
|
return (success ? NS_OK : NS_ERROR_FAILURE);
|
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult OpenURL(CFURLRef url) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoreleasePool localPool;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-29 22:04:20 +03:00
|
|
|
[[NSWorkspace sharedWorkspace] openURL:(NSURL*)url];
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-29 22:04:20 +03:00
|
|
|
return NS_OK;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult GetFileCreatorCode(CFURLRef url, OSType* creatorCode) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url) || NS_WARN_IF(!creatorCode)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2013-10-09 21:11:13 +04:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
|
|
|
NSString* resolvedPath = [[(NSURL*)url path] stringByResolvingSymlinksInPath];
|
|
|
|
if (!resolvedPath) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
NSDictionary* dict =
|
|
|
|
[[NSFileManager defaultManager] attributesOfItemAtPath:resolvedPath
|
|
|
|
error:nil];
|
2013-10-09 21:11:13 +04:00
|
|
|
if (!dict) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2010-07-22 07:25:24 +04:00
|
|
|
|
|
|
|
NSNumber* creatorNum = (NSNumber*)[dict objectForKey:NSFileHFSCreatorCode];
|
2013-10-09 21:11:13 +04:00
|
|
|
if (!creatorNum) {
|
|
|
|
return NS_ERROR_FAILURE;
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
2013-10-09 21:11:13 +04:00
|
|
|
*creatorCode = [creatorNum unsignedLongValue];
|
|
|
|
return NS_OK;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult SetFileCreatorCode(CFURLRef url, OSType creatorCode) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoreleasePool localPool;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
NSDictionary* dict = [NSDictionary
|
|
|
|
dictionaryWithObject:[NSNumber numberWithUnsignedLong:creatorCode]
|
|
|
|
forKey:NSFileHFSCreatorCode];
|
|
|
|
BOOL success =
|
|
|
|
[[NSFileManager defaultManager] setAttributes:dict
|
|
|
|
ofItemAtPath:[(NSURL*)url path]
|
|
|
|
error:nil];
|
2024-01-08 18:14:52 +03:00
|
|
|
|
2010-07-22 07:25:24 +04:00
|
|
|
return (success ? NS_OK : NS_ERROR_FAILURE);
|
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult GetFileTypeCode(CFURLRef url, OSType* typeCode) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url) || NS_WARN_IF(!typeCode)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2013-10-09 21:11:13 +04:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
|
|
|
NSString* resolvedPath = [[(NSURL*)url path] stringByResolvingSymlinksInPath];
|
|
|
|
if (!resolvedPath) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
NSDictionary* dict =
|
|
|
|
[[NSFileManager defaultManager] attributesOfItemAtPath:resolvedPath
|
|
|
|
error:nil];
|
2013-10-09 21:11:13 +04:00
|
|
|
if (!dict) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2010-07-22 07:25:24 +04:00
|
|
|
|
|
|
|
NSNumber* typeNum = (NSNumber*)[dict objectForKey:NSFileHFSTypeCode];
|
2013-10-09 21:11:13 +04:00
|
|
|
if (!typeNum) {
|
|
|
|
return NS_ERROR_FAILURE;
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
2013-10-09 21:11:13 +04:00
|
|
|
*typeCode = [typeNum unsignedLongValue];
|
|
|
|
return NS_OK;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult SetFileTypeCode(CFURLRef url, OSType typeCode) {
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
if (NS_WARN_IF(!url)) {
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoreleasePool localPool;
|
2010-07-22 07:25:24 +04:00
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
NSDictionary* dict = [NSDictionary
|
|
|
|
dictionaryWithObject:[NSNumber numberWithUnsignedLong:typeCode]
|
|
|
|
forKey:NSFileHFSTypeCode];
|
|
|
|
BOOL success =
|
|
|
|
[[NSFileManager defaultManager] setAttributes:dict
|
|
|
|
ofItemAtPath:[(NSURL*)url path]
|
|
|
|
error:nil];
|
2024-01-08 18:14:52 +03:00
|
|
|
|
2010-07-22 07:25:24 +04:00
|
|
|
return (success ? NS_OK : NS_ERROR_FAILURE);
|
|
|
|
|
2021-02-17 01:55:20 +03:00
|
|
|
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
|
2010-07-22 07:25:24 +04:00
|
|
|
}
|
|
|
|
|
2018-08-18 02:07:51 +03:00
|
|
|
// Can be called off of the main thread.
|
2023-09-05 14:23:01 +03:00
|
|
|
void AddOriginMetadataToFile(const CFStringRef filePath,
|
|
|
|
const CFURLRef sourceURL,
|
2016-09-13 10:25:36 +03:00
|
|
|
const CFURLRef referrerURL) {
|
2024-01-08 18:14:52 +03:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
typedef OSStatus (*MDItemSetAttribute_type)(MDItemRef, CFStringRef,
|
|
|
|
CFTypeRef);
|
2016-09-13 10:25:36 +03:00
|
|
|
static MDItemSetAttribute_type mdItemSetAttributeFunc = NULL;
|
|
|
|
|
|
|
|
static bool did_symbol_lookup = false;
|
|
|
|
if (!did_symbol_lookup) {
|
|
|
|
did_symbol_lookup = true;
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFBundleRef metadata_bundle =
|
|
|
|
::CFBundleGetBundleWithIdentifier(CFSTR("com.apple.Metadata"));
|
2016-09-13 10:25:36 +03:00
|
|
|
if (!metadata_bundle) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
mdItemSetAttributeFunc =
|
|
|
|
(MDItemSetAttribute_type)::CFBundleGetFunctionPointerForName(
|
|
|
|
metadata_bundle, CFSTR("MDItemSetAttribute"));
|
2016-09-13 10:25:36 +03:00
|
|
|
}
|
|
|
|
if (!mdItemSetAttributeFunc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
MDItemRef mdItem = ::MDItemCreate(NULL, filePath);
|
|
|
|
if (!mdItem) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
CFMutableArrayRef list = ::CFArrayCreateMutable(kCFAllocatorDefault, 2, NULL);
|
|
|
|
if (!list) {
|
|
|
|
::CFRelease(mdItem);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The first item in the list is the source URL of the downloaded file.
|
|
|
|
if (sourceURL) {
|
|
|
|
::CFArrayAppendValue(list, ::CFURLGetString(sourceURL));
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the referrer is known, store that in the second position.
|
|
|
|
if (referrerURL) {
|
|
|
|
::CFArrayAppendValue(list, ::CFURLGetString(referrerURL));
|
|
|
|
}
|
|
|
|
|
|
|
|
mdItemSetAttributeFunc(mdItem, kMDItemWhereFroms, list);
|
|
|
|
|
|
|
|
::CFRelease(list);
|
|
|
|
::CFRelease(mdItem);
|
|
|
|
}
|
|
|
|
|
2018-08-18 02:07:51 +03:00
|
|
|
// Can be called off of the main thread.
|
2023-09-05 14:23:01 +03:00
|
|
|
static CFMutableDictionaryRef CreateQuarantineDictionary(
|
|
|
|
const CFURLRef aFileURL, const bool aCreateProps) {
|
2024-01-08 18:14:52 +03:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2016-09-13 10:25:36 +03:00
|
|
|
CFDictionaryRef quarantineProps = NULL;
|
2018-07-23 17:01:26 +03:00
|
|
|
if (aCreateProps) {
|
2023-09-05 14:23:01 +03:00
|
|
|
quarantineProps = ::CFDictionaryCreate(NULL, NULL, NULL, 0,
|
|
|
|
&kCFTypeDictionaryKeyCallBacks,
|
2018-07-23 17:01:26 +03:00
|
|
|
&kCFTypeDictionaryValueCallBacks);
|
|
|
|
} else {
|
2023-09-05 14:23:01 +03:00
|
|
|
Boolean success = ::CFURLCopyResourcePropertyForKey(
|
|
|
|
aFileURL, kCFURLQuarantinePropertiesKey, &quarantineProps, NULL);
|
2018-07-23 17:01:26 +03:00
|
|
|
// If there aren't any quarantine properties then the user probably
|
|
|
|
// set up an exclusion and we don't need to add metadata.
|
|
|
|
if (!success || !quarantineProps) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-09-13 10:25:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// We don't know what to do if the props aren't a dictionary.
|
|
|
|
if (::CFGetTypeID(quarantineProps) != ::CFDictionaryGetTypeID()) {
|
|
|
|
::CFRelease(quarantineProps);
|
2018-07-23 17:01:26 +03:00
|
|
|
return NULL;
|
2016-09-13 10:25:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Make a mutable copy of the properties.
|
2023-09-05 14:23:01 +03:00
|
|
|
CFMutableDictionaryRef mutQuarantineProps = ::CFDictionaryCreateMutableCopy(
|
|
|
|
kCFAllocatorDefault, 0, (CFDictionaryRef)quarantineProps);
|
2016-09-13 10:25:36 +03:00
|
|
|
::CFRelease(quarantineProps);
|
|
|
|
|
2018-07-23 17:01:26 +03:00
|
|
|
return mutQuarantineProps;
|
|
|
|
}
|
|
|
|
|
2018-08-18 02:07:51 +03:00
|
|
|
// Can be called off of the main thread.
|
2023-09-05 14:23:01 +03:00
|
|
|
void AddQuarantineMetadataToFile(const CFStringRef filePath,
|
|
|
|
const CFURLRef sourceURL,
|
|
|
|
const CFURLRef referrerURL,
|
|
|
|
const bool isFromWeb,
|
2018-07-23 17:01:26 +03:00
|
|
|
const bool createProps /* = false */) {
|
2024-01-08 18:14:52 +03:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFURLRef fileURL = ::CFURLCreateWithFileSystemPath(
|
|
|
|
kCFAllocatorDefault, filePath, kCFURLPOSIXPathStyle, false);
|
2018-07-23 17:01:26 +03:00
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFMutableDictionaryRef mutQuarantineProps =
|
|
|
|
CreateQuarantineDictionary(fileURL, createProps);
|
2018-07-23 17:01:26 +03:00
|
|
|
if (!mutQuarantineProps) {
|
|
|
|
::CFRelease(fileURL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-13 10:25:36 +03:00
|
|
|
// Add metadata that the OS couldn't infer.
|
|
|
|
|
|
|
|
if (!::CFDictionaryGetValue(mutQuarantineProps, kLSQuarantineTypeKey)) {
|
2023-09-05 14:23:01 +03:00
|
|
|
CFStringRef type = isFromWeb ? kLSQuarantineTypeWebDownload
|
|
|
|
: kLSQuarantineTypeOtherDownload;
|
2016-09-13 10:25:36 +03:00
|
|
|
::CFDictionarySetValue(mutQuarantineProps, kLSQuarantineTypeKey, type);
|
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
if (!::CFDictionaryGetValue(mutQuarantineProps, kLSQuarantineOriginURLKey) &&
|
|
|
|
referrerURL) {
|
|
|
|
::CFDictionarySetValue(mutQuarantineProps, kLSQuarantineOriginURLKey,
|
|
|
|
referrerURL);
|
2016-09-13 10:25:36 +03:00
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
if (!::CFDictionaryGetValue(mutQuarantineProps, kLSQuarantineDataURLKey) &&
|
|
|
|
sourceURL) {
|
|
|
|
::CFDictionarySetValue(mutQuarantineProps, kLSQuarantineDataURLKey,
|
|
|
|
sourceURL);
|
2016-09-13 10:25:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set quarantine properties on file.
|
2023-09-05 14:23:01 +03:00
|
|
|
::CFURLSetResourcePropertyForKey(fileURL, kCFURLQuarantinePropertiesKey,
|
|
|
|
mutQuarantineProps, NULL);
|
2016-09-13 10:25:36 +03:00
|
|
|
|
|
|
|
::CFRelease(fileURL);
|
|
|
|
::CFRelease(mutQuarantineProps);
|
|
|
|
}
|
|
|
|
|
2018-08-18 02:07:51 +03:00
|
|
|
// Can be called off of the main thread.
|
2023-09-05 14:23:01 +03:00
|
|
|
void CopyQuarantineReferrerUrl(const CFStringRef aFilePath,
|
|
|
|
nsAString& aReferrer) {
|
2024-01-08 18:14:52 +03:00
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFURLRef fileURL = ::CFURLCreateWithFileSystemPath(
|
|
|
|
kCFAllocatorDefault, aFilePath, kCFURLPOSIXPathStyle, false);
|
2018-07-23 17:01:26 +03:00
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFMutableDictionaryRef mutQuarantineProps =
|
|
|
|
CreateQuarantineDictionary(fileURL, false);
|
2018-07-23 17:01:26 +03:00
|
|
|
::CFRelease(fileURL);
|
|
|
|
if (!mutQuarantineProps) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
CFTypeRef referrerRef =
|
|
|
|
::CFDictionaryGetValue(mutQuarantineProps, kLSQuarantineOriginURLKey);
|
2018-07-23 17:01:26 +03:00
|
|
|
if (referrerRef && ::CFGetTypeID(referrerRef) == ::CFURLGetTypeID()) {
|
|
|
|
// URL string must be copied prior to releasing the dictionary.
|
|
|
|
mozilla::CopyCocoaStringToXPCOMString(
|
2023-09-05 14:23:01 +03:00
|
|
|
(NSString*)::CFURLGetString(static_cast<CFURLRef>(referrerRef)),
|
|
|
|
aReferrer);
|
2018-07-23 17:01:26 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
::CFRelease(mutQuarantineProps);
|
|
|
|
}
|
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
CFTypeRefPtr<CFURLRef> GetTemporaryFolder() {
|
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2017-06-08 03:40:50 +03:00
|
|
|
NSString* tempDir = ::NSTemporaryDirectory();
|
2024-01-08 18:14:52 +03:00
|
|
|
return tempDir == nil ? NULL
|
|
|
|
: CFTypeRefPtr<CFURLRef>::WrapUnderGetRule(
|
|
|
|
(__bridge CFURLRef)[NSURL fileURLWithPath:tempDir
|
|
|
|
isDirectory:YES]);
|
2017-06-08 03:40:50 +03:00
|
|
|
}
|
|
|
|
|
2024-01-08 18:14:52 +03:00
|
|
|
CFTypeRefPtr<CFURLRef> GetProductDirectory(bool aLocal) {
|
|
|
|
nsAutoreleasePool localPool;
|
|
|
|
|
2023-09-05 14:23:01 +03:00
|
|
|
NSSearchPathDirectory folderType =
|
|
|
|
aLocal ? NSCachesDirectory : NSLibraryDirectory;
|
2022-12-20 12:04:57 +03:00
|
|
|
NSFileManager* manager = [NSFileManager defaultManager];
|
2024-01-08 18:14:52 +03:00
|
|
|
return CFTypeRefPtr<CFURLRef>::WrapUnderGetRule((__bridge CFURLRef)[[manager
|
|
|
|
URLsForDirectory:folderType
|
|
|
|
inDomains:NSUserDomainMask] firstObject]);
|
2022-12-20 12:04:57 +03:00
|
|
|
}
|
|
|
|
|
2010-07-22 07:25:24 +04:00
|
|
|
} // namespace CocoaFileUtils
|