arq_restore/Tree.m
2013-03-09 08:46:38 -05:00

316 lines
11 KiB
Objective-C

//
// Tree.m
// Backup
//
// Created by Stefan Reitshamer on 3/25/09.
// Copyright 2009 PhotoMinds LLC. All rights reserved.
//
#include <sys/stat.h>
#import "StringIO.h"
#import "IntegerIO.h"
#import "BooleanIO.h"
#import "Node.h"
#import "Tree.h"
#import "Blob.h"
#import "DataInputStream.h"
#import "RegexKitLite.h"
#import "BufferedInputStream.h"
#import "NSData-Gzip.h"
#import "GunzipInputStream.h"
#import "BlobKey.h"
#import "NSObject_extra.h"
#import "BlobKeyIO.h"
@interface Tree (internal)
- (BOOL)readHeader:(BufferedInputStream *)is error:(NSError **)error;
@end
@implementation Tree
@synthesize xattrsAreCompressed, xattrsBlobKey, xattrsSize, aclIsCompressed, aclBlobKey, uid, gid, mode, mtime_sec, mtime_nsec, flags, finderFlags, extendedFinderFlags, st_dev, treeVersion, st_rdev;
@synthesize ctime_sec, ctime_nsec, createTime_sec, createTime_nsec, st_nlink, st_ino, st_blocks, st_blksize;
@dynamic aggregateUncompressedDataSize;
+ (NSString *)errorDomain {
return @"TreeErrorDomain";
}
- (id)init {
if (self = [super init]) {
nodes = [[NSMutableDictionary alloc] init];
missingNodes = [[NSMutableDictionary alloc] init];
}
return self;
}
- (id)initWithBufferedInputStream:(BufferedInputStream *)is error:(NSError **)error {
if (self = [super init]) {
nodes = [[NSMutableDictionary alloc] init];
missingNodes = [[NSMutableDictionary alloc] init];
if (![self readHeader:is error:error]) {
[self release];
return nil;
}
if (treeVersion >= 12) {
if (![BooleanIO read:&xattrsAreCompressed from:is error:error]
|| ![BooleanIO read:&aclIsCompressed from:is error:error]) {
[self release];
return nil;
}
}
BOOL ret = [BlobKeyIO read:&xattrsBlobKey from:is treeVersion:treeVersion compressed:xattrsAreCompressed error:error]
&& [IntegerIO readUInt64:&xattrsSize from:is error:error]
&& [BlobKeyIO read:&aclBlobKey from:is treeVersion:treeVersion compressed:aclIsCompressed error:error]
&& [IntegerIO readInt32:&uid from:is error:error]
&& [IntegerIO readInt32:&gid from:is error:error]
&& [IntegerIO readInt32:&mode from:is error:error]
&& [IntegerIO readInt64:&mtime_sec from:is error:error]
&& [IntegerIO readInt64:&mtime_nsec from:is error:error]
&& [IntegerIO readInt64:&flags from:is error:error]
&& [IntegerIO readInt32:&finderFlags from:is error:error]
&& [IntegerIO readInt32:&extendedFinderFlags from:is error:error]
&& [IntegerIO readInt32:&st_dev from:is error:error]
&& [IntegerIO readInt32:&st_ino from:is error:error]
&& [IntegerIO readUInt32:&st_nlink from:is error:error]
&& [IntegerIO readInt32:&st_rdev from:is error:error]
&& [IntegerIO readInt64:&ctime_sec from:is error:error]
&& [IntegerIO readInt64:&ctime_nsec from:is error:error]
&& [IntegerIO readInt64:&st_blocks from:is error:error]
&& [IntegerIO readUInt32:&st_blksize from:is error:error];
[xattrsBlobKey retain];
[aclBlobKey retain];
if (!ret) {
goto initError;
}
if ([xattrsBlobKey sha1] == nil) {
[xattrsBlobKey release];
xattrsBlobKey = nil;
}
if ([aclBlobKey sha1] == nil) {
[aclBlobKey release];
aclBlobKey = nil;
}
if (treeVersion >= 11 && treeVersion <= 16) {
uint64_t unusedAggregateSizeOnDisk;
if (![IntegerIO readUInt64:&unusedAggregateSizeOnDisk from:is error:error]) {
goto initError;
}
}
if (treeVersion >= 15) {
if (![IntegerIO readInt64:&createTime_sec from:is error:error]
|| ![IntegerIO readInt64:&createTime_nsec from:is error:error]) {
goto initError;
}
}
if (treeVersion >= 18) {
uint32_t missingNodeCount;
if (![IntegerIO readUInt32:&missingNodeCount from:is error:error]) {
goto initError;
}
for (uint32_t i = 0; i < missingNodeCount; i++) {
NSString *missingNodeName = nil;
if (![StringIO read:&missingNodeName from:is error:error]) {
goto initError;
}
Node *node = [[[Node alloc] initWithInputStream:is treeVersion:treeVersion error:error] autorelease];
if (node == nil) {
goto initError;
}
[missingNodes setObject:node forKey:missingNodeName];
}
}
uint32_t nodeCount;
if (![IntegerIO readUInt32:&nodeCount from:is error:error]) {
goto initError;
}
for (uint32_t i = 0; i < nodeCount; i++) {
NSString *nodeName;
if (![StringIO read:&nodeName from:is error:error]) {
goto initError;
}
Node *node = [[Node alloc] initWithInputStream:is treeVersion:treeVersion error:error];
if (!node) {
goto initError;
}
[nodes setObject:node forKey:nodeName];
[node release];
}
goto initDone;
initError:
[self release];
self = nil;
}
initDone:
return self;
}
- (void)dealloc {
[xattrsBlobKey release];
[aclBlobKey release];
[nodes release];
[missingNodes release];
[super dealloc];
}
- (NSArray *)childNodeNames {
return [nodes allKeys];
}
- (Node *)childNodeWithName:(NSString *)name {
return [nodes objectForKey:name];
}
- (BOOL)containsNodeNamed:(NSString *)name {
return [nodes objectForKey:name] != nil;
}
- (BOOL)containsMissingItems {
if ([missingNodes count] > 0) {
return YES;
}
for (Node *node in [nodes allValues]) {
if ([node isTree] && [node treeContainsMissingItems]) {
return YES;
}
}
return NO;
}
- (NSArray *)missingChildNodeNames {
return [missingNodes allKeys];
}
- (Node *)missingChildNodeWithName:(NSString *)name {
return [missingNodes objectForKey:name];
}
- (NSDictionary *)nodes {
return nodes;
}
- (NSDictionary *)missingNodes {
return missingNodes;
}
- (NSData *)toData {
NSMutableData *data = [[[NSMutableData alloc] init] autorelease];
[BooleanIO write:xattrsAreCompressed to:data];
[BooleanIO write:aclIsCompressed to:data];
[BlobKeyIO write:xattrsBlobKey to:data];
[IntegerIO writeUInt64:xattrsSize to:data];
[BlobKeyIO write:aclBlobKey to:data];
[IntegerIO writeInt32:uid to:data];
[IntegerIO writeInt32:gid to:data];
[IntegerIO writeInt32:mode to:data];
[IntegerIO writeInt64:mtime_sec to:data];
[IntegerIO writeInt64:mtime_nsec to:data];
[IntegerIO writeInt64:flags to:data];
[IntegerIO writeInt32:finderFlags to:data];
[IntegerIO writeInt32:extendedFinderFlags to:data];
[IntegerIO writeInt32:st_dev to:data];
[IntegerIO writeInt32:st_ino to:data];
[IntegerIO writeUInt32:st_nlink to:data];
[IntegerIO writeInt32:st_rdev to:data];
[IntegerIO writeInt64:ctime_sec to:data];
[IntegerIO writeInt64:ctime_nsec to:data];
[IntegerIO writeInt64:st_blocks to:data];
[IntegerIO writeUInt32:st_blksize to:data];
[IntegerIO writeInt64:createTime_sec to:data];
[IntegerIO writeInt64:createTime_nsec to:data];
[IntegerIO writeUInt32:(uint32_t)[missingNodes count] to:data];
for (NSString *missingNodeName in [missingNodes allKeys]) {
[StringIO write:missingNodeName to:data];
[[missingNodes objectForKey:missingNodeName] writeToData:data];
}
[IntegerIO writeUInt32:(uint32_t)[nodes count] to:data];
NSMutableArray *nodeNames = [NSMutableArray arrayWithArray:[nodes allKeys]];
[nodeNames sortUsingSelector:@selector(compare:)];
for (NSString *nodeName in nodeNames) {
[StringIO write:nodeName to:data];
Node *node = [nodes objectForKey:nodeName];
[node writeToData:data];
}
char header[TREE_HEADER_LENGTH + 1];
sprintf(header, "TreeV%03d", CURRENT_TREE_VERSION);
NSMutableData *completeData = [[[NSMutableData alloc] init] autorelease];
[completeData appendBytes:header length:TREE_HEADER_LENGTH];
[completeData appendBytes:[data bytes] length:[data length]];
return completeData;
}
- (BOOL)ctimeMatchesStat:(struct stat *)st {
return st->st_ctimespec.tv_sec == ctime_sec && st->st_ctimespec.tv_nsec == ctime_nsec;
}
- (uint64_t)aggregateUncompressedDataSize {
//FIXME: This doesn't include the size of the ACL.
uint64_t ret = xattrsSize;
for (Node *node in [nodes allValues]) {
ret += [node uncompressedDataSize];
}
return ret;
}
#pragma mark NSObject
- (BOOL)isEqual:(id)object {
if (![object isKindOfClass:[Tree class]]) {
return NO;
}
Tree *other = (Tree *)object;
BOOL ret = (treeVersion == [other treeVersion]
&& xattrsAreCompressed == [other xattrsAreCompressed]
&& [NSObject equalObjects:xattrsBlobKey and:[other xattrsBlobKey]]
&& xattrsSize == [other xattrsSize]
&& aclIsCompressed == [other aclIsCompressed]
&& [NSObject equalObjects:aclBlobKey and:[other aclBlobKey]]
&& uid == [other uid]
&& gid == [other gid]
&& mode == [other mode]
&& mtime_sec == [other mtime_sec]
&& mtime_nsec == [other mtime_nsec]
&& flags == [other flags]
&& finderFlags == [other finderFlags]
&& extendedFinderFlags == [other extendedFinderFlags]
&& st_dev == [other st_dev]
&& st_ino == [other st_ino]
&& st_nlink == [other st_nlink]
&& st_rdev == [other st_rdev]
&& ctime_sec == [other ctime_sec]
&& ctime_nsec == [other ctime_nsec]
&& createTime_sec == [other createTime_sec]
&& createTime_nsec == [other createTime_nsec]
&& [missingNodes isEqual:[other missingNodes]]
&& st_blocks == [other st_blocks]
&& st_blksize == [other st_blksize]
&& [nodes isEqual:[other nodes]]);
return ret;
}
- (NSUInteger)hash {
return (NSUInteger)treeVersion + [nodes hash];
}
@end
@implementation Tree (internal)
- (BOOL)readHeader:(BufferedInputStream *)is error:(NSError **)error {
BOOL ret = NO;
unsigned char *buf = (unsigned char *)malloc(TREE_HEADER_LENGTH);
if (![is readExactly:TREE_HEADER_LENGTH into:buf error:error]) {
goto readHeader_error;
}
NSString *header = [[[NSString alloc] initWithBytes:buf length:TREE_HEADER_LENGTH encoding:NSASCIIStringEncoding] autorelease];
if (![header hasPrefix:@"TreeV"] || [header length] < 6) {
SETNSERROR([Tree errorDomain], ERROR_INVALID_OBJECT_VERSION, @"invalid Tree header: %@", header);
goto readHeader_error;
}
treeVersion = [[header substringFromIndex:5] intValue];
if (treeVersion < 10) {
SETNSERROR([Tree errorDomain], ERROR_INVALID_OBJECT_VERSION, @"invalid Tree header: %@", header);
goto readHeader_error;
}
if (treeVersion == 13) {
SETNSERROR([Tree errorDomain], ERROR_INVALID_OBJECT_VERSION, @"invalid Tree version 13");
goto readHeader_error;
}
ret = YES;
readHeader_error:
free(buf);
return ret;
}
@end