blob: 5733598efdb3bb3acc6a9e7657c574c595c2ca7d [file] [log] [blame]
/*
* Copyright (C) 2005 Apple Computer, Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "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 OR ITS 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 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#import <WebKit/WebNSFileManagerExtras.h>
#import <WebCore/FoundationExtras.h>
#import <WebKit/WebKitNSStringExtras.h>
#import <WebKitSystemInterface.h>
#import <wtf/Assertions.h>
#import <pthread.h>
#import <sys/mount.h>
@implementation NSFileManager (WebNSFileManagerExtras)
- (BOOL)_webkit_removeFileOnlyAtPath:(NSString *)path
{
struct statfs buf;
BOOL result = unlink([path fileSystemRepresentation]) == 0;
// For mysterious reasons, MNT_DOVOLFS is the flag for "supports resource fork"
if ((statfs([path fileSystemRepresentation], &buf) == 0) && !(buf.f_flags & MNT_DOVOLFS)) {
NSString *lastPathComponent = [path lastPathComponent];
if ([lastPathComponent length] != 0 && ![lastPathComponent isEqualToString:@"/"]) {
NSString *resourcePath = [[path stringByDeletingLastPathComponent] stringByAppendingString:[@"._" stringByAppendingString:lastPathComponent]];
if (unlink([resourcePath fileSystemRepresentation]) != 0) {
result = NO;
}
}
}
return result;
}
- (void)_webkit_backgroundRemoveFileAtPath:(NSString *)path
{
NSFileManager *manager;
NSString *moveToSubpath;
NSString *moveToPath;
int i;
manager = [NSFileManager defaultManager];
i = 0;
moveToSubpath = [path stringByDeletingLastPathComponent];
do {
moveToPath = [NSString stringWithFormat:@"%@/.tmp%d", moveToSubpath, i];
i++;
} while ([manager fileExistsAtPath:moveToPath]);
if ([manager moveItemAtPath:path toPath:moveToPath error:NULL])
[NSThread detachNewThreadSelector:@selector(_performRemoveFileAtPath:) toTarget:self withObject:moveToPath];
}
- (void)_webkit_backgroundRemoveLeftoverFiles:(NSString *)path
{
NSFileManager *manager;
NSString *leftoverSubpath;
NSString *leftoverPath;
int i;
manager = [NSFileManager defaultManager];
leftoverSubpath = [path stringByDeletingLastPathComponent];
i = 0;
while (1) {
leftoverPath = [NSString stringWithFormat:@"%@/.tmp%d", leftoverSubpath, i];
if (![manager fileExistsAtPath:leftoverPath]) {
break;
}
[NSThread detachNewThreadSelector:@selector(_performRemoveFileAtPath:) toTarget:self withObject:leftoverPath];
i++;
}
}
- (NSString *)_webkit_carbonPathForPath:(NSString *)posixPath
{
OSStatus error;
FSRef ref, rootRef, parentRef;
FSCatalogInfo info;
NSMutableArray *carbonPathPieces;
HFSUniStr255 nameString;
// Make an FSRef.
error = FSPathMakeRef((const UInt8 *)[posixPath fileSystemRepresentation], &ref, NULL);
if (error != noErr) {
return nil;
}
// Get volume refNum.
error = FSGetCatalogInfo(&ref, kFSCatInfoVolume, &info, NULL, NULL, NULL);
if (error != noErr) {
return nil;
}
// Get root directory FSRef.
error = FSGetVolumeInfo(info.volume, 0, NULL, kFSVolInfoNone, NULL, NULL, &rootRef);
if (error != noErr) {
return nil;
}
// Get the pieces of the path.
carbonPathPieces = [NSMutableArray array];
for (;;) {
error = FSGetCatalogInfo(&ref, kFSCatInfoNone, NULL, &nameString, NULL, &parentRef);
if (error != noErr) {
return nil;
}
[carbonPathPieces insertObject:[NSString stringWithCharacters:nameString.unicode length:nameString.length] atIndex:0];
if (FSCompareFSRefs(&ref, &rootRef) == noErr) {
break;
}
ref = parentRef;
}
// Volume names need trailing : character.
if ([carbonPathPieces count] == 1) {
[carbonPathPieces addObject:@""];
}
return [carbonPathPieces componentsJoinedByString:@":"];
}
typedef struct MetaDataInfo
{
NSString *URLString;
NSString *referrer;
NSString *path;
} MetaDataInfo;
static void *setMetaData(void* context)
{
MetaDataInfo *info = (MetaDataInfo *)context;
WKSetMetadataURL(info->URLString, info->referrer, info->path);
HardRelease(info->URLString);
HardRelease(info->referrer);
HardRelease(info->path);
free(info);
return 0;
}
- (void)_webkit_setMetadataURL:(NSString *)URLString referrer:(NSString *)referrer atPath:(NSString *)path
{
ASSERT(URLString);
ASSERT(path);
// Spawn a background thread for WKSetMetadataURL because this function will not return until mds has
// journaled the data we're're trying to set. Depending on what other I/O is going on, it can take some
// time.
pthread_t tid;
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
MetaDataInfo *info = malloc(sizeof(MetaDataInfo));
info->URLString = HardRetainWithNSRelease([URLString copy]);
info->referrer = HardRetainWithNSRelease([referrer copy]);
info->path = HardRetainWithNSRelease([path copy]);
pthread_create(&tid, &attr, setMetaData, info);
pthread_attr_destroy(&attr);
}
- (NSString *)_webkit_startupVolumeName
{
NSString *path = [self _webkit_carbonPathForPath:@"/"];
return [path substringToIndex:[path length]-1];
}
- (NSString *)_webkit_pathWithUniqueFilenameForPath:(NSString *)path
{
// "Fix" the filename of the path.
NSString *filename = [[path lastPathComponent] _webkit_filenameByFixingIllegalCharacters];
path = [[path stringByDeletingLastPathComponent] stringByAppendingPathComponent:filename];
NSFileManager *fileManager = [NSFileManager defaultManager];
if ([fileManager fileExistsAtPath:path]) {
// Don't overwrite existing file by appending "-n", "-n.ext" or "-n.ext.ext" to the filename.
NSString *extensions = nil;
NSString *pathWithoutExtensions;
NSString *lastPathComponent = [path lastPathComponent];
NSRange periodRange = [lastPathComponent rangeOfString:@"."];
if (periodRange.location == NSNotFound) {
pathWithoutExtensions = path;
} else {
extensions = [lastPathComponent substringFromIndex:periodRange.location + 1];
lastPathComponent = [lastPathComponent substringToIndex:periodRange.location];
pathWithoutExtensions = [[path stringByDeletingLastPathComponent] stringByAppendingPathComponent:lastPathComponent];
}
NSString *pathWithAppendedNumber;
unsigned i;
for (i = 1; 1; i++) {
pathWithAppendedNumber = [NSString stringWithFormat:@"%@-%d", pathWithoutExtensions, i];
path = [extensions length] ? [pathWithAppendedNumber stringByAppendingPathExtension:extensions] : pathWithAppendedNumber;
if (![fileManager fileExistsAtPath:path]) {
break;
}
}
}
return path;
}
@end
#ifdef BUILDING_ON_TIGER
@implementation NSFileManager (WebNSFileManagerTigerForwardCompatibility)
- (NSArray *)contentsOfDirectoryAtPath:(NSString *)path error:(NSError **)error
{
// We don't report errors via the NSError* output parameter, so ensure that the caller does not expect us to do so.
ASSERT_ARG(error, !error);
return [self directoryContentsAtPath:path];
}
- (NSString *)destinationOfSymbolicLinkAtPath:(NSString *)path error:(NSError **)error
{
// We don't report errors via the NSError* output parameter, so ensure that the caller does not expect us to do so.
ASSERT_ARG(error, !error);
return [self pathContentOfSymbolicLinkAtPath:path];
}
- (NSDictionary *)attributesOfFileSystemForPath:(NSString *)path error:(NSError **)error
{
// We don't report errors via the NSError* output parameter, so ensure that the caller does not expect us to do so.
ASSERT_ARG(error, !error);
return [self fileSystemAttributesAtPath:path];
}
- (NSDictionary *)attributesOfItemAtPath:(NSString *)path error:(NSError **)error
{
// We don't report errors via the NSError* output parameter, so ensure that the caller does not expect us to do so.
ASSERT_ARG(error, !error);
return [self fileAttributesAtPath:path traverseLink:NO];
}
- (BOOL)moveItemAtPath:(NSString *)srcPath toPath:(NSString *)dstPath error:(NSError **)error
{
// The implementation of moveItemAtPath:toPath:error: interacts with the NSFileManager's delegate.
// We are not matching that behaviour at the moment, but it should not be a problem as any client
// expecting that would need to call setDelegate: first which will generate a compile-time warning,
// as that method is not available on Tiger.
return [self movePath:srcPath toPath:dstPath handler:nil];
}
- (BOOL)removeItemAtPath:(NSString *)path error:(NSError **)error
{
// The implementation of removeItemAtPath:error: interacts with the NSFileManager's delegate.
// We are not matching that behaviour at the moment, but it should not be a problem as any client
// expecting that would need to call setDelegate: first which will generate a compile-time warning,
// as that method is not available on Tiger.
return [self removeFileAtPath:path handler:nil];
}
@end
#endif