mirror of
https://github.com/samsonjs/arq_restore.git
synced 2026-03-25 09:25:53 +00:00
263 lines
9.6 KiB
Objective-C
263 lines
9.6 KiB
Objective-C
//
|
|
// Tree.m
|
|
// Backup
|
|
//
|
|
// Created by Stefan Reitshamer on 3/25/09.
|
|
// Copyright 2009 PhotoMinds LLC. All rights reserved.
|
|
//
|
|
|
|
#import "StringIO.h"
|
|
#import "IntegerIO.h"
|
|
#import "BooleanIO.h"
|
|
#import "Node.h"
|
|
#import "Tree.h"
|
|
#import "Blob.h"
|
|
#import "DataInputStream.h"
|
|
#import "SetNSError.h"
|
|
#import "RegexKitLite.h"
|
|
#import "NSErrorCodes.h"
|
|
#import "BufferedInputStream.h"
|
|
#import "NSData-Gzip.h"
|
|
#import "GunzipInputStream.h"
|
|
#import "BlobKey.h"
|
|
#import "NSObject_extra.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;
|
|
@synthesize aggregateSizeOnDisk;
|
|
|
|
+ (NSString *)errorDomain {
|
|
return @"TreeErrorDomain";
|
|
}
|
|
- (id)init {
|
|
if (self = [super init]) {
|
|
nodes = [[NSMutableDictionary alloc] init];
|
|
}
|
|
return self;
|
|
}
|
|
- (id)initWithBufferedInputStream:(BufferedInputStream *)is error:(NSError **)error {
|
|
if (self = [super 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;
|
|
}
|
|
}
|
|
|
|
NSString *xattrsSHA1 = nil;
|
|
BOOL xattrsStretchedKey = NO;
|
|
NSString *aclSHA1 = nil;
|
|
BOOL aclStretchedKey = NO;
|
|
BOOL ret = [StringIO read:&xattrsSHA1 from:is error:error]
|
|
&& (treeVersion < 14 || [BooleanIO read:&xattrsStretchedKey from:is error:error])
|
|
&& [IntegerIO readUInt64:&xattrsSize from:is error:error]
|
|
&&[StringIO read:&aclSHA1 from:is error:error]
|
|
&& (treeVersion < 14 || [BooleanIO read:&aclStretchedKey from:is 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];
|
|
if (!ret) {
|
|
goto initError;
|
|
}
|
|
if (xattrsSHA1 != nil) {
|
|
xattrsBlobKey = [[BlobKey alloc] initWithSHA1:xattrsSHA1 stretchEncryptionKey:xattrsStretchedKey];
|
|
}
|
|
if (aclSHA1 != nil) {
|
|
aclBlobKey = [[BlobKey alloc] initWithSHA1:aclSHA1 stretchEncryptionKey:aclStretchedKey];
|
|
}
|
|
|
|
if (treeVersion >= 11) {
|
|
if (![IntegerIO readUInt64:&aggregateSizeOnDisk 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;
|
|
}
|
|
}
|
|
|
|
unsigned int nodeCount;
|
|
if (![IntegerIO readUInt32:&nodeCount from:is error:error]) {
|
|
goto initError;
|
|
}
|
|
nodes = [[NSMutableDictionary alloc] init];
|
|
for (unsigned int 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];
|
|
[super dealloc];
|
|
}
|
|
- (NSArray *)childNodeNames {
|
|
return [nodes allKeys];
|
|
}
|
|
- (Node *)childNodeWithName:(NSString *)name {
|
|
return [nodes objectForKey:name];
|
|
}
|
|
- (BOOL)containsNodeNamed:(NSString *)name {
|
|
return [nodes objectForKey:name] != nil;
|
|
}
|
|
- (NSDictionary *)nodes {
|
|
return nodes;
|
|
}
|
|
- (Blob *)toBlob {
|
|
NSMutableData *data = [[NSMutableData alloc] init];
|
|
[BooleanIO write:xattrsAreCompressed to:data];
|
|
[BooleanIO write:aclIsCompressed to:data];
|
|
[StringIO write:[xattrsBlobKey sha1] to:data];
|
|
[BooleanIO write:[xattrsBlobKey stretchEncryptionKey] to:data];
|
|
[IntegerIO writeUInt64:xattrsSize to:data];
|
|
[StringIO write:[aclBlobKey sha1] to:data];
|
|
[BooleanIO write:[aclBlobKey stretchEncryptionKey] 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 writeUInt64:aggregateSizeOnDisk to: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];
|
|
[completeData appendBytes:header length:TREE_HEADER_LENGTH];
|
|
|
|
[completeData appendBytes:[data bytes] length:[data length]];
|
|
|
|
Blob *ret =[[[Blob alloc] initWithData:completeData mimeType:@"binary/octet-stream" downloadName:@"Tree" dataDescription:@"tree"] autorelease];
|
|
[completeData release];
|
|
[data release];
|
|
return ret;
|
|
}
|
|
|
|
#pragma mark NSObject
|
|
- (BOOL)isEqual:(id)object {
|
|
if (![object isKindOfClass:[Tree class]]) {
|
|
return NO;
|
|
}
|
|
Tree *other = (Tree *)object;
|
|
return 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]
|
|
&& st_blocks == [other st_blocks]
|
|
&& st_blksize == [other st_blksize]
|
|
&& aggregateSizeOnDisk == [other aggregateSizeOnDisk]
|
|
&& [nodes isEqual:[other nodes]];
|
|
}
|
|
- (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];
|
|
NSRange versionRange = [header rangeOfRegex:@"^TreeV(\\d{3})$" capture:1];
|
|
treeVersion = 0;
|
|
if (versionRange.location != NSNotFound) {
|
|
NSNumberFormatter *nf = [[NSNumberFormatter alloc] init];
|
|
NSNumber *number = [nf numberFromString:[header substringWithRange:versionRange]];
|
|
treeVersion = [number intValue];
|
|
[nf release];
|
|
}
|
|
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
|