feat(worktree): add more bindings and api methods

This commit is contained in:
Aleksey Kulikov 2021-10-14 11:23:30 +03:00
parent 9f346c99f7
commit 15ce546f7a
3 changed files with 119 additions and 4 deletions

View file

@ -75,6 +75,27 @@ Pointer<git_worktree> lookup({
} }
} }
/// Check if the worktree prunable.
///
/// A worktree is not prunable in the following scenarios:
/// - the worktree is linking to a valid on-disk worktree.
/// - the worktree is locked.
///
/// Throws a [LibGit2Error] if error occured.
bool isPrunable(Pointer<git_worktree> wt) {
final opts = calloc<git_worktree_prune_options>();
final optsError = libgit2.git_worktree_prune_options_init(
opts,
GIT_WORKTREE_PRUNE_OPTIONS_VERSION,
);
if (optsError < 0) {
throw LibGit2Error(libgit2.git_error_last());
}
return libgit2.git_worktree_is_prunable(wt, opts) > 0 ? true : false;
}
/// Prune working tree. /// Prune working tree.
/// ///
/// Prune the working tree, that is remove the git data structures on disk. /// Prune the working tree, that is remove the git data structures on disk.
@ -118,5 +139,37 @@ String path(Pointer<git_worktree> wt) {
return libgit2.git_worktree_path(wt).cast<Utf8>().toDartString(); return libgit2.git_worktree_path(wt).cast<Utf8>().toDartString();
} }
/// Check if worktree is locked.
///
/// A worktree may be locked if the linked working tree is stored on a portable
/// device which is not available.
///
/// Throws a [LibGit2Error] if error occured.
bool isLocked(Pointer<git_worktree> wt) {
final result = libgit2.git_worktree_is_locked(nullptr, wt);
if (result < 0) {
throw LibGit2Error(libgit2.git_error_last());
} else if (result == 0) {
return false;
} else {
return true;
}
}
/// Lock worktree if not already locked.
void lock(Pointer<git_worktree> wt) => libgit2.git_worktree_lock(wt, nullptr);
/// Unlock a locked worktree.
void unlock(Pointer<git_worktree> wt) => libgit2.git_worktree_unlock(wt);
/// Check if worktree is valid.
///
/// A valid worktree requires both the git data structures inside the linked parent
/// repository and the linked working copy to be present.
bool isValid(Pointer<git_worktree> wt) {
return libgit2.git_worktree_validate(wt) == 0 ? true : false;
}
/// Free a previously allocated worktree. /// Free a previously allocated worktree.
void free(Pointer<git_worktree> wt) => libgit2.git_worktree_free(wt); void free(Pointer<git_worktree> wt) => libgit2.git_worktree_free(wt);

View file

@ -52,6 +52,29 @@ class Worktree {
/// Returns the filesystem path for the worktree. /// Returns the filesystem path for the worktree.
String get path => bindings.path(_worktreePointer); String get path => bindings.path(_worktreePointer);
/// Checks if worktree is locked.
///
/// A worktree may be locked if the linked working tree is stored on a portable
/// device which is not available.
///
/// Throws a [LibGit2Error] if error occured.
bool get isLocked => bindings.isLocked(_worktreePointer);
/// Locks worktree if not already locked.
void lock() => bindings.lock(_worktreePointer);
/// Unlocks a locked worktree.
void unlock() => bindings.unlock(_worktreePointer);
/// Checks if the worktree prunable.
///
/// A worktree is not prunable in the following scenarios:
/// - the worktree is linking to a valid on-disk worktree.
/// - the worktree is locked.
///
/// Throws a [LibGit2Error] if error occured.
bool get isPrunable => bindings.isPrunable(_worktreePointer);
/// Prunes working tree. /// Prunes working tree.
/// ///
/// Prune the working tree, that is remove the git data structures on disk. /// Prune the working tree, that is remove the git data structures on disk.
@ -59,6 +82,12 @@ class Worktree {
/// Throws a [LibGit2Error] if error occured. /// Throws a [LibGit2Error] if error occured.
void prune() => bindings.prune(_worktreePointer); void prune() => bindings.prune(_worktreePointer);
/// Checks if worktree is valid.
///
/// A valid worktree requires both the git data structures inside the linked parent
/// repository and the linked working copy to be present.
bool get isValid => bindings.isValid(_worktreePointer);
/// Releases memory allocated for worktree object. /// Releases memory allocated for worktree object.
void free() => bindings.free(_worktreePointer); void free() => bindings.free(_worktreePointer);

View file

@ -33,21 +33,18 @@ void main() {
name: worktreeName, name: worktreeName,
path: worktreeDir.path, path: worktreeDir.path,
); );
final lookedupWorktree = repo.lookupWorktree(worktreeName);
final branches = repo.branches; final branches = repo.branches;
expect(repo.worktrees, [worktreeName]); expect(repo.worktrees, [worktreeName]);
expect(branches.any((branch) => branch.name == worktreeName), true); expect(branches.any((branch) => branch.name == worktreeName), true);
expect(worktree.name, worktreeName); expect(worktree.name, worktreeName);
expect(lookedupWorktree.name, worktreeName);
expect(worktree.path, worktreeDir.path); expect(worktree.path, worktreeDir.path);
expect(lookedupWorktree.path, worktreeDir.path); expect(worktree.isLocked, false);
expect(File('${worktreeDir.path}/.git').existsSync(), true); expect(File('${worktreeDir.path}/.git').existsSync(), true);
for (final branch in branches) { for (final branch in branches) {
branch.free(); branch.free();
} }
lookedupWorktree.free();
worktree.free(); worktree.free();
}); });
@ -87,6 +84,37 @@ void main() {
worktree.free(); worktree.free();
}); });
test('successfully lookups worktree', () {
final worktree = repo.createWorktree(
name: worktreeName,
path: worktreeDir.path,
);
final lookedupWorktree = repo.lookupWorktree(worktreeName);
expect(lookedupWorktree.name, worktreeName);
expect(lookedupWorktree.path, worktreeDir.path);
expect(lookedupWorktree.isLocked, false);
lookedupWorktree.free();
worktree.free();
});
test('successfully locks and unlocks worktree', () {
final worktree = repo.createWorktree(
name: worktreeName,
path: worktreeDir.path,
);
expect(worktree.isLocked, false);
worktree.lock();
expect(worktree.isLocked, true);
worktree.unlock();
expect(worktree.isLocked, false);
worktree.free();
});
test('successfully prunes worktree', () { test('successfully prunes worktree', () {
expect(repo.worktrees, []); expect(repo.worktrees, []);
@ -95,8 +123,13 @@ void main() {
path: worktreeDir.path, path: worktreeDir.path,
); );
expect(repo.worktrees, [worktreeName]); expect(repo.worktrees, [worktreeName]);
expect(worktree.isPrunable, false);
expect(worktree.isValid, true);
worktreeDir.deleteSync(recursive: true); worktreeDir.deleteSync(recursive: true);
expect(worktree.isPrunable, true);
expect(worktree.isValid, false);
worktree.prune(); worktree.prune();
expect(repo.worktrees, []); expect(repo.worktrees, []);