Merge pull request 'Add installation instructions and initial test coverage' (#5) from add-tests into main
Reviewed-on: https://codeberg.org/svenvanheugten/git-check-assertions/pulls/5
This commit is contained in:
commit
e219f037a5
6 changed files with 180 additions and 26 deletions
37
README.md
37
README.md
|
|
@ -13,6 +13,43 @@ You include a small bash script inside your commit messages, and `git-check-asse
|
||||||
|
|
||||||
⚠️ Only run this on repositories and branches that you trust, since the `bash` scripts in the commit messages can do whatever they want.
|
⚠️ Only run this on repositories and branches that you trust, since the `bash` scripts in the commit messages can do whatever they want.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
On most systems, clone this repository and add the `bin` directory to your `PATH`.
|
||||||
|
|
||||||
|
If you use Nix with flakes, you can simply add it to your program's devshell instead:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
@@ -2,12 +2,18 @@
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:nixos/nixpkgs/nixpkgs-unstable";
|
||||||
|
flake-utils.url = "github:numtide/flake-utils";
|
||||||
|
+ git-check-assertions = {
|
||||||
|
+ url = "git+https://codeberg.org/svenvanheugten/git-check-assertions.git?ref=main";
|
||||||
|
+ inputs.nixpkgs.follows = "nixpkgs";
|
||||||
|
+ inputs.flake-utils.follows = "flake-utils";
|
||||||
|
+ };
|
||||||
|
};
|
||||||
|
outputs =
|
||||||
|
{
|
||||||
|
self,
|
||||||
|
nixpkgs,
|
||||||
|
flake-utils,
|
||||||
|
+ git-check-assertions,
|
||||||
|
}:
|
||||||
|
flake-utils.lib.eachDefaultSystem (
|
||||||
|
system:
|
||||||
|
@@ -17,7 +23,10 @@
|
||||||
|
{
|
||||||
|
packages.default = pkgs.callPackage ./default.nix { };
|
||||||
|
devShells.default = pkgs.mkShell {
|
||||||
|
- packages = [ ];
|
||||||
|
+ packages = [ git-check-assertions.packages.${system}.default ];
|
||||||
|
};
|
||||||
|
}
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
## Examples of commit messages
|
## Examples of commit messages
|
||||||
|
|
||||||
Assert that a commit builds:
|
Assert that a commit builds:
|
||||||
|
|
|
||||||
|
|
@ -25,7 +25,10 @@
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
assert_fails() {
|
assert_fails() {
|
||||||
! "$@" || { echo "Expected command to fail, but it succeeded."; exit 1; }
|
! "$@" || {
|
||||||
|
echo "Expected command to fail, but it succeeded."
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
}
|
}
|
||||||
export -f assert_fails
|
export -f assert_fails
|
||||||
|
|
||||||
|
|
@ -42,9 +45,9 @@ for commit_hash in "${commits[@]}"; do
|
||||||
git -c advice.detachedHead=false checkout -q "$commit_hash"
|
git -c advice.detachedHead=false checkout -q "$commit_hash"
|
||||||
commit_msg="$(git log -1 --format=%B "$commit_hash")"
|
commit_msg="$(git log -1 --format=%B "$commit_hash")"
|
||||||
# shellcheck disable=SC2016
|
# shellcheck disable=SC2016
|
||||||
block="$(printf '%s\n' "$commit_msg" \
|
block="$(printf '%s\n' "$commit_msg" |
|
||||||
| sed -n '/^```git-check-assertions[[:space:]]*$/,/^```[[:space:]]*$/p' \
|
sed -n '/^```git-check-assertions[[:space:]]*$/,/^```[[:space:]]*$/p' |
|
||||||
| sed '1d;$d')"
|
sed '1d;$d')"
|
||||||
if [ -n "$block" ]; then
|
if [ -n "$block" ]; then
|
||||||
echo "git-check-assertions block in $commit_hash:"
|
echo "git-check-assertions block in $commit_hash:"
|
||||||
printf '%s\n' "$block" | sed 's/^/> /'
|
printf '%s\n' "$block" | sed 's/^/> /'
|
||||||
25
default.nix
25
default.nix
|
|
@ -8,6 +8,7 @@
|
||||||
shellcheck-minimal,
|
shellcheck-minimal,
|
||||||
gitMinimal,
|
gitMinimal,
|
||||||
resholve,
|
resholve,
|
||||||
|
shfmt,
|
||||||
}:
|
}:
|
||||||
|
|
||||||
let
|
let
|
||||||
|
|
@ -20,8 +21,9 @@ resholve.mkDerivation {
|
||||||
src = fs.toSource {
|
src = fs.toSource {
|
||||||
root = ./.;
|
root = ./.;
|
||||||
fileset = fs.unions [
|
fileset = fs.unions [
|
||||||
./git-check-assertions
|
./.editorconfig
|
||||||
./test.bats
|
./bin
|
||||||
|
./test
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -31,19 +33,20 @@ resholve.mkDerivation {
|
||||||
|
|
||||||
nativeCheckInputs = [
|
nativeCheckInputs = [
|
||||||
shellcheck-minimal
|
shellcheck-minimal
|
||||||
bats
|
(bats.withLibraries (p: [
|
||||||
|
p.bats-assert
|
||||||
|
p.bats-support
|
||||||
|
p.bats-file
|
||||||
|
]))
|
||||||
gitMinimal
|
gitMinimal
|
||||||
|
shfmt
|
||||||
];
|
];
|
||||||
|
|
||||||
checkPhase = ''
|
checkPhase = ''
|
||||||
runHook preCheck
|
runHook preCheck
|
||||||
shellcheck git-check-assertions test.bats
|
shellcheck bin/git-check-assertions test/git-check-assertions.bats
|
||||||
git init
|
shfmt -d bin/git-check-assertions test/git-check-assertions.bats
|
||||||
git config user.email test
|
./test/git-check-assertions.bats
|
||||||
git config user.name test
|
|
||||||
git checkout -b main
|
|
||||||
git commit --allow-empty -m "initial"
|
|
||||||
./test.bats
|
|
||||||
runHook postCheck
|
runHook postCheck
|
||||||
'';
|
'';
|
||||||
|
|
||||||
|
|
@ -52,7 +55,7 @@ resholve.mkDerivation {
|
||||||
installPhase = ''
|
installPhase = ''
|
||||||
runHook preInstall
|
runHook preInstall
|
||||||
mkdir -p $out/bin
|
mkdir -p $out/bin
|
||||||
cp $src/git-check-assertions $out/bin/git-check-assertions
|
cp $src/bin/git-check-assertions $out/bin/git-check-assertions
|
||||||
runHook postInstall
|
runHook postInstall
|
||||||
'';
|
'';
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -16,7 +16,14 @@
|
||||||
{
|
{
|
||||||
packages.default = pkgs.callPackage ./default.nix { };
|
packages.default = pkgs.callPackage ./default.nix { };
|
||||||
devShells.default = pkgs.mkShell {
|
devShells.default = pkgs.mkShell {
|
||||||
packages = with pkgs; [ bats ];
|
packages = [
|
||||||
|
(pkgs.bats.withLibraries (p: [
|
||||||
|
p.bats-assert
|
||||||
|
p.bats-support
|
||||||
|
p.bats-file
|
||||||
|
]))
|
||||||
|
pkgs.shfmt
|
||||||
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
|
||||||
10
test.bats
10
test.bats
|
|
@ -1,10 +0,0 @@
|
||||||
#!/usr/bin/env bats
|
|
||||||
|
|
||||||
setup() {
|
|
||||||
DIR="$( cd "$( dirname "$BATS_TEST_FILENAME" )" >/dev/null 2>&1 && pwd )"
|
|
||||||
PATH="$DIR:$PATH"
|
|
||||||
}
|
|
||||||
|
|
||||||
@test "can run" {
|
|
||||||
git-check-assertions
|
|
||||||
}
|
|
||||||
114
test/git-check-assertions.bats
Executable file
114
test/git-check-assertions.bats
Executable file
|
|
@ -0,0 +1,114 @@
|
||||||
|
#!/usr/bin/env bats
|
||||||
|
|
||||||
|
setup() {
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
bats_load_library bats-support
|
||||||
|
bats_load_library bats-assert
|
||||||
|
bats_load_library bats-file
|
||||||
|
|
||||||
|
DIR="$(cd "$(dirname "$BATS_TEST_FILENAME")" >/dev/null 2>&1 && pwd)"
|
||||||
|
PATH="$DIR/../bin:$PATH"
|
||||||
|
|
||||||
|
mkdir "$BATS_TEST_TMPDIR/repo"
|
||||||
|
cd "$BATS_TEST_TMPDIR/repo"
|
||||||
|
git init -b main
|
||||||
|
git config user.email "john.doe@example.com"
|
||||||
|
git config user.name "John Doe"
|
||||||
|
touch readme
|
||||||
|
git add readme
|
||||||
|
git commit -m "initial"
|
||||||
|
}
|
||||||
|
|
||||||
|
commit_with_assertion() {
|
||||||
|
git commit --allow-empty -F - <<-EOF
|
||||||
|
test
|
||||||
|
|
||||||
|
\`\`\`git-check-assertions
|
||||||
|
$1
|
||||||
|
\`\`\`
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should not run any assertion blocks when on main" {
|
||||||
|
commit_with_assertion "touch ../test"
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
assert_file_not_exists ../test
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should not run any assertion blocks from main when on a feature branch" {
|
||||||
|
commit_with_assertion "touch ../test"
|
||||||
|
git checkout -b feature
|
||||||
|
git commit --allow-empty -m "feature"
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
assert_file_not_exists ../test
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should run all succeeding assertion blocks on the feature branch and finally return to the original branch" {
|
||||||
|
git checkout -b feature
|
||||||
|
commit_with_assertion "touch ../test1"
|
||||||
|
commit_with_assertion "touch ../test2"
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
assert_file_exists ../test1
|
||||||
|
assert_file_exists ../test2
|
||||||
|
run git symbolic-ref --short HEAD
|
||||||
|
assert_output "feature"
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should stop at the first failing assertion block and return to the original branch" {
|
||||||
|
git checkout -b feature
|
||||||
|
commit_with_assertion "touch ../test1 && exit 3"
|
||||||
|
commit_with_assertion "touch ../test2"
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_failure
|
||||||
|
assert_file_exists ../test1
|
||||||
|
assert_file_not_exists ../test2
|
||||||
|
run git symbolic-ref --short HEAD
|
||||||
|
assert_output "feature"
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should restore worktree when finished" {
|
||||||
|
git checkout -b feature
|
||||||
|
commit_with_assertion 'echo blah >> readme'
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
assert_size_zero readme
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "should restore worktree between commits" {
|
||||||
|
git checkout -b feature
|
||||||
|
commit_with_assertion 'echo blah >> readme'
|
||||||
|
commit_with_assertion 'cp readme ../test'
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
assert_size_zero ../test
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "assertions should run against the version of the code inside of the commit" {
|
||||||
|
git checkout -b feature
|
||||||
|
echo commit1 >readme
|
||||||
|
git add readme
|
||||||
|
commit_with_assertion 'grep commit1 readme'
|
||||||
|
echo commit2 >readme
|
||||||
|
git add readme
|
||||||
|
commit_with_assertion 'grep commit2 readme'
|
||||||
|
|
||||||
|
run git-check-assertions
|
||||||
|
|
||||||
|
assert_success
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue