Передайте пароль от echo к hdiutil с помощью target-c

Я изо всех сил пытаюсь выучить obj-C и могу использовать некоторую помощь. Я пишу "инструмент командной строки" для создания зашифрованного DMG, а затем безопасно удалить включенные файлы. Когда hdiutil создает DMG, он запрашивает пароль для шифрования, и я пытаюсь передать этот пароль из bin/echo в hdiutil.

DMG создается как положено, но когда я пытаюсь его смонтировать, пароль не принимается. Я попытался смонтировать как с пустым паролем, так и с дополнительным пробелом в конце.

Когда я NSLog значение из канала, он выглядит правильно, но это, вероятно, потому что я только что прочитал первые 4 символа. Я предполагаю, что к паролю добавлен какой-то дополнительный символ, но я не могу понять, почему и что.

Два вопроса 1. Как передать "правильное" значение в качестве пароля от NSTask passwordCmd к NSTask backupCmd?

2: Как мне NSLog точно такое же значение из канала, которое передается в [backupCmd setStandardInput:pipe]

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        NSTask *passwordCmd = [NSTask new];
        NSTask *backupCmd = [NSTask new];

        NSPipe *pipe;
        pipe = [NSPipe pipe];

        // Enter password by calling echo with a NStask
        [passwordCmd setLaunchPath:@"/bin/echo"];
        [passwordCmd setStandardOutput:pipe]; // write to pipe
        [passwordCmd setArguments: [NSArray arrayWithObjects: @"test", nil]];
        [passwordCmd launch];
        [passwordCmd waitUntilExit];

        // Log the value of the pipe for debugging 
        NSData *output = [[pipe fileHandleForReading] readDataOfLength:4];
        NSString *string = [[NSString alloc] initWithData:output encoding:NSUTF8StringEncoding];
        NSLog(@"'%@'", string);

        // Create a encrypted DMG based on a folder
        [backupCmd setLaunchPath:@"/usr/bin/hdiutil"];
        [backupCmd setCurrentDirectoryPath:@"/Volumes/Macintosh HD/Users/kalle/Desktop/test/"];
        [backupCmd setArguments:[NSArray arrayWithObjects:@"create",@"-format",@"UDZO",@"-srcfolder",@"backup",@"/Volumes/Macintosh HD/Users/kalle/Desktop/backup.dmg",@"-encryption",@"AES-256",@"-stdinpass",@"-quiet",nil]];
        [backupCmd setStandardInput:pipe]; // read from pipe
        [backupCmd launch];
        [backupCmd waitUntilExit];

        // Do some more stuff...

    }
    return 0;
}

Любая помощь высоко ценится!

2 ответа

Решение

Я вижу две проблемы в вашем коде:

1) Документация "hdiutil" гласит:

-stdinpass
читать терминированную фразу-пароль со стандартного ввода.... Помните, что пароль будет содержать любые новые строки до NULL.

Но "/bin/echo" всегда добавляет новую строку к выводу. Таким образом, ваш пароль установлен на "test \ n".

2) Если вы прочитали пароль из канала для записи в журнал, данные "ушли" и больше не будут считываться задачей резервного копирования. (РЕДАКТИРОВАТЬ: Это было также опубликовано Рами Аль Zuhoury, когда я писал этот ответ!)

Я бы не использовал задачу "/bin/echo" для передачи пароля в задачу резервного копирования. Вы можете лучше записать необходимые данные прямо в канал:

NSString *passwd = @"test\0\n"; // password + NULL character + newline
NSData *passwdData = [passwd dataUsingEncoding:NSUTF8StringEncoding];
[[pipe fileHandleForWriting] writeData:passwdData];
[[pipe fileHandleForWriting] closeFile];

(Я не уверен, действительно ли "hdiutil" ожидает символ новой строки после символа NULL. Вы также можете попробовать его без символа новой строки.)

Вы уже прочитали символы из конвейера, которые так "поглощены", и в следующий раз, когда вы прочитаете их, вы больше их не найдете.

Вы должны быть в состоянии сделать это, если вы удалите эту строку:

NSData *output = [[pipe fileHandleForReading] readDataOfLength:4];

Делаем так, чтобы 4 символа еще оставались доступными в трубе.

Другие вопросы по тегам