Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
216 changes: 216 additions & 0 deletions GitUpKit/Core/GCCommitSigning-Tests.m
Original file line number Diff line number Diff line change
@@ -0,0 +1,216 @@
// Copyright (C) 2015-2019 Pierre-Olivier Latour <info@pol-online.net>
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.

#if !__has_feature(objc_arc)
#error This file requires ARC
#endif

#import "GCTestCase.h"
#import "GCRepository+Index.h"

static NSString* _CommitSignature(GCCommit* commit) {
git_buf buffer = {0};
int status = git_commit_header_field(&buffer, commit.private, "gpgsig");
if (status != GIT_OK) {
git_buf_free(&buffer);
return nil;
}
NSString* signature = [[NSString alloc] initWithBytes:buffer.ptr length:buffer.size encoding:NSUTF8StringEncoding];
git_buf_free(&buffer);
return signature;
}

static BOOL _WriteLocalConfigOption(GCRepository* repository, NSString* variable, NSString* value) {
return [repository writeConfigOptionForLevel:kGCConfigLevel_Local variable:variable withValue:value error:NULL];
}

static BOOL _CommitHasSSHSignature(GCCommit* commit) {
return [_CommitSignature(commit) containsString:@"BEGIN SSH SIGNATURE"];
}

static BOOL _ConfigureSSHSigningWithKeyPath(GCRepository* repository, NSString* keyPath) {
if (!_WriteLocalConfigOption(repository, @"commit.gpgsign", @"true")) {
return NO;
}
if (!_WriteLocalConfigOption(repository, @"gpg.format", @"ssh")) {
return NO;
}
return _WriteLocalConfigOption(repository, @"user.signingkey", keyPath);
}

static NSString* _CreateFakeSSHSigner(NSString* directory, int exitStatus) {
NSString* path = [directory stringByAppendingPathComponent:[[NSProcessInfo processInfo] globallyUniqueString]];
NSString* contents;

if (exitStatus == 0) {
NSArray* lines = @[
@"#!/bin/sh",
@"cat >/dev/null",
@"printf '%s\\n' '-----BEGIN SSH SIGNATURE-----' 'fake-signature' '-----END SSH SIGNATURE-----'",
@""
];
contents = [lines componentsJoinedByString:@"\n"];
} else {
NSArray* lines = @[
@"#!/bin/sh",
@"cat >/dev/null",
@"echo signer failed >&2",
[NSString stringWithFormat:@"exit %i", exitStatus],
@""
];
contents = [lines componentsJoinedByString:@"\n"];
}

if (![contents writeToFile:path atomically:YES encoding:NSUTF8StringEncoding error:NULL]) {
return nil;
}
NSDictionary* attributes = @{NSFilePosixPermissions : @(0755)};
if (![[NSFileManager defaultManager] setAttributes:attributes ofItemAtPath:path error:NULL]) {
return nil;
}
return path;
}

static GCCommit* _CreateCommitFromRepositoryIndex(GCRepository* repository, NSString* message, NSError** error) {
GCCommit* commit = nil;
git_index* index = NULL;
git_tree* tree = NULL;

index = [repository reloadRepositoryIndex:error];
if (!index) {
goto cleanup;
}

git_oid oid;
CALL_LIBGIT2_FUNCTION_GOTO(cleanup, git_index_write_tree_to, &oid, index, repository.private);
CALL_LIBGIT2_FUNCTION_GOTO(cleanup, git_tree_lookup, &tree, repository.private, &oid);
commit = GCCreateCommitFromTreeWithOptionalSignature(repository,
tree,
NULL,
0,
NULL,
message,
error);

cleanup:
git_tree_free(tree);
git_index_free(index);
return commit;
}

@implementation GCEmptyRepositoryTests (GCCommitSigning)

- (void)testCommitSigningLeavesCommitsUnsignedWhenDisabledOrUnsupported {
[self updateFileAtPath:@"unsigned.txt" withString:@"unsigned\n"];
XCTAssertTrue([self.repository addFileToIndex:@"unsigned.txt" error:NULL]);

GCCommit* unsignedCommit = _CreateCommitFromRepositoryIndex(self.repository, @"Unsigned", NULL);
XCTAssertNotNil(unsignedCommit);
XCTAssertNil(_CommitSignature(unsignedCommit));

XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"commit.gpgsign", @"true"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.format", @"openpgp"));
[self updateFileAtPath:@"openpgp.txt" withString:@"openpgp\n"];
XCTAssertTrue([self.repository addFileToIndex:@"openpgp.txt" error:NULL]);

GCCommit* openPGPCommit = _CreateCommitFromRepositoryIndex(self.repository, @"OpenPGP config remains unsigned", NULL);
XCTAssertNotNil(openPGPCommit);
XCTAssertNil(_CommitSignature(openPGPCommit));
}

