I have set up a GitHub action that is supposed to change the commit it was started on. This looks something like
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
with:
fetch-depth: 0
- name: Do some computations
run: |
# Create some new files that should be added to the commit
dvc repro
- name: Commit Results
run: |
git config --local user.email "[email protected]"
git config --local user.name "GitHub Action"
git add .
git commit --amend --no-edit
git push --force
This works fine, when there are no updates to the branch while the action is running. It becomes more complicated when I try to have multiple actions running with some small offset and try to use the following to combine the results:
- name: Commit Results
run: |
git config --local user.email "[email protected]"
git config --local user.name "GitHub Action"
CIREVISION=$(git rev-parse HEAD)
# stash changes
git add .
git stash
# update for new commit
git pull --rebase
# start rebase only on the commit of this CI run
GIT_SEQUENCE_EDITOR="sed -i -re '0,/pick/{s/pick/edit/}'" git rebase -i "$CIREVISION^"
# update files and commit
git stash pop
git add .
git commit --amend --no-edit
# finish rebase <- this fails with a merge conflict
git rebase --continue
git push --force
which in the case of a single changed file metrics.json
fails with
Auto-merging metrics.json
CONFLICT (content): Merge conflict in metrics.json
Rebasing (2/3)
error: could not apply ABCD123... TEST01
hint: Resolve all conflicts manually, mark them as resolved with
I tried to use something like git status | grep both | awk '{print $3}' | xargs git checkout --ours .
to not merge but always accept the version of the file present in the current branch, but I could not resolve the merge conflict.
A little more context:
I am using Github CI together with cml.dev to version control and allow for full reproducibility of research data. I update a parameter file, this triggers some computation and the results of these computations are saved externally, but some small metrics should be stored via git together with the correct parameters. Having one commit with the updated parameters but the old metrics is not feasable, therefore I must change the original commit.
I am aware of the consequences it has, to force push to a repository and I am willing to take the risk.
I usually use git for software development and have not looked into changing an existing git structure, because I usually don't want to do this.
If all you need is to commit the current state regardless of upstream changes, then
fetch
andreset
:Naturally the
push
may fail so the whole thing should be in a retry loop. This avoids the scariness of CI doing--force
pushes.