feat(commit): add ability to create commit with different number of parents

This commit is contained in:
Aleksey Kulikov 2021-08-30 16:45:37 +03:00
parent c90561ed8f
commit 94b4116adf
15 changed files with 217 additions and 260 deletions

View file

@ -58,19 +58,19 @@ Pointer<git_oid> create(
final messageEncodingC = final messageEncodingC =
messageEncoding?.toNativeUtf8().cast<Int8>() ?? nullptr; messageEncoding?.toNativeUtf8().cast<Int8>() ?? nullptr;
final messageC = message.toNativeUtf8().cast<Int8>(); final messageC = message.toNativeUtf8().cast<Int8>();
late final Pointer<Pointer<git_commit>> parentsC; Pointer<Pointer<git_commit>> parentsC =
calloc.call<Pointer<git_commit>>(parentCount);
if (parents.isNotEmpty) { if (parents.isNotEmpty) {
parentsC = calloc(parentCount);
for (var i = 0; i < parentCount; i++) { for (var i = 0; i < parentCount; i++) {
final oid = oid_bindings.fromSHA(parents[i]); final oid = oid_bindings.fromSHA(parents[i]);
final commit = lookup(repo, oid); var commit = calloc<IntPtr>();
parentsC[i] = commit; commit = lookup(repo, oid).cast();
parentsC[i] = commit.cast();
} }
} else { } else {
throw UnimplementedError( final commit = calloc<IntPtr>();
'Writing commit without parents is not implemented'); parentsC[0] = commit.cast();
} }
final error = libgit2.git_commit_create( final error = libgit2.git_commit_create(
@ -89,6 +89,9 @@ Pointer<git_oid> create(
calloc.free(updateRefC); calloc.free(updateRefC);
calloc.free(messageEncodingC); calloc.free(messageEncodingC);
calloc.free(messageC); calloc.free(messageC);
for (var i = 0; i < parentCount; i++) {
free(parentsC[i]);
}
calloc.free(parentsC); calloc.free(parentsC);
if (error < 0) { if (error < 0) {
@ -134,7 +137,7 @@ int parentCount(Pointer<git_commit> commit) =>
Pointer<git_oid> parentId(Pointer<git_commit> commit, int n) { Pointer<git_oid> parentId(Pointer<git_commit> commit, int n) {
final parentOid = libgit2.git_commit_parent_id(commit, n); final parentOid = libgit2.git_commit_parent_id(commit, n);
if (parentOid is int && parentOid as int < 0) { if (parentOid == nullptr) {
throw LibGit2Error(libgit2.git_error_last()); throw LibGit2Error(libgit2.git_error_last());
} else { } else {
return parentOid; return parentOid;
@ -163,8 +166,5 @@ Pointer<git_oid> tree(Pointer<git_commit> commit) {
Pointer<git_repository> owner(Pointer<git_commit> commit) => Pointer<git_repository> owner(Pointer<git_commit> commit) =>
libgit2.git_commit_owner(commit); libgit2.git_commit_owner(commit);
/// Close an open commit. /// Close an open commit to release memory.
///
/// IMPORTANT: It is necessary to call this method when you stop using a commit.
/// Failure to do so will cause a memory leak.
void free(Pointer<git_commit> commit) => libgit2.git_commit_free(commit); void free(Pointer<git_commit> commit) => libgit2.git_commit_free(commit);

View file

@ -38,6 +38,26 @@ Pointer<git_oid> fromSHA(String hex) {
} }
} }
/// Copy an already raw oid into a git_oid structure.
///
/// Throws a [LibGit2Error] if error occured.
Pointer<git_oid> fromRaw(Array<Uint8> raw) {
final out = calloc<git_oid>();
var rawC = calloc<Uint8>(20);
for (var i = 0; i < 20; i++) {
rawC[i] = raw[i];
}
final error = libgit2.git_oid_fromraw(out, rawC);
calloc.free(rawC);
if (error < 0) {
throw LibGit2Error(libgit2.git_error_last());
} else {
return out;
}
}
/// Format a git_oid into a hex string. /// Format a git_oid into a hex string.
/// ///
/// Throws a [LibGit2Error] if error occured. /// Throws a [LibGit2Error] if error occured.

View file

@ -1,7 +1,8 @@
import 'dart:ffi'; import 'dart:ffi';
import 'bindings/libgit2_bindings.dart'; import 'bindings/libgit2_bindings.dart';
import 'bindings/commit.dart' as bindings; import 'bindings/commit.dart' as bindings;
import 'bindings/oid.dart' as oid_bindings;
import 'bindings/tree.dart' as tree_bindings;
import 'repository.dart'; import 'repository.dart';
import 'oid.dart'; import 'oid.dart';
import 'signature.dart'; import 'signature.dart';
@ -38,30 +39,13 @@ class Commit {
required Signature author, required Signature author,
required Signature commiter, required Signature commiter,
required String treeSHA, required String treeSHA,
required List<String> parentsSHA, required List<String> parents,
String? updateRef, String? updateRef,
String? messageEncoding, String? messageEncoding,
}) { }) {
libgit2.git_libgit2_init(); final treeOid = oid_bindings.fromStrN(treeSHA);
final tree =
final parentCount = parentsSHA.length; Tree(tree_bindings.lookupPrefix(repo.pointer, treeOid, treeSHA.length));
late final Tree tree;
if (treeSHA.length == 40) {
final treeOid = Oid.fromSHA(treeSHA);
tree = Tree.lookup(
repo.pointer,
treeOid.pointer,
);
} else {
final odb = repo.odb;
final treeOid = Oid.fromShortSHA(treeSHA, odb);
tree = Tree.lookup(
repo.pointer,
treeOid.pointer,
);
odb.free();
}
final result = Oid(bindings.create( final result = Oid(bindings.create(
repo.pointer, repo.pointer,
@ -71,12 +55,11 @@ class Commit {
messageEncoding, messageEncoding,
message, message,
tree.pointer, tree.pointer,
parentCount, parents.length,
parentsSHA, parents,
)); ));
tree.free(); tree.free();
libgit2.git_libgit2_init();
return result; return result;
} }
@ -105,18 +88,13 @@ class Commit {
/// Returns list of parent commits. /// Returns list of parent commits.
/// ///
/// Throws a [LibGit2Error] if error occured. /// Throws a [LibGit2Error] if error occured.
List<Commit> get parents { List<Oid> get parents {
var parents = <Commit>[]; var parents = <Oid>[];
final parentCount = bindings.parentCount(_commitPointer); final parentCount = bindings.parentCount(_commitPointer);
for (var i = 0; i < parentCount; i++) { for (var i = 0; i < parentCount; i++) {
final parentOid = bindings.parentId(_commitPointer, i); final parentOid = bindings.parentId(_commitPointer, i);
parents.add(Oid(parentOid));
if (parentOid != nullptr) {
final owner = bindings.owner(_commitPointer);
final commit = bindings.lookup(owner, parentOid);
parents.add(Commit(commit));
}
} }
return parents; return parents;
@ -128,6 +106,5 @@ class Commit {
/// Releases memory allocated for commit object. /// Releases memory allocated for commit object.
void free() { void free() {
bindings.free(_commitPointer); bindings.free(_commitPointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -142,6 +142,5 @@ class Config {
/// Releases memory allocated for config object. /// Releases memory allocated for config object.
void free() { void free() {
bindings.free(_configPointer); bindings.free(_configPointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -3,10 +3,9 @@ import 'package:ffi/ffi.dart';
import 'package:libgit2dart/src/tree.dart'; import 'package:libgit2dart/src/tree.dart';
import 'bindings/libgit2_bindings.dart'; import 'bindings/libgit2_bindings.dart';
import 'bindings/index.dart' as bindings; import 'bindings/index.dart' as bindings;
import 'bindings/repository.dart' as repo_bindings;
import 'odb.dart';
import 'oid.dart'; import 'oid.dart';
import 'enums.dart'; import 'enums.dart';
import 'repository.dart';
import 'util.dart'; import 'util.dart';
class Index { class Index {
@ -101,15 +100,9 @@ class Index {
} else if (target is Tree) { } else if (target is Tree) {
tree = target; tree = target;
} else if (isValidShaHex(target as String)) { } else if (isValidShaHex(target as String)) {
if (target.length == 40) { final repo = Repository(bindings.owner(_indexPointer));
oid = Oid.fromSHA(target); oid = Oid.fromSHA(repo, target);
tree = Tree.lookup(bindings.owner(_indexPointer), oid.pointer); tree = Tree.lookup(repo.pointer, oid.pointer);
} else {
final odb = Odb(repo_bindings.odb(bindings.owner(_indexPointer)));
oid = Oid.fromShortSHA(target, odb);
odb.free();
tree = Tree.lookup(bindings.owner(_indexPointer), oid.pointer);
}
} else { } else {
throw ArgumentError.value( throw ArgumentError.value(
'$target should be either Oid object, SHA hex string or Tree object'); '$target should be either Oid object, SHA hex string or Tree object');
@ -146,7 +139,6 @@ class Index {
/// Releases memory allocated for index object. /// Releases memory allocated for index object.
void free() { void free() {
bindings.free(_indexPointer); bindings.free(_indexPointer);
libgit2.git_libgit2_shutdown();
} }
} }
@ -158,7 +150,7 @@ class IndexEntry {
late final Pointer<git_index_entry> _indexEntryPointer; late final Pointer<git_index_entry> _indexEntryPointer;
/// Unique identity of the index entry. /// Unique identity of the index entry.
Oid get id => Oid.fromSHA(sha); Oid get id => Oid.fromRaw(_indexEntryPointer.ref.id);
set id(Oid oid) => _indexEntryPointer.ref.id = oid.pointer.ref; set id(Oid oid) => _indexEntryPointer.ref.id = oid.pointer.ref;

View file

@ -10,9 +10,11 @@ class Odb {
libgit2.git_libgit2_init(); libgit2.git_libgit2_init();
} }
/// Pointer to memory address for allocated oid object.
late final Pointer<git_odb> _odbPointer; late final Pointer<git_odb> _odbPointer;
/// Pointer to memory address for allocated oid object.
Pointer<git_odb> get pointer => _odbPointer;
/// Determine if an object can be found in the object database by an abbreviated object ID. /// Determine if an object can be found in the object database by an abbreviated object ID.
/// ///
/// Throws a [LibGit2Error] if error occured. /// Throws a [LibGit2Error] if error occured.
@ -26,6 +28,5 @@ class Odb {
/// Releases memory allocated for odb object. /// Releases memory allocated for odb object.
void free() { void free() {
bindings.free(_odbPointer); bindings.free(_odbPointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -1,7 +1,7 @@
import 'dart:ffi'; import 'dart:ffi';
import 'bindings/libgit2_bindings.dart'; import 'bindings/libgit2_bindings.dart';
import 'bindings/oid.dart' as bindings; import 'bindings/oid.dart' as bindings;
import 'odb.dart'; import 'repository.dart';
import 'util.dart'; import 'util.dart';
class Oid { class Oid {
@ -11,24 +11,26 @@ class Oid {
libgit2.git_libgit2_init(); libgit2.git_libgit2_init();
} }
/// Initializes a new instance of [Oid] class from provided /// Initializes a new instance of [Oid] class by determining if an object can be found
/// hexadecimal [sha] string. /// in the ODB of [repository] with provided hexadecimal [sha] string that is 40 characters
/// long or shorter.
/// ///
/// Throws a [LibGit2Error] if error occured. /// Throws a [LibGit2Error] if error occured.
Oid.fromSHA(String sha) { Oid.fromSHA(Repository repository, String sha) {
libgit2.git_libgit2_init(); libgit2.git_libgit2_init();
if (sha.length == 40) {
_oidPointer = bindings.fromSHA(sha); _oidPointer = bindings.fromSHA(sha);
} else {
final odb = repository.odb;
_oidPointer = odb.existsPrefix(bindings.fromStrN(sha), sha.length);
odb.free();
}
} }
/// Initializes a new instance of [Oid] class by determining if an object can be found /// Initializes a new instance of [Oid] class from provided raw git_oid.
/// in the object database of repository with provided hexadecimal [sha] string that is Oid.fromRaw(git_oid raw) {
/// lesser than 40 characters long and [Odb] object.
///
/// Throws a [LibGit2Error] if error occured.
Oid.fromShortSHA(String sha, Odb odb) {
libgit2.git_libgit2_init(); libgit2.git_libgit2_init();
_oidPointer = odb.existsPrefix(bindings.fromStrN(sha), sha.length); _oidPointer = bindings.fromRaw(raw.id);
} }
late final Pointer<git_oid> _oidPointer; late final Pointer<git_oid> _oidPointer;

View file

@ -1,11 +1,10 @@
import 'dart:ffi'; import 'dart:ffi';
import 'bindings/libgit2_bindings.dart'; import 'bindings/libgit2_bindings.dart';
import 'bindings/reference.dart' as bindings; import 'bindings/reference.dart' as bindings;
import 'bindings/repository.dart' as repo_bindings;
import 'odb.dart';
import 'oid.dart'; import 'oid.dart';
import 'reflog.dart'; import 'reflog.dart';
import 'enums.dart'; import 'enums.dart';
import 'repository.dart';
import 'util.dart'; import 'util.dart';
class References { class References {
@ -139,13 +138,8 @@ class Reference {
late final Oid oid; late final Oid oid;
if (isValidShaHex(target)) { if (isValidShaHex(target)) {
if (target.length == 40) { final repo = Repository(bindings.owner(_refPointer));
oid = Oid.fromSHA(target); oid = Oid.fromSHA(repo, target);
} else {
final odb = Odb(repo_bindings.odb(owner));
oid = Oid.fromShortSHA(target, odb);
odb.free();
}
} else { } else {
final ref = Reference( final ref = Reference(
_repoPointer, _repoPointer,
@ -233,6 +227,5 @@ class Reference {
/// Releases memory allocated for reference object. /// Releases memory allocated for reference object.
void free() { void free() {
bindings.free(_refPointer); bindings.free(_refPointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -45,7 +45,6 @@ class RefLog {
/// Releases memory allocated for reflog object. /// Releases memory allocated for reflog object.
void free() { void free() {
bindings.free(_reflogPointer); bindings.free(_reflogPointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -10,6 +10,14 @@ import 'bindings/repository.dart' as bindings;
import 'util.dart'; import 'util.dart';
class Repository { class Repository {
/// Initializes a new instance of the [Repository] class from provided
/// pointer to repository object in memory.
///
/// Should be freed with `free()` to release allocated memory.
Repository(this._repoPointer) {
libgit2.git_libgit2_init();
}
/// Initializes a new instance of the [Repository] class by creating a new /// Initializes a new instance of the [Repository] class by creating a new
/// Git repository in the given folder. /// Git repository in the given folder.
/// ///
@ -119,13 +127,7 @@ class Repository {
late final Oid oid; late final Oid oid;
if (isValidShaHex(target)) { if (isValidShaHex(target)) {
if (target.length == 40) { oid = Oid.fromSHA(this, target);
oid = Oid.fromSHA(target);
} else {
final odb = this.odb;
oid = Oid.fromShortSHA(target, odb);
odb.free();
}
bindings.setHeadDetached(_repoPointer, oid.pointer); bindings.setHeadDetached(_repoPointer, oid.pointer);
} else { } else {
bindings.setHead(_repoPointer, target); bindings.setHead(_repoPointer, target);
@ -208,7 +210,6 @@ class Repository {
/// Releases memory allocated for repository object. /// Releases memory allocated for repository object.
void free() { void free() {
bindings.free(_repoPointer); bindings.free(_repoPointer);
libgit2.git_libgit2_shutdown();
} }
/// Returns the configuration file for this repository. /// Returns the configuration file for this repository.
@ -269,7 +270,7 @@ class Repository {
oid = target as Oid; oid = target as Oid;
isDirect = true; isDirect = true;
} else if (isValidShaHex(target as String)) { } else if (isValidShaHex(target as String)) {
oid = _getOid(target); oid = Oid.fromSHA(this, target);
isDirect = true; isDirect = true;
} else { } else {
isDirect = false; isDirect = false;
@ -313,24 +314,10 @@ class Repository {
late final Oid oid; late final Oid oid;
if (isValidShaHex(sha)) { if (isValidShaHex(sha)) {
oid = _getOid(sha); oid = Oid.fromSHA(this, sha);
} else { } else {
throw ArgumentError.value('$sha is not a valid sha hex string'); throw ArgumentError.value('$sha is not a valid sha hex string');
} }
return Commit.lookup(_repoPointer, oid.pointer); return Commit.lookup(_repoPointer, oid.pointer);
} }
/// Returns [Oid] for provided sha hex that is 40 characters long or less.
Oid _getOid(String sha) {
late final Oid oid;
if (sha.length == 40) {
oid = Oid.fromSHA(sha);
} else {
final odb = this.odb;
oid = Oid.fromShortSHA(sha, odb);
odb.free();
}
return oid;
}
} }

View file

@ -68,6 +68,5 @@ class Signature {
/// Releases memory allocated for signature object. /// Releases memory allocated for signature object.
void free() { void free() {
bindings.free(_signaturePointer); bindings.free(_signaturePointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -27,6 +27,5 @@ class Tree {
/// Releases memory allocated for tree object. /// Releases memory allocated for tree object.
void free() { void free() {
bindings.free(_treePointer); bindings.free(_treePointer);
libgit2.git_libgit2_shutdown();
} }
} }

View file

@ -11,6 +11,11 @@ void main() {
late Repository repo; late Repository repo;
final tmpDir = '${Directory.systemTemp.path}/commit_testrepo/'; final tmpDir = '${Directory.systemTemp.path}/commit_testrepo/';
const message = "Commit message.\n\nSome description.\n";
const tree = '7796359a96eb722939c24bafdb1afe9f07f2f628';
late Signature author;
late Signature commiter;
setUp(() async { setUp(() async {
if (await Directory(tmpDir).exists()) { if (await Directory(tmpDir).exists()) {
await Directory(tmpDir).delete(recursive: true); await Directory(tmpDir).delete(recursive: true);
@ -20,152 +25,135 @@ void main() {
to: await Directory(tmpDir).create(), to: await Directory(tmpDir).create(),
); );
repo = Repository.open(tmpDir); repo = Repository.open(tmpDir);
author = Signature.create(
name: 'Author Name',
email: 'author@email.com',
time: 123,
);
commiter = Signature.create(
name: 'Commiter',
email: 'commiter@email.com',
time: 124,
);
}); });
tearDown(() async { tearDown(() async {
author.free();
commiter.free();
repo.free(); repo.free();
await Directory(tmpDir).delete(recursive: true); await Directory(tmpDir).delete(recursive: true);
}); });
group('lookup', () { test('successfully returns when 40 char sha hex is provided', () {
test('successful when 40 char sha hex is provided', () { final commit = repo[mergeCommit];
final commit = repo[mergeCommit]; expect(commit, isA<Commit>());
expect(commit, isA<Commit>()); commit.free();
commit.free();
});
test('successful when sha hex is short', () {
final commit = repo[mergeCommit.substring(0, 5)];
expect(commit, isA<Commit>());
commit.free();
});
test('throws when provided sha hex is invalid', () {
expect(() => repo['invalid'], throwsA(isA<ArgumentError>()));
});
test('throws when nothing found', () {
expect(() => repo['970ae5c'], throwsA(isA<LibGit2Error>()));
});
test('returns with correct fields', () {
final signature = Signature.create(
name: 'Aleksey Kulikov',
email: 'skinny.mind@gmail.com',
time: 1626091184,
offset: 180,
);
final commit = repo[mergeCommit];
final parents = commit.parents;
expect(commit.messageEncoding, 'utf-8');
expect(commit.message, 'Merge branch \'feature\'\n');
expect(commit.id.sha, mergeCommit);
expect(parents.length, 2);
expect(
parents[0].id.sha,
'c68ff54aabf660fcdd9a2838d401583fe31249e3',
);
expect(commit.time, 1626091184);
expect(commit.committer, signature);
expect(commit.author, signature);
expect(commit.tree.sha, '7796359a96eb722939c24bafdb1afe9f07f2f628');
for (var p in parents) {
p.free();
}
signature.free();
commit.free();
});
}); });
group('.create()', () { test('successfully returns when sha hex is short', () {
test('successfuly creates commit', () { final commit = repo[mergeCommit.substring(0, 5)];
const message = "Commit message.\n\nSome description.\n"; expect(commit, isA<Commit>());
const tree = '7796359a96eb722939c24bafdb1afe9f07f2f628'; commit.free();
final author = Signature.create( });
name: 'Author Name',
email: 'author@email.com',
time: 123,
);
final commiter = Signature.create(
name: 'Commiter',
email: 'commiter@email.com',
time: 124,
);
final oid = Commit.create( test('successfully creates commit', () {
repo: repo, final oid = Commit.create(
message: message, repo: repo,
author: author, message: message,
commiter: commiter, author: author,
treeSHA: tree, commiter: commiter,
parentsSHA: [mergeCommit], treeSHA: tree,
); parents: [mergeCommit],
);
final commit = repo[oid.sha]; final commit = repo[oid.sha];
final parents = commit.parents;
expect(commit.id.sha, oid.sha); expect(commit.id.sha, oid.sha);
expect(commit.message, message); expect(commit.message, message);
expect(commit.author, author); expect(commit.messageEncoding, 'utf-8');
expect(commit.committer, commiter); expect(commit.author, author);
expect(commit.time, 124); expect(commit.committer, commiter);
expect(commit.tree.sha, tree); expect(commit.time, 124);
expect(parents.length, 1); expect(commit.tree.sha, tree);
expect(parents[0].id.sha, mergeCommit); expect(commit.parents.length, 1);
expect(commit.parents[0].sha, mergeCommit);
for (var p in parents) { commit.free();
p.free(); });
}
author.free();
commiter.free();
commit.free();
});
test('successfuly creates commit with short sha of tree', () { test('successfully creates commit without parents', () {
const message = "Commit message.\n\nSome description.\n"; final oid = Commit.create(
const tree = '7796359a96eb722939c24bafdb1afe9f07f2f628'; repo: repo,
final author = Signature.create( message: message,
name: 'Author Name', author: author,
email: 'author@email.com', commiter: commiter,
time: 123, treeSHA: tree,
); parents: [],
final commiter = Signature.create( );
name: 'Commiter',
email: 'commiter@email.com',
time: 124,
);
final oid = Commit.create( final commit = repo[oid.sha];
repo: repo,
message: message,
author: author,
commiter: commiter,
treeSHA: tree.substring(0, 5),
parentsSHA: [mergeCommit],
);
final commit = repo[oid.sha]; expect(commit.id.sha, oid.sha);
final parents = commit.parents; expect(commit.message, message);
expect(commit.messageEncoding, 'utf-8');
expect(commit.author, author);
expect(commit.committer, commiter);
expect(commit.time, 124);
expect(commit.tree.sha, tree);
expect(commit.parents.length, 0);
expect(commit.id.sha, oid.sha); commit.free();
expect(commit.message, message); });
expect(commit.author, author);
expect(commit.committer, commiter);
expect(commit.time, 124);
expect(commit.tree.sha, tree);
expect(parents.length, 1);
expect(parents[0].id.sha, mergeCommit);
for (var p in parents) { test('successfully creates commit with 2 parents', () {
p.free(); final oid = Commit.create(
} repo: repo,
author.free(); message: message,
commiter.free(); author: author,
commit.free(); commiter: commiter,
}); treeSHA: tree,
parents: [mergeCommit, 'fc38877b2552ab554752d9a77e1f48f738cca79b'],
);
final commit = repo[oid.sha];
expect(commit.id.sha, oid.sha);
expect(commit.message, message);
expect(commit.messageEncoding, 'utf-8');
expect(commit.author, author);
expect(commit.committer, commiter);
expect(commit.time, 124);
expect(commit.tree.sha, tree);
expect(commit.parents.length, 2);
expect(commit.parents[0].sha, mergeCommit);
expect(commit.parents[1].sha, 'fc38877b2552ab554752d9a77e1f48f738cca79b');
commit.free();
});
test('successfully creates commit with short sha of tree', () {
final oid = Commit.create(
repo: repo,
message: message,
author: author,
commiter: commiter,
treeSHA: tree.substring(0, 5),
parents: [mergeCommit],
);
final commit = repo[oid.sha];
expect(commit.id.sha, oid.sha);
expect(commit.message, message);
expect(commit.messageEncoding, 'utf-8');
expect(commit.author, author);
expect(commit.committer, commiter);
expect(commit.time, 124);
expect(commit.tree.sha, tree);
expect(commit.parents.length, 1);
expect(commit.parents[0].sha, mergeCommit);
commit.free();
}); });
}); });
} }

View file

@ -33,11 +33,8 @@ void main() {
}); });
test('finds object by short oid', () { test('finds object by short oid', () {
final shortSha = '78b8bf'; final oid = Oid.fromSHA(repo, lastCommit.substring(0, 5));
final odb = repo.odb;
final oid = Oid.fromShortSHA(shortSha, odb);
expect(oid.sha, lastCommit); expect(oid.sha, lastCommit);
odb.free();
}); });
}); });
} }

View file

@ -1,5 +1,5 @@
import 'dart:io'; import 'dart:io';
import 'dart:ffi';
import 'package:test/test.dart'; import 'package:test/test.dart';
import 'package:libgit2dart/libgit2dart.dart'; import 'package:libgit2dart/libgit2dart.dart';
import 'helpers/util.dart'; import 'helpers/util.dart';
@ -30,46 +30,50 @@ void main() {
}); });
group('fromSHA()', () { group('fromSHA()', () {
test('initializes successfully', () { test('initializes successfully', () {
final oid = Oid.fromSHA(sha); final oid = Oid.fromSHA(repo, sha);
expect(oid, isA<Oid>());
expect(oid.sha, sha);
});
test('initializes successfully from short hex string', () {
final oid = Oid.fromSHA(repo, sha.substring(0, 5));
expect(oid, isA<Oid>()); expect(oid, isA<Oid>());
expect(oid.sha, sha); expect(oid.sha, sha);
}); });
}); });
group('fromShortSHA()', () { group('fromRaw()', () {
test('initializes successfully from short hex string', () { test('initializes successfully', () {
final odb = repo.odb; final sourceOid = Oid.fromSHA(repo, sha);
final oid = Oid.fromShortSHA(sha.substring(0, 5), odb); final oid = Oid.fromRaw(sourceOid.pointer.ref);
expect(oid, isA<Oid>()); expect(oid, isA<Oid>());
expect(oid.sha, sha); expect(oid.sha, sha);
odb.free();
}); });
}); });
test('returns sha hex string', () { test('returns sha hex string', () {
final oid = Oid.fromSHA(sha); final oid = Oid.fromSHA(repo, sha);
expect(oid.sha, equals(sha)); expect(oid.sha, equals(sha));
}); });
group('compare', () { group('compare', () {
test('< and <=', () { test('< and <=', () {
final oid1 = Oid.fromSHA(sha); final oid1 = Oid.fromSHA(repo, sha);
final oid2 = Oid.fromSHA(biggerSha); final oid2 = Oid.fromSHA(repo, biggerSha);
expect(oid1 < oid2, true); expect(oid1 < oid2, true);
expect(oid1 <= oid2, true); expect(oid1 <= oid2, true);
}); });
test('==', () { test('==', () {
final oid1 = Oid.fromSHA(sha); final oid1 = Oid.fromSHA(repo, sha);
final oid2 = Oid.fromSHA(sha); final oid2 = Oid.fromSHA(repo, sha);
expect(oid1 == oid2, true); expect(oid1 == oid2, true);
}); });
test('> and >=', () { test('> and >=', () {
final oid1 = Oid.fromSHA(sha); final oid1 = Oid.fromSHA(repo, sha);
final oid2 = Oid.fromSHA(lesserSha); final oid2 = Oid.fromSHA(repo, lesserSha);
expect(oid1 > oid2, true); expect(oid1 > oid2, true);
expect(oid1 >= oid2, true); expect(oid1 >= oid2, true);
}); });