Git rebase inside github actions

2.9k Views Asked by At

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.

1

There are 1 best solutions below

0
On

If all you need is to commit the current state regardless of upstream changes, then fetch and reset:

BRANCH=experiment
git fetch --all
git reset --mixed origin/$BRANCH
git add .
git commit -m "a new experiment"
git push origin HEAD:$BRANCH

Naturally the push may fail so the whole thing should be in a retry loop. This avoids the scariness of CI doing --force pushes.