- (void)testCommitSigningRequiresSSHKey {
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"commit.gpgsign", @"true"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.format", @"ssh"));
[self updateFileAtPath:@"missing-key.txt" withString:@"missing key\n"];
XCTAssertTrue([self.repository addFileToIndex:@"missing-key.txt" error:NULL]);

NSError* error;
XCTAssertNil(_CreateCommitFromRepositoryIndex(self.repository, @"Missing key", &error));
XCTAssertTrue([error.localizedDescription containsString:@"user.signingkey"]);
}

- (void)testCommitSigningSupportsInlineKeyAndDefaultKeyCommand {
NSString* signer = _CreateFakeSSHSigner(self.temporaryPath, 0);
NSString* inlineKey = @"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIFakeInlineKey test@example.com";
NSString* defaultKeyCommand = @"printf 'key::ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIDefaultCommandKey test@example.com\\n'";

XCTAssertNotNil(signer);
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"commit.gpgsign", @"true"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.format", @"ssh"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.ssh.program", signer));

XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"user.signingkey", inlineKey));
[self updateFileAtPath:@"inline-key.txt" withString:@"inline key\n"];
XCTAssertTrue([self.repository addFileToIndex:@"inline-key.txt" error:NULL]);

GCCommit* inlineCommit = _CreateCommitFromRepositoryIndex(self.repository, @"Inline key", NULL);
XCTAssertNotNil(inlineCommit);
XCTAssertTrue(_CommitHasSSHSignature(inlineCommit));

XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"user.signingkey", nil));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.ssh.defaultKeyCommand", defaultKeyCommand));
[self updateFileAtPath:@"default-key-command.txt" withString:@"default key command\n"];
XCTAssertTrue([self.repository addFileToIndex:@"default-key-command.txt" error:NULL]);

GCCommit* defaultCommandCommit = _CreateCommitFromRepositoryIndex(self.repository, @"Default key command", NULL);
XCTAssertNotNil(defaultCommandCommit);
XCTAssertTrue(_CommitHasSSHSignature(defaultCommandCommit));
}

- (void)testCommitSigningSupportsKeyPath {
NSString* keyPath = [self.temporaryPath stringByAppendingPathComponent:@"signing_key"];
GCTask* keygen = [[GCTask alloc] initWithExecutablePath:@"/usr/bin/ssh-keygen"];
int status;
NSArray* keygenArguments = @[ @"-t", @"ed25519", @"-f", keyPath, @"-N", @"", @"-q" ];
BOOL keygenSuccess = [keygen runWithArguments:keygenArguments stdin:nil stdout:NULL stderr:NULL exitStatus:&status error:NULL];
XCTAssertTrue(keygenSuccess);
XCTAssertEqual(status, 0);
XCTAssertTrue(_ConfigureSSHSigningWithKeyPath(self.repository, keyPath));

[self updateFileAtPath:@"path-key.txt" withString:@"path key\n"];
XCTAssertTrue([self.repository addFileToIndex:@"path-key.txt" error:NULL]);

GCCommit* commit = _CreateCommitFromRepositoryIndex(self.repository, @"Path key", NULL);
XCTAssertNotNil(commit);
XCTAssertTrue(_CommitHasSSHSignature(commit));

NSString* publicKey = [NSString stringWithContentsOfFile:[keyPath stringByAppendingString:@".pub"] encoding:NSUTF8StringEncoding error:NULL];
NSString* allowedSignersPath = [self.temporaryPath stringByAppendingPathComponent:@"allowed_signers"];
NSString* allowedSigners = [NSString stringWithFormat:@"bot@example.com %@", publicKey];
XCTAssertTrue([allowedSigners writeToFile:allowedSignersPath atomically:YES encoding:NSUTF8StringEncoding error:NULL]);

NSString* allowedSignersConfig = [NSString stringWithFormat:@"gpg.ssh.allowedSignersFile=%@", allowedSignersPath];
NSString* verifyOutput = [self runGitCLTWithRepository:self.repository command:@"-c", allowedSignersConfig, @"verify-commit", commit.SHA1, nil];
XCTAssertNotNil(verifyOutput);
}

- (void)testCommitSigningFailsOnSignerFailure {
NSString* signer = _CreateFakeSSHSigner(self.temporaryPath, 7);
NSString* signingKey = @"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIFailingSignerKey test@example.com";

XCTAssertNotNil(signer);
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"commit.gpgsign", @"true"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.format", @"ssh"));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"gpg.ssh.program", signer));
XCTAssertTrue(_WriteLocalConfigOption(self.repository, @"user.signingkey", signingKey));
[self updateFileAtPath:@"failing-signer.txt" withString:@"failing signer\n"];
XCTAssertTrue([self.repository addFileToIndex:@"failing-signer.txt" error:NULL]);

NSError* error;
XCTAssertNil(_CreateCommitFromRepositoryIndex(self.repository, @"Failing signer", &error));
XCTAssertTrue([error.localizedDescription containsString:@"non-zero status"]);
}

@end
Loading
Loading