mirror of
https://github.com/samsonjs/arq_restore.git
synced 2026-03-25 09:25:53 +00:00
316 lines
11 KiB
Objective-C
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
|