/* * 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 #import #import #import #import @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:@":"]; } - (void)_webkit_setMetadataURL:(NSString *)URLString referrer:(NSString *)referrer atPath:(NSString *)path { ASSERT(URLString); ASSERT(path); WKSetMetadataURL(URLString, referrer, path); } - (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]; } - (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]; } - (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