README.md 15.8 KB
Newer Older
1
# GitLab API
N
Nihad Abbasov 已提交
2

3
Automate GitLab via a simple and powerful API. All definitions can be found
4
under [`/lib/api`](https://gitlab.com/gitlab-org/gitlab-ce/tree/master/lib/api).
5

6
## Resources
7

8 9 10
Documentation for various API resources can be found separately in the
following locations:

11
- [Award Emoji](award_emoji.md)
12
- [Branches](branches.md)
13
- [Broadcast Messages](broadcast_messages.md)
S
Shinya Maeda 已提交
14 15
- [Project-level Variables](project_level_variables.md)
- [Group-level Variables](group_level_variables.md)
A
Achilleas Pipinellis 已提交
16
- [Commits](commits.md)
17
- [Custom Attributes](custom_attributes.md)
18
- [Deployments](deployments.md)
A
Achilleas Pipinellis 已提交
19
- [Deploy Keys](deploy_keys.md)
20
- [Environments](environments.md)
M
Mark Fletcher 已提交
21
- [Events](events.md)
R
Rémy Coutable 已提交
22
- [Feature flags](features.md)
23 24
- [Gitignores templates](templates/gitignores.md)
- [GitLab CI Config templates](templates/gitlab_ci_ymls.md)
A
Achilleas Pipinellis 已提交
25
- [Groups](groups.md)
26 27
- [Group Access Requests](access_requests.md)
- [Group Members](members.md)
28
- [Issues](issues.md)
29
- [Issue Boards](boards.md)
30
- [Jobs](jobs.md)
A
Achilleas Pipinellis 已提交
31
- [Keys](keys.md)
R
Robert Schilling 已提交
32
- [Labels](labels.md)
A
Achilleas Pipinellis 已提交
33
- [Merge Requests](merge_requests.md)
F
Felipe Artur 已提交
34 35
- [Project milestones](milestones.md)
- [Group milestones](group_milestones.md)
36
- [Namespaces](namespaces.md)
A
Achilleas Pipinellis 已提交
37
- [Notes](notes.md) (comments)
38
- [Notification settings](notification_settings.md)
39
- [Open source license templates](templates/licenses.md)
40
- [Pipelines](pipelines.md)
41
- [Pipeline Triggers](pipeline_triggers.md)
S
Add doc  
Shinya Maeda 已提交
42
- [Pipeline Schedules](pipeline_schedules.md)
A
Achilleas Pipinellis 已提交
43
- [Projects](projects.md) including setting Webhooks
44 45
- [Project Access Requests](access_requests.md)
- [Project Members](members.md)
A
Achilleas Pipinellis 已提交
46
- [Project Snippets](project_snippets.md)
E
Eric 已提交
47
- [Protected Branches](protected_branches.md)
A
Achilleas Pipinellis 已提交
48 49
- [Repositories](repositories.md)
- [Repository Files](repository_files.md)
50
- [Runners](runners.md)
A
Achilleas Pipinellis 已提交
51 52 53
- [Services](services.md)
- [Session](session.md)
- [Settings](settings.md)
54
- [Sidekiq metrics](sidekiq_metrics.md)
A
Achilleas Pipinellis 已提交
55 56
- [System Hooks](system_hooks.md)
- [Tags](tags.md)
57
- [Todos](todos.md)
58
- [Users](users.md)
59
- [Validate CI configuration](lint.md)
O
Oswaldo Ferreira 已提交
60
- [V3 to V4](v3_to_v4.md)
R
Robert Schilling 已提交
61
- [Version](version.md)
B
blackst0ne 已提交
62
- [Wikis](wikis.md)
A
Achilleas Pipinellis 已提交
63

64 65
## Road to GraphQL

66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
Going forward, we will start on moving to
[GraphQL](http://graphql.org/learn/best-practices/) and deprecate the use of
controller-specific endpoints. GraphQL has a number of benefits:

1. We avoid having to maintain two different APIs.
2. Callers of the API can request only what they need.
3. It is versioned by default.

It will co-exist with the current v4 REST API. If we have a v5 API, this should
be a compatibility layer on top of GraphQL.

Although there were some patenting and licensing concerns with GraphQL, these
have been resolved to our satisfaction by the relicensing of the reference
implementations under MIT, and the use of the OWF license for the GraphQL
specification.
81

82 83 84
## Basic usage

API requests should be prefixed with `api` and the API version. The API version
85 86
is defined in [`lib/api.rb`][lib-api-url]. For example, the root of the v4 API
is at `/api/v4`.
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113

For endpoints that require [authentication](#authentication), you need to pass
a `private_token` parameter via query string or header. If passed as a header,
the header name must be `PRIVATE-TOKEN` (uppercase and with a dash instead of
an underscore).

Example of a valid API request:

```
GET /projects?private_token=9koXpg98eAheJpvBs5tK
```

Example of a valid API request using cURL and authentication via header:

```shell
curl --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" "https://gitlab.example.com/api/v4/projects"
```

Example of a valid API request using cURL and authentication via a query string:

```shell
curl "https://gitlab.example.com/api/v4/projects?private_token=9koXpg98eAheJpvBs5tK"
```

The API uses JSON to serialize data. You don't need to specify `.json` at the
end of an API URL.

114
## Authentication
A
Achilleas Pipinellis 已提交
115

116 117 118
Most API requests require authentication via a session cookie or token. For
those cases where it is not required, this will be mentioned in the documentation
for each individual endpoint. For example, the [`/projects/:id` endpoint](projects.md).
119

120
There are three types of access tokens available:
121

122 123 124
1. [OAuth2 tokens](#oauth2-tokens)
1. [Private tokens](#private-tokens)
1. [Personal access tokens](#personal-access-tokens)
N
Nihad Abbasov 已提交
125

126 127
If authentication information is invalid or omitted, an error message will be
returned with status code `401`:
N
Nihad Abbasov 已提交
128 129 130 131 132 133 134

```json
{
  "message": "401 Unauthorized"
}
```

135
### Session cookie
136 137 138 139 140

When signing in to GitLab as an ordinary user, a `_gitlab_session` cookie is
set. The API will use this cookie for authentication if it is present, but using
the API to generate a new session cookie is currently not supported.

141
### OAuth2 tokens
N
Nihad Abbasov 已提交
142

143 144 145 146
You can use an OAuth 2 token to authenticate with the API by passing it either in the
`access_token` parameter or in the `Authorization` header.

Example of using the OAuth2 token in the header:
147

148
```shell
R
Robert Schilling 已提交
149
curl --header "Authorization: Bearer OAUTH-TOKEN" https://gitlab.example.com/api/v4/projects
150 151
```

152 153
Read more about [GitLab as an OAuth2 client](oauth2.md).

154
### Private tokens
N
Nihad Abbasov 已提交
155

156 157 158
Private tokens provide full access to the GitLab API. Anyone with access to
them can interact with GitLab as if they were you. You can find or reset your
private token in your account page (`/profile/account`).
V
Valery Sizov 已提交
159

160
For examples of usage, [read the basic usage section](#basic-usage).
V
Valery Sizov 已提交
161

162
### Personal access tokens
163

164 165 166
Instead of using your private token which grants full access to your account,
personal access tokens could be a better fit because of their granular
permissions.
167

168 169 170
Once you have your token, pass it to the API using either the `private_token`
parameter or the `PRIVATE-TOKEN` header. For examples of usage,
[read the basic usage section](#basic-usage).
171

172
[Read more about personal access tokens.][pat]
173

174
### Impersonation tokens
V
Valery Sizov 已提交
175

176
> [Introduced][ce-9099] in GitLab 9.0. Needs admin permissions.
V
Valery Sizov 已提交
177

178
Impersonation tokens are a type of [personal access token][pat]
179
that can only be created by an admin for a specific user.
180

181 182 183 184 185
They are a better alternative to using the user's password/private token
or using the [Sudo](#sudo) feature which also requires the admin's password
or private token, since the password/token can change over time. Impersonation
tokens are a great fit if you want to build applications or tools which
authenticate with the API as a specific user.
186

187
For more information, refer to the
188
[users API](users.md#retrieve-user-impersonation-tokens) docs.
189

190 191
For examples of usage, [read the basic usage section](#basic-usage).

192
### Sudo
193

194
> Needs admin permissions.
195

196
All API requests support performing an API call as if you were another user,
197
provided your private token is from an administrator account. You need to pass
198 199 200
the `sudo` parameter either via query string or a header with an ID/username of
the user you want to perform the operation as. If passed as a header, the
header name must be `SUDO` (uppercase).
201

202 203
If a non administrative `private_token` is provided, then an error message will
be returned with status code `403`:
204 205 206

```json
{
207
  "message": "403 Forbidden - Must be admin to use sudo"
208 209 210
}
```

211
If the sudo user ID or username cannot be found, an error message will be
212
returned with status code `404`:
213 214 215 216 217 218 219

```json
{
  "message": "404 Not Found: No user id or username for: <id/username>"
}
```

220 221 222 223
---

Example of a valid API call and a request using cURL with sudo request,
providing a username:
224

225
```
226
GET /projects?private_token=9koXpg98eAheJpvBs5tK&sudo=username
227
```
228

229
```shell
R
Robert Schilling 已提交
230
curl --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" --header "SUDO: username" "https://gitlab.example.com/api/v4/projects"
231 232
```

233 234
Example of a valid API call and a request using cURL with sudo request,
providing an ID:
235

236
```
237
GET /projects?private_token=9koXpg98eAheJpvBs5tK&sudo=23
238
```
239

240
```shell
R
Robert Schilling 已提交
241
curl --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" --header "SUDO: 23" "https://gitlab.example.com/api/v4/projects"
242
```
243

244 245 246 247 248 249 250 251 252 253 254 255
## Status codes

The API is designed to return different status codes according to context and
action. This way, if a request results in an error, the caller is able to get
insight into what went wrong.

The following table gives an overview of how the API functions generally behave.

| Request type | Description |
| ------------ | ----------- |
| `GET`   | Access one or more resources and return the result as JSON. |
| `POST`  | Return `201 Created` if the resource is successfully created and return the newly created resource as JSON. |
R
Robert Schilling 已提交
256 257
| `GET` / `PUT` | Return `200 OK` if the resource is accessed or modified successfully. The (modified) result is returned as JSON. |
| `DELETE` | Returns `204 No Content` if the resuource was deleted successfully. |
258 259 260 261 262 263 264 265 266 267 268 269 270 271 272

The following table shows the possible return codes for API requests.

| Return values | Description |
| ------------- | ----------- |
| `200 OK` | The `GET`, `PUT` or `DELETE` request was successful, the resource(s) itself is returned as JSON. |
| `204 No Content` | The server has successfully fulfilled the request and that there is no additional content to send in the response payload body. |
| `201 Created` | The `POST` request was successful and the resource is returned as JSON. |
| `304 Not Modified` | Indicates that the resource has not been modified since the last request. |
| `400 Bad Request` | A required attribute of the API request is missing, e.g., the title of an issue is not given. |
| `401 Unauthorized` | The user is not authenticated, a valid [user token](#authentication) is necessary. |
| `403 Forbidden` | The request is not allowed, e.g., the user is not allowed to delete a project. |
| `404 Not Found` | A resource could not be accessed, e.g., an ID for a resource could not be found. |
| `405 Method Not Allowed` | The request is not supported. |
| `409 Conflict` | A conflicting resource already exists, e.g., creating a project with a name that already exists. |
R
Robert Schilling 已提交
273
| `412` | Indicates the request was denied. May happen if the `If-Unmodified-Since` header is provided when trying to delete a resource, which was modified in between. |
274 275
| `422 Unprocessable` | The entity could not be processed. |
| `500 Server Error` | While handling the request something went wrong server-side. |
276

D
Dmitriy Zaporozhets 已提交
277
## Pagination
N
Nihad Abbasov 已提交
278

279 280
Sometimes the returned result will span across many pages. When listing
resources you can pass the following parameters:
N
Nihad Abbasov 已提交
281

282 283
| Parameter | Description |
| --------- | ----------- |
284 285
| `page`    | Page number (default: `1`) |
| `per_page`| Number of items to list per page (default: `20`, max: `100`) |
N
Nihad Abbasov 已提交
286

287 288 289
In the example below, we list 50 [namespaces](namespaces.md) per page.

```bash
R
Robert Schilling 已提交
290
curl --request PUT --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" "https://gitlab.example.com/api/v4/namespaces?per_page=50
291 292
```

293
### Pagination Link header
294

295
[Link headers](http://www.w3.org/wiki/LinkHeader) are sent back with each
296 297
response. They have `rel` set to prev/next/first/last and contain the relevant
URL. Please use these links instead of generating your own URLs.
298

299 300 301 302 303
In the cURL example below, we limit the output to 3 items per page (`per_page=3`)
and we request the second page (`page=2`) of [comments](notes.md) of the issue
with ID `8` which belongs to the project with ID `8`:

```bash
R
Robert Schilling 已提交
304
curl --head --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" https://gitlab.example.com/api/v4/projects/8/issues/8/notes?per_page=3&page=2
305 306
```

307 308 309 310 311 312 313 314
The response will then be:

```
HTTP/1.1 200 OK
Cache-Control: no-cache
Content-Length: 1103
Content-Type: application/json
Date: Mon, 18 Jan 2016 09:43:18 GMT
R
Robert Schilling 已提交
315
Link: <https://gitlab.example.com/api/v4/projects/8/issues/8/notes?page=1&per_page=3>; rel="prev", <https://gitlab.example.com/api/v4/projects/8/issues/8/notes?page=3&per_page=3>; rel="next", <https://gitlab.example.com/api/v4/projects/8/issues/8/notes?page=1&per_page=3>; rel="first", <https://gitlab.example.com/api/v4/projects/8/issues/8/notes?page=3&per_page=3>; rel="last"
316 317 318 319 320 321 322 323 324 325
Status: 200 OK
Vary: Origin
X-Next-Page: 3
X-Page: 2
X-Per-Page: 3
X-Prev-Page: 1
X-Request-Id: 732ad4ee-9870-4866-a199-a9db0cde3c86
X-Runtime: 0.108688
X-Total: 8
X-Total-Pages: 3
326 327 328 329
```

### Other pagination headers

330 331 332 333 334 335 336 337 338 339
Additional pagination headers are also sent back.

| Header | Description |
| ------ | ----------- |
| `X-Total`       | The total number of items |
| `X-Total-Pages` | The total number of pages |
| `X-Per-Page`    | The number of items per page |
| `X-Page`        | The index of the current page (starting at 1) |
| `X-Next-Page`   | The index of the next page |
| `X-Prev-Page`   | The index of the previous page |
340

341 342
## Namespaced path encoding

343
If using namespaced API calls, make sure that the `NAMESPACE/PROJECT_NAME` is
344 345
URL-encoded.

346
For example, `/` is represented by `%2F`:
347 348

```
349 350 351 352 353 354 355 356 357 358 359 360
GET /api/v4/projects/diaspora%2Fdiaspora
```

## Branches & tags name encoding

If your branch or tag contains a `/`, make sure the branch/tag name is
URL-encoded.

For example, `/` is represented by `%2F`:

```
GET /api/v4/projects/1/branches/my%2Fbranch/commits
361 362
```

363
## `id` vs `iid`
D
Dmitriy Zaporozhets 已提交
364

365 366 367 368
When you work with the API, you may notice two similar fields in API entities:
`id` and `iid`. The main difference between them is scope.

For example, an issue might have `id: 46` and `iid: 5`.
369

370 371
| Parameter | Description |
| --------- | ----------- |
372 373
| `id`  | Is unique across all issues and is used for any API call |
| `iid` | Is unique only in scope of a single project. When you browse issues or merge requests with the Web UI, you see the `iid` |
D
Dmitriy Zaporozhets 已提交
374

375
That means that if you want to get an issue via the API you should use the `id`:
376

377
```
378
GET /projects/42/issues/:id
379
```
D
Dmitriy Zaporozhets 已提交
380

381 382
On the other hand, if you want to create a link to a web page you should use
the `iid`:
D
Dmitriy Zaporozhets 已提交
383

384
```
385
GET /projects/42/issues/:iid
386
```
J
jubianchi 已提交
387 388 389

## Data validation and error reporting

390
When working with the API you may encounter validation errors, in which case
391 392
the API will answer with an HTTP `400` status.

J
jubianchi 已提交
393 394
Such errors appear in two cases:

395
- A required attribute of the API request is missing, e.g., the title of an
396
  issue is not given
397
- An attribute did not pass the validation, e.g., user bio is too long
J
jubianchi 已提交
398 399 400

When an attribute is missing, you will get something like:

401
```
402 403 404 405 406 407
HTTP/1.1 400 Bad Request
Content-Type: application/json
{
    "message":"400 (Bad request) \"title\" not given"
}
```
J
jubianchi 已提交
408

409 410
When a validation error occurs, error messages will be different. They will
hold all details of validation errors:
411

412
```
413 414 415 416 417 418 419
HTTP/1.1 400 Bad Request
Content-Type: application/json
{
    "message": {
        "bio": [
            "is too long (maximum is 255 characters)"
        ]
J
jubianchi 已提交
420
    }
421 422
}
```
J
jubianchi 已提交
423

424 425
This makes error messages more machine-readable. The format can be described as
follows:
J
jubianchi 已提交
426

427 428 429 430 431 432 433 434 435
```json
{
    "message": {
        "<property-name>": [
            "<error-message>",
            "<error-message>",
            ...
        ],
        "<embed-entity>": {
J
jubianchi 已提交
436 437 438 439 440 441 442
            "<property-name>": [
                "<error-message>",
                "<error-message>",
                ...
            ],
        }
    }
443 444 445
}
```

D
Dmitriy Zaporozhets 已提交
446
## Unknown route
447

D
Dmitriy Zaporozhets 已提交
448
When you try to access an API URL that does not exist you will receive 404 Not Found.
449 450

```
451
HTTP/1.1 404 Not Found
452 453
Content-Type: application/json
{
454
    "error": "404 Not Found"
455 456 457
}
```

458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474
## Encoding `+` in ISO 8601 dates

If you need to include a `+` in a query parameter, you may need to use `%2B` instead due
a [W3 recommendation]((http://www.w3.org/Addressing/URL/4_URI_Recommentations.html) that
causes a `+` to be interpreted as a space. For example, in an ISO 8601 date, you may want to pass
a time in Mountain Standard Time, such as:

```
2017-10-17T23:11:13.000+05:30
```

The correct encoding for the query parameter would be:

```
2017-10-17T23:11:13.000%2B05:30
```

475 476 477
## Clients

There are many unofficial GitLab API Clients for most of the popular
478
programming languages. Visit the [GitLab website] for a complete list.
479

480 481
[GitLab website]: https://about.gitlab.com/applications/#api-clients "Clients using the GitLab API"
[lib-api-url]: https://gitlab.com/gitlab-org/gitlab-ce/tree/master/lib/api/api.rb
482
[ce-3749]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/3749
483
[ce-5951]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/5951
484
[ce-9099]: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/9099
485
[pat]: ../user/profile/personal_access_tokens.md