Git: Which branch is the tag on?
To build your pipelines, you may need the name of the branch on which the pipeline is running. In most cases, the predefined variables CI_COMMIT_BRANCH
or CI_COMMIT_REF_NAME
can be used for this. According to GitLab Docs:
CI_COMMIT_BRANCH - The commit branch name. Available in branch pipelines, including pipelines for the default branch. Not available in merge request pipelines or tag pipelines
CI_COMMIT_REF_NAME - The branch or tag name for which project is built
But what if I want to know on which branch my tag pipeline was started?
Why is this a question?
This questions can seem nonsensical for an experienced git user, but it is not obvious why. There seems to be some kind of misunderstanding about this question. Here are some links on GitLab issues with similar questions that I’ve found during my research: (1), (2),(3)
I think I’m not the only one who looked for the answer; that’s why I want to share a detailed answer here.
Explanation
Firstly, let’s clarify the misconception of “the branch of the tag”.
- Tags in git are very similar to branches; they both are just the references to commits. So, tags are not connected with branches; they are connected with commits.
- One commit can be contained by multiple branches.
- The original branch from which the tag was created may be deleted at all, and the tag may still be present in other branches.
So, technically, there is no direct connection between tag and branch. Moreover, multiple branches can contain the same tag. We cannot be sure that there are no other branches containing the tagged commit. And we cannot be sure that the existing branch containing the tag is the original one.
Here are some examples:
- Create new tag
tag1
on a branchdevelop
, then create branchfeature
fromdevelop
. Tagtag1
will be in bothdevelop
andfeature
branches. - Create a new tag
tag2
on thedevelop
branch, then merge it into themain
. Tagtag2
will be in bothdevelop
andmain
branches.
But practically, in most cases, only one branch will contain the tag when it’s created and the pipeline starts. So there might be a workaround here.
Now, when it’s clear that it is not possible to get a branch of the tag and why CI_COMMIT_BRANCH
is empty in tags pipelines, we can move on to solutions.
Possible solutions
There is no universal solution, but there are some workarounds. Most of them implement the same concept: you cannot obtain the branch, but you can list all branches, filter them, order, and take the first one.
Several ways to do this are discussed on this StackOverflow topic, each with its own pros and cons.
In my case, this one by Swags became the most useful. I used it to redefine a CI_COMMIT_BRANCH
variable in the before_script
section.
1
2
before_script:
- CI_COMMIT_BRANCH=$(git ls-remote --heads origin | grep $CI_COMMIT_SHORT_SHA | sed "s/.*\///")
It lists only the heads of remote branches and takes the first one that matches with the CI_COMMIT_SHORT_SHA
. Now CI_COMMIT_BRANCH
is not empty even in tag pipelines.
Note: This will require
git
installed on your base image.
The main disadvantage of this method is that it will only work when you’re tagging the heads of the branches. But when you are, it works well.
Conclusion
Later, I decided that it would be better to build pipelines logic based on tags names and regular expressions rather than branch names. So, as for me, I want to leave here this tip:
Tip: Don’t try to find out on which branch the tag was created. It would be better to establish cleaner tag naming conventions.
But who knows, maybe this note will be helpful for someone else.