How to create commit automatically in a merge request in GitLab

There are situations when you would like to create a commit during continuous integration (CI) of your merge request (MR). For example:
  • apply automatic fixes from a linter
  • extracting translations from the code
  • In this article, I'll show how to set it up with GitLab.
    Assumptions
    I will job to be triggered manually. In the real-world examples listed above, you would like to automatically trigger them and just let the git ignore empty commits. As I'm always doing some change in my simplified example, doing it manually saves us from an infinite loop of an ever-changing branch.
    I want the changes to be done only on the non-main (non-master) branches. There could be cases where changing master directly makes sense, but I'm used to MR-heavy workflow & I find direct commits to main a bit icky.
    Credentials
    This approach is only achievable with Personal Access Tokens. You will need somebody on your team to set it up via their account and pretty much make them share the key with the whole team. And it will stop working as soon as you remove the credentials from the person - which is likely to complicate stuff even further when your CI maintainer leaves the company.
    Creating s token
    Go to Personal Access Tokens page, and create a token with at least write_repository permission:
    Adding the token
    Once created, you need to add it as a variable in your project's CI/CD setting page. The variable name I use further in the code is GITLAB_PUSH_TOKEN. With the above assumptions, you need to set the variable not to be protected - it has to be available on non-main branches. I recommend you make it masked - it will still be available for people who have access to this setting page, but at least your key will not appear in the job logs.
    Variables
    In the following configuration, I use a lot of predefined GitLab variables. You can find the complete list of them in the doc.
    The configuration
    stages:
      - test
    Stages definition, I called it test following what would make sense in the lint changes use case.
    add-commit:
      stage: test
    Creates a job & sets the stage.
    rules:
        - if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
          when: never
        - if: $CI_COMMIT_BRANCH != $CI_DEFAULT_BRANCH
          when: manual
    I make it never run on main/master & manually run on all the other branches.
    script:
        - echo 'test' >> $CI_COMMIT_BRANCH
        - git add .
    My dummy changes. I put test into the file named after the branch.
    - git status
    Sanity check. Helped me when I was developing the example and will be useful in troubleshooting if needed later on.
    - git -c user.email="$GITLAB_USER_EMAIL" -c user.name="$GITLAB_USER_NAME" commit -m "add change in $CI_PIPELINE_ID"
    The meaning is as follows:
  • -c user. ...- sets git configuration for the command alone. Feels cleaner to me than calling git config --global ..., even though the context in which it would be run is discarded immediately after the job is done.
  • commit ... is already a standard commit call, but it has to be after the configuration was set
  • - git push "https://gitlab-ci-token:$GITLAB_PUSH_TOKEN@gitlab.com/marcin-wosinek/automated-commit.git" HEAD:$CI_COMMIT_BRANCH
  • git push https://gitlab-ci-token:$GITLAB_PUSH_TOKEN pushes with the token read from $GITLAB_PUSH_TOKEN we defined earlier. Any configuration mismatch will fail here.
  • gitlab.com/marcin-wosinek/automated-commit.git - of course, you will need to replace marcin-wosinek with your username/organization & `automated-commit with your project name. If you use a different GitLab server, it should be set here as well.
  • HEAD:$CI_COMMIT_BRANCH - the CI is in the detached HEAD state - there is no current branch. To work around it, I'm explicitly setting what branch I'm pushing to.
  • Complete .gitlab-ci.yml
    `yml
    stages:
  • test
  • add-commit:
    stage: test
    rules:
    - if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
    when: never
    - if: $CI_COMMIT_BRANCH != $CI_DEFAULT_BRANCH
    when: manual
    script:
    - echo 'test' >> $CI_COMMIT_BRANCH
    - git add .
    - git status
    - git -c user.email="$GITLAB_USER_EMAIL" -c user.name="$GITLAB_USER_NAME" commit -m "add change in $CI_PIPELINE_ID"
    - git push "https://gitlab-ci-token:$GITLAB_PUSH_TOKEN@gitlab.com/marcin-wosinek/automated-commit.git" HEAD:$CI_COMMIT_BRANCH
    `
    Links
    You can find my test repository here. The CI created commit.
    Summary
    In this article, we have seen how to make automated commits in GitLab CI. Would you mind sharing in a comment what use case you want to use this for?

    37

    This website collects cookies to deliver better user experience

    How to create commit automatically in a merge request in GitLab