- Verify syntax
- Verify variables
- GitLab CI/CD documentation
- Common CI/CD issues
- Pipeline warnings
- How to get help
Troubleshooting CI/CD
GitLab provides several tools to help make troubleshooting your pipelines easier.
This guide also lists common issues and possible solutions.
Verify syntax
An early source of problems can be incorrect syntax. The pipeline shows a yaml invalid
badge and does not start running if any syntax or formatting problems are found.
Edit gitlab-ci.yml
with the pipeline editor
The pipeline editor is the recommended editing experience (rather than the single file editor or the Web IDE). It includes:
- Code completion suggestions that ensure you are only using accepted keywords.
- Automatic syntax highlighting and validation.
- The CI/CD configuration visualization,
a graphical representation of your
.gitlab-ci.yml
file.
If you prefer to use another editor, you can use a schema like
with your editor of choice.
The CI Lint tool is a simple way to ensure the syntax of a CI/CD configuration
file is correct. Paste in full When a A key part of troubleshooting CI/CD is to verify which variables are present in a
pipeline, and what their values are. A lot of pipeline configuration is dependent
on variables, and verifying them is one of the fastest ways to find the source of
a problem.
Export the full list of variables
available in each problematic job. Check if the variables you expect are present,
and check if their values are what you expect.
The complete You can also look at a large number of pipeline configuration examples
and templates.
Some pipeline types have their own detailed usage guides that you should read
if you are using that type:
Troubleshooting guides are available for some CI/CD features and related topics:
A lot of common pipeline issues can be fixed by analyzing the behavior of the If your The If a pipeline does not seem to run at all, with no error message, it may also be
due to If you are converting from The common Two pipelines can run when pushing a commit to a branch that has an open merge request
associated with it. Usually one pipeline is a merge request pipeline, and the other
is a branch pipeline.
This situation is usually caused by the GitLab determines if a job is added to a pipeline based on the Before a pipeline can run, GitLab evaluates all the jobs in the configuration and tries
to add them to all available pipeline types. A pipeline does not run if no jobs are added
to it at the end of the evaluation.
If a pipeline did not run, it’s likely that all the jobs had If the wrong pipeline type ran, then the It’s also possible that your A common reason a job is added to a pipeline unexpectedly is because the The
Previously, you’d have encountered unexpected pipeline failures when you force-pushed
a branch to its remote repository. To illustrate the problem, suppose you’ve had the current workflow:
This occurs because the previous pipeline cannot find a checkout-SHA (which is associated with the pipeline record)
from the As of GitLab 12.4, we’ve improved this behavior by persisting pipeline refs exclusively.
To illustrate its life cycle:
The merge request pipeline widget shows information about the pipeline status in
a merge request. It’s displayed above the ability to merge status widget.
There is a
where a merge request can be stuck with the If your merge request has this message and it does not disappear after a few minutes,
you can try one of these workarounds:
This message is shown when the merge request has no pipeline associated with the
latest commit yet. This might be because:
After the pipeline is created, the message updates with the pipeline status.
The merge request status widget shows the Merge button and whether or not a merge
request is ready to merge. If the merge request can’t be merged, the reason for this
is displayed.
If the pipeline is still running, the Merge button is replaced with the
Merge when pipeline succeeds button.
If Merge Trains
are enabled, the button is either Add to merge train or Add to merge train when pipeline succeeds.
This message is shown if the Pipelines must succeed
setting is enabled in the project and a pipeline has not yet run successfully.
This also applies if the pipeline has not been created yet, or if you are waiting
for an external CI service. If you don’t use pipelines for your project, then you
should disable Pipelines must succeed so you can accept merge requests.
This message is shown if the merge request pipeline,
merged results pipeline,
or merge train pipeline
has failed or been canceled.
If a merge request pipeline or merged result pipeline was canceled or failed, you can:
If the merge train pipeline has failed, you can:
If the merge train pipeline was canceled before the merge request was merged, without a failure, you can:
This message is shown if configuration is added with To resolve this, check that:
This message displays when the YAML configuration is too large or nested too deeply.
YAML files with a large number of includes, and thousands of lines overall, are
more likely to hit this memory limit. For example, a YAML file that is 200kb is
likely to hit the default memory limit.
To reduce the configuration size, you can:
On a self-managed instance, you can increase the size limits.
Pipeline configuration warnings are shown when you:
When you use To prevent duplicate pipelines, use
If you are unable to resolve pipeline issues, you can get help from:
Verify syntax with CI Lint tool
.gitlab-ci.yml
files or individual jobs configuration,
to verify the basic syntax.
.gitlab-ci.yml
file is present in a project, you can also use the CI Lint
tool to simulate the creation of a full pipeline.
It does deeper verification of the configuration syntax.
Verify variables
GitLab CI/CD documentation
.gitlab-ci.yml
reference contains a full list of
every keyword you can use to configure your pipelines.
Documentation for pipeline types
Troubleshooting Guides for CI/CD features
Common CI/CD issues
rules
or only/except
configuration. You shouldn’t use these two configurations in the same
pipeline, as they behave differently. It’s hard to predict how a pipeline runs with
this mixed behavior.
rules
or only/except
configuration makes use of predefined variables
like CI_PIPELINE_SOURCE
, CI_MERGE_REQUEST_ID
, you should verify them
as the first troubleshooting step.
Jobs or pipelines don’t run when expected
rules
or only/except
keywords are what determine whether or not a job is
added to a pipeline. If a pipeline runs, but a job is not added to the pipeline,
it’s usually due to rules
or only/except
configuration issues.
rules
or only/except
configuration, or the workflow: rules
keyword.
only/except
to the rules
keyword, you should check
the rules
configuration details carefully. The behavior
of only/except
and rules
is different and can cause unexpected behavior when migrating
between the two.
if
clauses for rules
can be very helpful for examples of how to write rules that behave the way you expect.
Two pipelines run at the same time
rules
configuration, and there are several ways to
prevent duplicate pipelines.
A job is not in the pipeline
only/except
or rules
defined for the job. If it didn’t run, it’s probably
not evaluating as you expect.
No pipeline or the wrong type of pipeline runs
rules
or only/except
that
blocked them from being added to the pipeline.
rules
or only/except
configuration should
be checked to make sure the jobs are added to the correct pipeline type. For
example, if a merge request pipeline did not run, the jobs may have been added to
a branch pipeline instead.
workflow: rules
configuration
blocked the pipeline, or allowed the wrong pipeline type.
A job runs unexpectedly
changes
keyword always evaluates to true in certain cases. For example, changes
is always
true in certain pipeline types, including scheduled pipelines and pipelines for tags.
changes
keyword is used in combination with only/except
or rules
). It’s recommended to use changes
with
rules
or only/except
configuration that ensures the job is only added to branch
pipelines or merge request pipelines.
“fatal: reference is not a tree” error
example
and pushes it to a remote repository.
example
branch.
example
branch on the latest default branch and force-pushes it to its remote repository.
example
branch again, however,
the previous pipeline (2) fails because of fatal: reference is not a tree:
error.
example
branch that the commit history has already been overwritten by the force-push.
Similarly, Merged results pipelines
might have failed intermittently due to the same reason.
example
.
refs/pipelines/<pipeline-id>
,
which retains the checkout-SHA of the associated pipeline record.
This persistent ref stays intact during the pipeline execution,
even if the commit history of the example
branch has been overwritten by force-push.
Merge request pipeline messages
“Checking ability to merge automatically” message
Checking ability to merge automatically
message.
/rebase
quick action.
/merge
quick action.
“Checking pipeline status” message
Merge request status messages
“A CI/CD pipeline must run and be successful before merge” message
“Merge blocked: pipeline must succeed. Push a new commit that fixes the failure” message
/merge
quick action to immediately add the merge request to the train again.
Project
group/project
not found or access denied
include
and one of the following:
my-group/my-project
and does not include
any folders in the repository.
“The parsed YAML is too big” message
script
sections into standalone scripts in the project.
Pipeline warnings
“Job may allow multiple pipelines to run for a single action” warning
rules
with a when
clause without an if
clause, multiple pipelines may run. Usually this occurs when you push a commit to
a branch that has an open merge request associated with it.
workflow: rules
or rewrite your rules to control
which pipelines can run.
Console workaround if job using resource_group gets stuck
# find resource group by name
resource_group = Project.find_by_full_path('...').resource_groups.find_by(key: 'the-group-name')
busy_resources = resource_group.resources.where('build_id IS NOT NULL')
# identify which builds are occupying the resource
# (I think it should be 1 as of today)
busy_resources.pluck(:build_id)
# it's good to check why this build is holding the resource.
# Is it stuck? Has it been forcefully dropped by the system?
# free up busy resources
busy_resources.update_all(build_id: nil)
How to get help