2020-10-24 23:57:45 +05:30
---
stage: Verify
2021-11-11 11:23:49 +05:30
group: Pipeline Authoring
2021-02-22 17:27:13 +05:30
info: To determine the technical writer assigned to the Stage/Group associated with this page, see https://about.gitlab.com/handbook/engineering/ux/technical-writing/#assignments
2020-10-24 23:57:45 +05:30
---
2021-09-04 01:27:46 +05:30
# CI Lint API **(FREE)**
2021-01-03 14:25:43 +05:30
## Validate the CI YAML configuration
2016-09-29 09:46:39 +05:30
2021-01-03 14:25:43 +05:30
Checks if CI/CD YAML configuration is valid. This endpoint validates basic CI/CD
2021-11-11 11:23:49 +05:30
configuration syntax. It doesn't have any namespace-specific context.
2016-09-29 09:46:39 +05:30
2021-09-04 01:27:46 +05:30
Access to this endpoint does not require authentication when the instance
[allows new sign ups ](../user/admin_area/settings/sign_up_restrictions.md#disable-new-sign-ups )
and:
- Does not have an [allowlist or denylist ](../user/admin_area/settings/sign_up_restrictions.md#allow-or-deny-sign-ups-using-specific-email-domains ).
- Does not [require administrator approval for new sign ups ](../user/admin_area/settings/sign_up_restrictions.md#require-administrator-approval-for-new-sign-ups ).
- Does not have additional [sign up
restrictions](../user/admin_area/settings/sign_up_restrictions.html#sign-up-restrictions).
Otherwise, authentication is required.
2021-03-11 19:13:27 +05:30
2020-04-08 14:13:33 +05:30
```plaintext
2019-10-12 21:52:04 +05:30
POST /ci/lint
2016-09-29 09:46:39 +05:30
```
| Attribute | Type | Required | Description |
| ---------- | ------- | -------- | -------- |
2021-01-03 14:25:43 +05:30
| `content` | string | yes | The CI/CD configuration content. |
| `include_merged_yaml` | boolean | no | If the [expanded CI/CD configuration ](#yaml-expansion ) should be included in the response. |
2021-12-11 22:18:48 +05:30
| `include_jobs` | boolean | no | If the list of jobs should be included in the response. This is false by default. |
2016-09-29 09:46:39 +05:30
2020-03-13 15:44:24 +05:30
```shell
2021-03-11 19:13:27 +05:30
curl --header "Content-Type: application/json" --header "PRIVATE-TOKEN: < your_access_token > " "https://gitlab.example.com/api/v4/ci/lint" --data '{"content": "{ \"image\": \"ruby:2.6\", \"services\": [\"postgres\"], \"before_script\": [\"bundle install\", \"bundle exec rake db:create\"], \"variables\": {\"DB_NAME\": \"postgres\"}, \"types\": [\"test\", \"deploy\", \"notify\"], \"rspec\": { \"script\": \"rake spec\", \"tags\": [\"ruby\", \"postgres\"], \"only\": [\"branches\"]}}"}'
2016-09-29 09:46:39 +05:30
```
2021-01-03 14:25:43 +05:30
Be sure to paste the exact contents of your GitLab CI/CD YAML configuration because YAML
is very sensitive about indentation and spacing.
2016-09-29 09:46:39 +05:30
Example responses:
2019-03-02 22:35:43 +05:30
- Valid content:
2016-09-29 09:46:39 +05:30
2019-09-30 21:07:59 +05:30
```json
{
"status": "valid",
2021-01-29 00:20:46 +05:30
"errors": [],
"warnings": []
}
```
- Valid content with warnings:
```json
{
"status": "valid",
"errors": [],
"warnings": ["jobs:job may allow multiple pipelines to run for a single action due to
`rules:when` clause with no `workflow:rules` - read more:
https://docs.gitlab.com/ee/ci/troubleshooting.html#pipeline-warnings"]
2019-09-30 21:07:59 +05:30
}
```
2016-09-29 09:46:39 +05:30
2019-03-02 22:35:43 +05:30
- Invalid content:
2016-09-29 09:46:39 +05:30
2019-09-30 21:07:59 +05:30
```json
{
"status": "invalid",
"errors": [
"variables config should be a hash of key value pairs"
2021-01-29 00:20:46 +05:30
],
"warnings": []
2019-09-30 21:07:59 +05:30
}
```
2016-09-29 09:46:39 +05:30
2019-03-02 22:35:43 +05:30
- Without the content attribute:
2016-09-29 09:46:39 +05:30
2019-09-30 21:07:59 +05:30
```json
{
"error": "content is missing"
}
```
2021-01-03 14:25:43 +05:30
### YAML expansion
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/29568) in GitLab 13.5.
The CI lint returns an expanded version of the configuration. The expansion does not
2021-09-30 23:02:18 +05:30
work for CI configuration added with [`include: local` ](../ci/yaml/index.md#includelocal ),
2022-04-04 11:22:00 +05:30
and the [`extends:` ](../ci/yaml/index.md#extends ) keyword is [not fully supported ](https://gitlab.com/gitlab-org/gitlab/-/issues/258843 ).
2021-01-03 14:25:43 +05:30
Example contents of a `.gitlab-ci.yml` passed to the CI Lint API with
2021-12-11 22:18:48 +05:30
`include_merged_yaml` and `include_jobs` set as true:
2021-01-03 14:25:43 +05:30
```yaml
include:
remote: 'https://example.com/remote.yaml'
test:
stage: test
script:
- echo 1
```
Example contents of `https://example.com/remote.yaml` :
```yaml
another_test:
stage: test
script:
- echo 2
```
Example response:
```json
{
"status": "valid",
"errors": [],
2021-12-11 22:18:48 +05:30
"merged_yaml": "---\n:another_test:\n :stage: test\n :script: echo 2\n:test:\n :stage: test\n :script: echo 1\n",
"jobs": [
{
"name":"test",
"stage":"test",
"before_script":[],
"script":["echo 1"],
"after_script":[],
"tag_list":[],
"environment":null,
"when":"on_success",
"allow_failure":false,
"only":{
"refs":["branches","tags"]
},
"except":null
},
{
"name":"another_test",
"stage":"test",
"before_script":[],
"script":["echo 2"],
"after_script":[],
"tag_list":[],
"environment":null,
"when":"on_success",
"allow_failure":false,
"only":{
"refs":["branches","tags"]
},
"except":null
}
]
2021-01-03 14:25:43 +05:30
}
```
2021-01-29 00:20:46 +05:30
## Validate a CI YAML configuration with a namespace
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/231352) in GitLab 13.6.
Checks if CI/CD YAML configuration is valid. This endpoint has namespace
specific context.
```plaintext
POST /projects/:id/ci/lint
```
| Attribute | Type | Required | Description |
| ---------- | ------- | -------- | -------- |
| `content` | string | yes | The CI/CD configuration content. |
2022-04-04 11:22:00 +05:30
| `dry_run` | boolean | no | Run [pipeline creation simulation ](../ci/lint.md#simulate-a-pipeline ), or only do static check. This is false by default. |
2021-12-11 22:18:48 +05:30
| `include_jobs` | boolean | no | If the list of jobs that would exist in a static check or pipeline simulation should be included in the response. This is false by default. |
2022-04-04 11:22:00 +05:30
| `ref` | string | no | When `dry_run` is `true` , sets the branch or tag to use. Defaults to the project's default branch when not set. |
2021-01-29 00:20:46 +05:30
Example request:
```shell
curl --header "Content-Type: application/json" "https://gitlab.example.com/api/v4/projects/:id/ci/lint" --data '{"content": "{ \"image\": \"ruby:2.6\", \"services\": [\"postgres\"], \"before_script\": [\"bundle install\", \"bundle exec rake db:create\"], \"variables\": {\"DB_NAME\": \"postgres\"}, \"types\": [\"test\", \"deploy\", \"notify\"], \"rspec\": { \"script\": \"rake spec\", \"tags\": [\"ruby\", \"postgres\"], \"only\": [\"branches\"]}}"}'
```
Example responses:
- Valid configuration:
```json
{
"valid": true,
"merged_yaml": "---\n:test_job:\n :script: echo 1\n",
"errors": [],
"warnings": []
}
```
- Invalid configuration:
```json
{
"valid": false,
"merged_yaml": "---\n:test_job:\n :script: echo 1\n",
"errors": [
"jobs config should contain at least one visible job"
],
"warnings": []
}
```
2021-01-03 14:25:43 +05:30
## Validate a project's CI configuration
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/231352) in GitLab 13.5.
Checks if a project's latest (`HEAD` of the project's default branch)
`.gitlab-ci.yml` configuration is valid. This endpoint uses all namespace
specific data available, including variables, local includes, and so on.
```plaintext
GET /projects/:id/ci/lint
```
| Attribute | Type | Required | Description |
| ---------- | ------- | -------- | -------- |
| `dry_run` | boolean | no | Run pipeline creation simulation, or only do static check. |
2021-12-11 22:18:48 +05:30
| `include_jobs` | boolean | no | If the list of jobs that would exist in a static check or pipeline simulation should be included in the response. This is false by default. |
2022-04-04 11:22:00 +05:30
| `ref` | string | no | When `dry_run` is `true` , sets the branch or tag to use. Defaults to the project's default branch when not set. |
2021-01-03 14:25:43 +05:30
Example request:
```shell
curl "https://gitlab.example.com/api/v4/projects/:id/ci/lint"
```
Example responses:
- Valid configuration:
```json
{
"valid": true,
"merged_yaml": "---\n:test_job:\n :script: echo 1\n",
"errors": [],
"warnings": []
}
```
- Invalid configuration:
```json
{
"valid": false,
"merged_yaml": "---\n:test_job:\n :script: echo 1\n",
"errors": [
"jobs config should contain at least one visible job"
],
"warnings": []
}
```
## Use jq to create and process YAML & JSON payloads
To `POST` a YAML configuration to the CI Lint endpoint, it must be properly escaped and JSON encoded.
You can use `jq` and `curl` to escape and upload YAML to the GitLab API.
### Escape YAML for JSON encoding
To escape quotes and encode your YAML in a format suitable for embedding within
a JSON payload, you can use `jq` . For example, create a file named `example-gitlab-ci.yml` :
```yaml
.api_test:
rules:
- if: '$CI_PIPELINE_SOURCE=="merge_request_event"'
changes:
- src/api/*
deploy:
extends:
- .api_test
rules:
- when: manual
allow_failure: true
script:
- echo "hello world"
```
Next, use `jq` to escape and encode the YAML file into JSON:
```shell
jq --raw-input --slurp < example-gitlab-ci.yml
```
To escape and encode an input YAML file (`example-gitlab-ci.yml`), and `POST` it to the
GitLab API using `curl` and `jq` in a one-line command:
```shell
jq --null-input --arg yaml "$(< example-gitlab-ci.yml ) " ' . content = $yaml' \
2021-02-22 17:27:13 +05:30
| curl "https://gitlab.com/api/v4/ci/lint?include_merged_yaml=true" \
2021-01-03 14:25:43 +05:30
--header 'Content-Type: application/json' \
--data @-
```
### Parse a CI Lint response
To reformat the CI Lint response, you can use `jq` . You can pipe the CI Lint response to `jq` ,
or store the API response as a text file and provide it as an argument:
```shell
jq --raw-output '.merged_yaml | fromjson' < your_input_here >
```
Example input:
```json
{"status":"valid","errors":[],"merged_yaml":"---\n:.api_test:\n :rules:\n - :if: $CI_PIPELINE_SOURCE==\"merge_request_event\"\n :changes:\n - src/api/*\n:deploy:\n :rules:\n - :when: manual\n :allow_failure: true\n :extends:\n - \".api_test\"\n :script:\n - echo \"hello world\"\n"}
```
Becomes:
```yaml
:.api_test:
:rules:
- :if: $CI_PIPELINE_SOURCE=="merge_request_event"
:changes:
- src/api/*
:deploy:
:rules:
- :when: manual
:allow_failure: true
:extends:
- ".api_test"
:script:
- echo "hello world"
```
With a one-line command, you can:
1. Escape the YAML
1. Encode it in JSON
1. POST it to the API with curl
1. Format the response
```shell
jq --null-input --arg yaml "$(< example-gitlab-ci.yml ) " ' . content = $yaml' \
2021-02-22 17:27:13 +05:30
| curl "https://gitlab.com/api/v4/ci/lint?include_merged_yaml=true" \
2021-01-03 14:25:43 +05:30
--header 'Content-Type: application/json' --data @- \
| jq --raw-output '.merged_yaml | fromjson'
```