2021-11-01 15:38:34 +00:00
# API for contributors
2024-12-17 21:39:24 +00:00
- [Authentication ](#authentication )
2022-09-22 21:41:57 +00:00
- [Packs ](#packs )
2023-04-11 13:18:32 +00:00
- [Mobile device management (MDM) ](#mobile-device-management-mdm )
2022-10-05 12:35:36 +00:00
- [Get or apply configuration files ](#get-or-apply-configuration-files )
2022-09-22 21:41:57 +00:00
- [Live query ](#live-query )
2022-12-05 16:35:45 +00:00
- [Trigger cron schedule ](#trigger-cron-schedule )
2022-09-22 21:41:57 +00:00
- [Device-authenticated routes ](#device-authenticated-routes )
2024-11-12 17:05:45 +00:00
- [Orbit-authenticated routes ](#orbit-authenticated-routes )
2022-09-22 21:41:57 +00:00
- [Downloadable installers ](#downloadable-installers )
- [Setup ](#setup )
2023-10-10 22:00:45 +00:00
- [Scripts ](#scripts )
2024-05-23 21:07:07 +00:00
- [Software ](#software )
2025-03-10 17:17:57 +00:00
- [Users ](#users )
2025-05-22 21:20:56 +00:00
- [Conditional access ](#conditional-access )
2022-09-22 21:41:57 +00:00
2024-08-06 21:34:11 +00:00
> These endpoints are used by the Fleet UI, Fleet Desktop, and `fleetctl` clients and frequently change to reflect current functionality.
2021-11-01 15:38:34 +00:00
2024-08-06 21:34:11 +00:00
This document includes the internal Fleet API routes that are helpful when developing or contributing to Fleet.
2023-09-29 22:03:29 +00:00
If you are interested in gathering information from Fleet in a production environment, please see the [public Fleet REST API documentation ](https://fleetdm.com/docs/using-fleet/rest-api ).
2021-11-01 15:38:34 +00:00
2024-12-17 21:39:24 +00:00
## Authentication
### Create session
`POST /api/v1/fleet/sessions`
#### Parameters
| Name | Type | In | Description |
| token | string | body | **Required** . The token retrieved from the magic link email. |
#### Response
See [the Log in endpoint ](https://fleetdm.com/docs/rest-api/rest-api#log-in ) for the current
successful response format.
2022-09-22 21:41:57 +00:00
## Packs
2022-10-26 23:26:49 +00:00
Scheduling queries in Fleet is the best practice for collecting data from hosts. To learn how to schedule queries, [check out the docs here ](https://fleetdm.com/docs/using-fleet/fleet-ui#schedule-a-query ).
2022-09-22 21:41:57 +00:00
The API routes to control packs are supported for backwards compatibility.
- [Create pack ](#create-pack )
- [Modify pack ](#modify-pack )
- [Get pack ](#get-pack )
- [List packs ](#list-packs )
- [Delete pack ](#delete-pack )
- [Delete pack by ID ](#delete-pack-by-id )
- [Get scheduled queries in a pack ](#get-scheduled-queries-in-a-pack )
- [Add scheduled query to a pack ](#add-scheduled-query-to-a-pack )
- [Get scheduled query ](#get-scheduled-query )
- [Modify scheduled query ](#modify-scheduled-query )
- [Delete scheduled query ](#delete-scheduled-query )
### Create pack
`POST /api/v1/fleet/packs`
#### Parameters
| Name | Type | In | Description |
| ----------- | ------ | ---- | ----------------------------------------------------------------------- |
| name | string | body | **Required** . The pack's name. |
| description | string | body | The pack's description. |
| host_ids | list | body | A list containing the targeted host IDs. |
| label_ids | list | body | A list containing the targeted label's IDs. |
| team_ids | list | body | _Available in Fleet Premium_ A list containing the targeted teams' IDs. |
#### Example
`POST /api/v1/fleet/packs`
##### Request query parameters
```json
{
"description": "Collects osquery data.",
"host_ids": [],
"label_ids": [6],
"name": "query_pack_1"
}
```
##### Default response
`Status: 200`
```json
{
"pack": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 17,
"name": "query_pack_1",
"description": "Collects osquery data.",
"query_count": 0,
"total_hosts_count": 223,
"host_ids": [],
"label_ids": [
6
],
"team_ids": []
}
}
```
### Modify pack
`PATCH /api/v1/fleet/packs/{id}`
#### Parameters
| Name | Type | In | Description |
| ----------- | ------- | ---- | ----------------------------------------------------------------------- |
| id | integer | path | **Required.** The pack's id. |
| name | string | body | The pack's name. |
| description | string | body | The pack's description. |
| host_ids | list | body | A list containing the targeted host IDs. |
| label_ids | list | body | A list containing the targeted label's IDs. |
| team_ids | list | body | _Available in Fleet Premium_ A list containing the targeted teams' IDs. |
#### Example
`PATCH /api/v1/fleet/packs/{id}`
##### Request query parameters
```json
{
"description": "MacOS hosts are targeted",
"host_ids": [],
"label_ids": [7]
}
```
##### Default response
`Status: 200`
```json
{
"pack": {
"created_at": "2021-01-25T22:32:45Z",
"updated_at": "2021-01-25T22:32:45Z",
"id": 17,
"name": "Title2",
"description": "MacOS hosts are targeted",
"query_count": 0,
"total_hosts_count": 110,
"host_ids": [],
"label_ids": [
7
],
"team_ids": []
}
}
```
### Get pack
`GET /api/v1/fleet/packs/{id}`
#### Parameters
| Name | Type | In | Description |
| ---- | ------- | ---- | ---------------------------- |
| id | integer | path | **Required.** The pack's id. |
#### Example
`GET /api/v1/fleet/packs/17`
##### Default response
`Status: 200`
```json
{
"pack": {
"created_at": "2021-01-25T22:32:45Z",
"updated_at": "2021-01-25T22:32:45Z",
"id": 17,
"name": "Title2",
"description": "MacOS hosts are targeted",
"disabled": false,
"type": null,
"query_count": 0,
"total_hosts_count": 110,
"host_ids": [],
"label_ids": [
7
],
"team_ids": []
}
}
```
### List packs
`GET /api/v1/fleet/packs`
#### Parameters
| Name | Type | In | Description |
| --------------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------- |
| order_key | string | query | What to order results by. Can be any column in the packs table. |
| order_direction | string | query | **Requires `order_key`** . The direction of the order given the order key. Options include `asc` and `desc` . Default is `asc` . |
#### Example
`GET /api/v1/fleet/packs`
##### Default response
`Status: 200`
```json
{
"packs": [
{
"created_at": "2021-01-05T21:13:04Z",
"updated_at": "2021-01-07T19:12:54Z",
"id": 1,
"name": "pack_number_one",
"description": "This pack has a description",
"disabled": true,
"query_count": 1,
"total_hosts_count": 53,
"host_ids": [],
"label_ids": [
8
],
2022-12-09 18:23:08 +00:00
"team_ids": []
2022-09-22 21:41:57 +00:00
},
{
"created_at": "2021-01-19T17:08:31Z",
"updated_at": "2021-01-19T17:08:31Z",
"id": 2,
"name": "query_pack_2",
"query_count": 5,
"total_hosts_count": 223,
"host_ids": [],
"label_ids": [
6
],
"team_ids": []
2022-12-09 18:23:08 +00:00
}
2022-09-22 21:41:57 +00:00
]
}
```
### Delete pack
Delete pack by name.
`DELETE /api/v1/fleet/packs/{name}`
#### Parameters
| Name | Type | In | Description |
| ---- | ------ | ---- | ------------------------------ |
| name | string | path | **Required.** The pack's name. |
#### Example
`DELETE /api/v1/fleet/packs/pack_number_one`
##### Default response
`Status: 200`
### Delete pack by ID
`DELETE /api/v1/fleet/packs/id/{id}`
#### Parameters
| Name | Type | In | Description |
| ---- | ------- | ---- | ---------------------------- |
| id | integer | path | **Required.** The pack's ID. |
#### Example
`DELETE /api/v1/fleet/packs/id/1`
##### Default response
`Status: 200`
### Get scheduled queries in a pack
`GET /api/v1/fleet/packs/{id}/scheduled`
#### Parameters
| Name | Type | In | Description |
| ---- | ------- | ---- | ---------------------------- |
| id | integer | path | **Required.** The pack's ID. |
#### Example
`GET /api/v1/fleet/packs/1/scheduled`
##### Default response
`Status: 200`
```json
{
"scheduled": [
{
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 49,
"pack_id": 15,
"name": "new_query",
"query_id": 289,
"query_name": "new_query",
"query": "SELECT * FROM osquery_info",
"interval": 456,
"snapshot": false,
"removed": true,
"platform": "windows",
"version": "4.6.0",
"shard": null,
"denylist": null
},
{
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 50,
"pack_id": 15,
"name": "new_title_for_my_query",
"query_id": 288,
"query_name": "new_title_for_my_query",
"query": "SELECT * FROM osquery_info",
"interval": 677,
"snapshot": true,
"removed": false,
"platform": "windows",
"version": "4.6.0",
"shard": null,
"denylist": null
},
{
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 51,
"pack_id": 15,
"name": "osquery_info",
"query_id": 22,
"query_name": "osquery_info",
"query": "SELECT i.*, p.resident_size, p.user_time, p.system_time, time.minutes AS counter FROM osquery_info i, processes p, time WHERE p.pid = i.pid;",
"interval": 6667,
"snapshot": true,
"removed": false,
"platform": "windows",
"version": "4.6.0",
"shard": null,
"denylist": null
}
]
}
```
### Add scheduled query to a pack
`POST /api/v1/fleet/schedule`
#### Parameters
| Name | Type | In | Description |
| -------- | ------- | ---- | ------------------------------------------------------------------------------------------------------------- |
| pack_id | integer | body | **Required.** The pack's ID. |
| query_id | integer | body | **Required.** The query's ID. |
| interval | integer | body | **Required.** The amount of time, in seconds, the query waits before running. |
| snapshot | boolean | body | **Required.** Whether the queries logs show everything in its current state. |
| removed | boolean | body | **Required.** Whether "removed" actions should be logged. |
| platform | string | body | The computer platform where this query will run (other platforms ignored). Empty value runs on all platforms. |
| shard | integer | body | Restrict this query to a percentage (1-100) of target hosts. |
| version | string | body | The minimum required osqueryd version installed on a host. |
#### Example
`POST /api/v1/fleet/schedule`
#### Request body
```json
{
"interval": 120,
"pack_id": 15,
"query_id": 23,
"removed": true,
"shard": null,
"snapshot": false,
"version": "4.5.0",
"platform": "windows"
}
```
##### Default response
`Status: 200`
```json
{
"scheduled": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 56,
"pack_id": 17,
"name": "osquery_events",
"query_id": 23,
"query_name": "osquery_events",
"query": "SELECT name, publisher, type, subscriptions, events, active FROM osquery_events;",
"interval": 120,
"snapshot": false,
"removed": true,
"platform": "windows",
"version": "4.5.0",
"shard": 10
}
}
```
### Get scheduled query
`GET /api/v1/fleet/schedule/{id}`
#### Parameters
| Name | Type | In | Description |
| ---- | ------- | ---- | --------------------------------------- |
| id | integer | path | **Required.** The scheduled query's ID. |
#### Example
`GET /api/v1/fleet/schedule/56`
##### Default response
`Status: 200`
```json
{
"scheduled": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"id": 56,
"pack_id": 17,
"name": "osquery_events",
"query_id": 23,
"query_name": "osquery_events",
"query": "SELECT name, publisher, type, subscriptions, events, active FROM osquery_events;",
"interval": 120,
"snapshot": false,
"removed": true,
"platform": "windows",
"version": "4.5.0",
"shard": 10,
"denylist": null
}
}
```
### Modify scheduled query
`PATCH /api/v1/fleet/schedule/{id}`
#### Parameters
| Name | Type | In | Description |
| -------- | ------- | ---- | ------------------------------------------------------------------------------------------------------------- |
| id | integer | path | **Required.** The scheduled query's ID. |
| interval | integer | body | The amount of time, in seconds, the query waits before running. |
| snapshot | boolean | body | Whether the queries logs show everything in its current state. |
| removed | boolean | body | Whether "removed" actions should be logged. |
| platform | string | body | The computer platform where this query will run (other platforms ignored). Empty value runs on all platforms. |
| shard | integer | body | Restrict this query to a percentage (1-100) of target hosts. |
| version | string | body | The minimum required osqueryd version installed on a host. |
#### Example
`PATCH /api/v1/fleet/schedule/56`
#### Request body
```json
{
"platform": ""
}
```
##### Default response
`Status: 200`
```json
{
"scheduled": {
"created_at": "2021-01-28T19:40:04Z",
"updated_at": "2021-01-28T19:40:04Z",
"id": 56,
"pack_id": 17,
"name": "osquery_events",
"query_id": 23,
"query_name": "osquery_events",
"query": "SELECT name, publisher, type, subscriptions, events, active FROM osquery_events;",
"interval": 120,
"snapshot": false,
"removed": true,
"platform": "",
"version": "4.5.0",
"shard": 10
}
}
```
### Delete scheduled query
`DELETE /api/v1/fleet/schedule/{id}`
#### Parameters
| Name | Type | In | Description |
| ---- | ------- | ---- | --------------------------------------- |
| id | integer | path | **Required.** The scheduled query's ID. |
#### Example
`DELETE /api/v1/fleet/schedule/56`
##### Default response
`Status: 200`
---
2023-04-11 13:18:32 +00:00
## Mobile device management (MDM)
2022-12-05 16:35:45 +00:00
2023-01-24 16:57:22 +00:00
> Only Fleet MDM specific endpoints are located within the root /mdm/ path.
2022-12-16 18:39:36 +00:00
The MDM endpoints exist to support the related command-line interface sub-commands of `fleetctl` , such as `fleetctl generate mdm-apple` and `fleetctl get mdm-apple` , as well as the Fleet UI.
2022-12-05 16:35:45 +00:00
2024-06-27 15:06:39 +00:00
- [Generate Apple Business Manager public key (ADE) ](#generate-apple-business-manager-public-key-ade )
2023-01-25 19:44:29 +00:00
- [Request Certificate Signing Request (CSR) ](#request-certificate-signing-request-csr )
2024-06-27 15:06:39 +00:00
- [Upload APNS certificate ](#upload-apns-certificate )
2024-09-20 17:21:52 +00:00
- [Add ABM token ](#add-abm-token )
2024-12-05 01:19:16 +00:00
- [Count ABM tokens ](#count-abm-tokens )
2024-06-27 15:06:39 +00:00
- [Turn off Apple MDM ](#turn-off-apple-mdm )
2024-09-20 17:21:52 +00:00
- [Update ABM token's teams ](#update-abm-tokens-teams )
- [Renew ABM token ](#renew-abm-token )
- [Delete ABM token ](#delete-abm-token )
- [Add VPP token ](#add-VPP-token )
- [Update VPP token's teams ](#update-vpp-tokens-teams )
- [Renew VPP token ](#renew-vpp-token )
- [Delete VPP token ](#delete-vpp-token )
2024-01-24 17:13:56 +00:00
- [Batch-apply MDM custom settings ](#batch-apply-mdm-custom-settings )
2024-10-01 21:09:33 +00:00
- [Batch-apply packages ](#batch-apply-packages )
- [Batch-apply App Store apps ](#batch-apply-app-store-apps )
- [Get token to download package ](#get-token-to-download-package )
- [Download package using a token ](#download-package-using-a-token )
2023-05-05 17:36:13 +00:00
- [Initiate SSO during DEP enrollment ](#initiate-sso-during-dep-enrollment )
- [Complete SSO during DEP enrollment ](#complete-sso-during-dep-enrollment )
2024-09-23 20:56:59 +00:00
- [Over the air enrollment ](#over-the-air-enrollment )
2023-05-31 13:24:22 +00:00
- [Preassign profiles to devices ](#preassign-profiles-to-devices )
- [Match preassigned profiles ](#match-preassigned-profiles )
2023-10-06 22:04:33 +00:00
- [Get FileVault statistics ](#get-filevault-statistics )
2024-08-09 17:00:21 +00:00
- [Upload VPP content token ](#upload-vpp-content-token )
- [Disable VPP ](#disable-vpp )
2024-11-12 17:05:33 +00:00
- [SCEP proxy ](#scep-proxy )
2025-03-19 15:03:02 +00:00
- [Get Android Enterprise signup URL ](#get-android-enterprise-signup-url )
- [Connect Android Enterprise ](#connect-android-enterprise )
- [Delete Android Enterprise ](#delete-android-enterprise )
2025-04-04 19:28:09 +00:00
- [Get Android enrollment token ](#get-android-enrollment-token )
2025-03-19 15:03:02 +00:00
- [Create Android enrollment token ](#create-android-enrollment-token )
- [Get Android Enterprise server-sent event ](#get-android-enterprise-server-sent-event )
- [Android Enterprise PubSub push endpoint ](#android-enterprise-pubsub-push-endpoint )
2024-08-09 17:00:21 +00:00
2023-01-23 23:05:24 +00:00
2024-06-27 15:06:39 +00:00
### Generate Apple Business Manager public key (ADE)
2022-12-16 18:39:36 +00:00
2024-06-27 15:06:39 +00:00
`GET /api/v1/fleet/mdm/apple/abm_public_key`
#### Example
2022-12-16 18:39:36 +00:00
2024-06-27 15:06:39 +00:00
`GET /api/v1/fleet/mdm/apple/abm_public_key`
##### Default response
`Status: 200`
```json
{
"public_key": "23K9LCBGG26gc2AjcmV9Kz="
}
```
### Request Certificate Signing Request (CSR)
`GET /api/v1/fleet/mdm/apple/request_csr`
2022-12-16 18:39:36 +00:00
#### Example
2024-06-27 15:06:39 +00:00
`GET /api/v1/fleet/mdm/apple/request_csr`
2022-12-16 18:39:36 +00:00
##### Default response
2024-06-27 15:06:39 +00:00
```
Status: 200
```
2023-09-22 21:57:40 +00:00
```json
2022-12-16 18:39:36 +00:00
{
2024-06-27 15:06:39 +00:00
"csr": "lKT5LCBJJ20gc2VjcmV0Cg="
2022-12-16 18:39:36 +00:00
}
```
2024-06-27 15:06:39 +00:00
### Upload APNS certificate
2023-01-25 19:44:29 +00:00
2024-06-27 15:06:39 +00:00
`POST /api/v1/fleet/mdm/apple/apns_certificate`
2023-01-25 19:44:29 +00:00
#### Parameters
2024-06-27 15:06:39 +00:00
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| certificate | file | form | *Required* The file conataining the APNS certificate (.pem) |
2023-01-25 19:44:29 +00:00
#### Example
2024-06-27 15:06:39 +00:00
`POST /api/v1/fleet/mdm/apple/apns_certificate`
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
```
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="certificate"; filename="apns_cert.pem"
Content-Type: application/octet-stream
< CERTIFICATE_DATA >
--------------------------f02md47480und42y
```
2023-01-25 19:44:29 +00:00
##### Default response
2024-06-27 15:06:39 +00:00
`Status: 200`
2024-09-20 17:21:52 +00:00
### Add ABM token
2024-06-27 15:06:39 +00:00
2024-09-20 17:21:52 +00:00
`POST /api/v1/fleet/abm_tokens`
2024-06-27 15:06:39 +00:00
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| token | file | form | *Required* The file containing the token (.p7m) from Apple Business Manager |
#### Example
2024-09-20 17:21:52 +00:00
`POST /api/v1/fleet/abm_tokens`
2024-06-27 15:06:39 +00:00
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
2023-01-25 19:44:29 +00:00
```
2024-06-27 15:06:39 +00:00
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="token"; filename="server_token_abm.p7m"
Content-Type: application/octet-stream
< TOKEN_DATA >
--------------------------f02md47480und42y
```
##### Default response
`Status: 200`
2024-09-20 17:21:52 +00:00
```json
"abm_token": {
"id": 1,
"apple_id": "apple@example.com",
"org_name": "Fleet Device Management Inc.",
"mdm_server_url": "https://example.com/mdm/apple/mdm",
"renew_date": "2024-10-20T00:00:00Z",
"terms_expired": false,
"macos_team": null,
"ios_team": null,
"ipados_team": null
}
```
2024-06-27 15:06:39 +00:00
2024-12-05 01:19:16 +00:00
### Count ABM tokens
`GET /api/v1/fleet/abm_tokens/count`
Get the number of ABM tokens on the Fleet server.
#### Parameters
None.
#### Example
`GET /api/v1/fleet/abm_tokens/count`
##### Default response
`Status: 200`
```json
{
"count": 1
}
```
2024-06-27 15:06:39 +00:00
### Turn off Apple MDM
`DELETE /api/v1/fleet/mdm/apple/apns_certificate`
#### Example
`DELETE /api/v1/fleet/mdm/apple/apns_certificate`
##### Default response
`Status: 204`
2024-09-20 17:21:52 +00:00
### Update ABM token's teams
`PATCH /api/v1/fleet/abm_tokens/:id/teams`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The ABM token's ID |
| macos_team_id | integer | body | macOS hosts are automatically added to this team in Fleet when they appear in Apple Business Manager. If not specified, defaults to "No team" |
| ios_team_id | integer | body | iOS hosts are automatically added to this team in Fleet when they appear in Apple Business Manager. If not specified, defaults to "No team" |
| ipados_team_id | integer | body | iPadOS hosts are automatically added to this team in Fleet when they appear in Apple Business Manager. If not specified, defaults to "No team" |
#### Example
`PATCH /api/v1/fleet/abm_tokens/1/teams`
##### Request body
```json
{
"macos_team_id": 1,
"ios_team_id": 2,
"ipados_team_id": 3
}
```
##### Default response
`Status: 200`
```json
"abm_token": {
"id": 1,
"apple_id": "apple@example.com",
"org_name": "Fleet Device Management Inc.",
"mdm_server_url": "https://example.com/mdm/apple/mdm",
"renew_date": "2024-11-29T00:00:00Z",
"terms_expired": false,
"macos_team": 1,
"ios_team": 2,
"ipados_team": 3
}
```
### Renew ABM token
`PATCH /api/v1/fleet/abm_tokens/:id/renew`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The ABM token's ID |
#### Example
`PATCH /api/v1/fleet/abm_tokens/1/renew`
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
```
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="token"; filename="server_token_abm.p7m"
Content-Type: application/octet-stream
< TOKEN_DATA >
--------------------------f02md47480und42y
```
##### Default response
`Status: 200`
```json
"abm_token": {
"id": 1,
"apple_id": "apple@example.com",
"org_name": "Fleet Device Management Inc.",
"mdm_server_url": "https://example.com/mdm/apple/mdm",
"renew_date": "2025-10-20T00:00:00Z",
"terms_expired": false,
"macos_team": null,
"ios_team": null,
"ipados_team": null
}
```
### Delete ABM token
2024-06-27 15:06:39 +00:00
2024-09-20 17:21:52 +00:00
`DELETE /api/v1/fleet/abm_tokens/:id`
2024-06-27 15:06:39 +00:00
2024-09-20 17:21:52 +00:00
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The ABM token's ID |
2024-06-27 15:06:39 +00:00
#### Example
2024-09-20 17:21:52 +00:00
`DELETE /api/v1/fleet/abm_tokens/1`
2024-06-27 15:06:39 +00:00
##### Default response
`Status: 204`
2023-02-17 15:28:28 +00:00
2024-09-20 17:21:52 +00:00
### Add VPP token
`POST /api/v1/fleet/vpp_tokens`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| token | file | form | *Required* The file containing the content token (.vpptoken) from Apple Business Manager |
#### Example
`POST /api/v1/fleet/vpp_tokens`
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
```
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="token"; filename="sToken_for_Acme.vpptoken"
Content-Type: application/octet-stream
< TOKEN_DATA >
--------------------------f02md47480und42y
```
##### Default response
`Status: 200`
```json
"vpp_token": {
"id": 1,
"org_name": "Fleet Device Management Inc.",
"location": "https://example.com/mdm/apple/mdm",
"renew_date": "2024-10-20T00:00:00Z",
"terms_expired": false,
"teams": null
}
```
### Update VPP token's teams
`PATCH /api/v1/fleet/vpp_tokens/:id/teams`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The ABM token's ID |
| team_ids | list | body | If you choose specific teams, App Store apps in this VPP account will only be available to install on hosts in these teams. If not specified, defaults to all teams. |
#### Example
`PATCH /api/v1/fleet/vpp_tokens/1/teams`
##### Request body
```json
{
"team_ids": [1, 2, 3]
}
```
##### Default response
`Status: 200`
```json
"vpp_token": {
"id": 1,
"org_name": "Fleet Device Management Inc.",
"location": "https://example.com/mdm/apple/mdm",
"renew_date": "2024-10-20T00:00:00Z",
"terms_expired": false,
2024-09-25 14:44:08 +00:00
"teams": [
{
"team_id": 1,
"name": "Team 1"
},
{
"team_id": 2,
"name": "Team 2"
},
{
"team_id": 2,
"name": "Team 3"
},
]
2024-09-20 17:21:52 +00:00
}
```
### Renew VPP token
`PATCH /api/v1/fleet/vpp_tokens/:id/renew`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The VPP token's ID |
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
```
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="token"; filename="sToken_for_Acme.vpptoken"
Content-Type: application/octet-stream
< TOKEN_DATA >
--------------------------f02md47480und42y
```
##### Default response
`Status: 200`
```json
"vpp_token": {
"id": 1,
"org_name": "Fleet Device Management Inc.",
"location": "https://example.com/mdm/apple/mdm",
"renew_date": "2025-10-20T00:00:00Z",
"terms_expired": false,
"teams": [1, 2, 3]
}
```
### Delete VPP token
`DELETE /api/v1/fleet/vpp_token/:id`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| id | integer | path | *Required* The VPP token's ID |
#### Example
`DELETE /api/v1/fleet/vpp_tokens/1`
##### Default response
`Status: 204`
2023-03-02 00:36:59 +00:00
2024-01-24 17:13:56 +00:00
### Batch-apply MDM custom settings
2023-02-15 18:01:44 +00:00
2024-01-24 17:13:56 +00:00
`POST /api/v1/fleet/mdm/profiles/batch`
2023-02-15 18:01:44 +00:00
#### Parameters
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
| Name | Type | In | Description |
| --------- | ------ | ----- | --------------------------------------------------------------------------------------------------------------------------------- |
2024-01-24 17:13:56 +00:00
| team_id | number | query | _Available in Fleet Premium_ The team ID to apply the custom settings to. Only one of `team_name` /`team_id` can be provided. |
| team_name | string | query | _Available in Fleet Premium_ The name of the team to apply the custom settings to. Only one of `team_name` /`team_id` can be provided. |
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
| dry_run | bool | query | Validate the provided profiles and return any validation errors, but do not apply the changes. |
2025-04-04 19:28:09 +00:00
| no_cache | bool | query | Do not use the cached version of Fleet's configuration. This parameter should only be used when the configuration was updated less than 1 second ago. |
2024-12-03 23:17:42 +00:00
| profiles | json | body | An array of objects, consisting of a `profile` base64-encoded .mobileconfig or JSON for macOS and XML (Windows) file, `labels_include_all` , `labels_include_any` , or `labels_exclude_any` array of strings (label names), and `name` display name (for Windows configuration profiles and macOS declaration profiles). |
2024-02-23 18:54:18 +00:00
2023-02-15 18:01:44 +00:00
2024-01-24 17:13:56 +00:00
If no team (id or name) is provided, the profiles are applied for all hosts (for _Fleet Free_ ) or for hosts that are not assigned to any team (for _Fleet Premium_ ). After the call, the provided list of `profiles` will be the active profiles for that team (or no team) - that is, any existing profile that is not part of that list will be removed, and an existing profile with the same payload identifier (macOS) as a new profile will be edited. If the list of provided `profiles` is empty, all profiles are removed for that team (or no team).
2023-02-15 18:01:44 +00:00
#### Example
2024-01-24 17:13:56 +00:00
`POST /api/v1/fleet/mdm/profiles/batch`
2023-02-15 18:01:44 +00:00
##### Default response
`204`
2023-05-05 17:36:13 +00:00
### Initiate SSO during DEP enrollment
This endpoint initiates the SSO flow, the response contains an URL that the client can use to redirect the user to initiate the SSO flow in the configured IdP.
`POST /api/v1/fleet/mdm/sso`
#### Parameters
None.
#### Example
`POST /api/v1/fleet/mdm/sso`
##### Default response
2023-09-22 21:57:40 +00:00
```json
2023-05-05 17:36:13 +00:00
{
"url": "https://idp-provider.com/saml?SAMLRequest=...",
}
```
### Complete SSO during DEP enrollment
This is the callback endpoint that the identity provider will use to send security assertions to Fleet. This is where Fleet receives and processes the response from the identify provider.
`POST /api/v1/fleet/mdm/sso/callback`
#### Parameters
| Name | Type | In | Description |
| ------------ | ------ | ---- | ----------------------------------------------------------- |
| SAMLResponse | string | body | **Required** . The SAML response from the identity provider. |
#### Example
`POST /api/v1/fleet/mdm/sso/callback`
##### Request body
```json
{
"SAMLResponse": "< SAML response from IdP > "
}
```
##### Default response
`Status: 302`
If the credentials are valid, the server redirects the client to the Fleet UI. The URL contains the following query parameters that can be used to complete the DEP enrollment flow:
2023-05-18 15:50:00 +00:00
- `enrollment_reference` a reference that must be passed along with `profile_token` to the endpoint to download an enrollment profile.
2023-05-05 17:36:13 +00:00
- `profile_token` is a token that can be used to download an enrollment profile (.mobileconfig).
- `eula_token` (optional) if an EULA was uploaded, this contains a token that can be used to view the EULA document.
2024-09-23 20:56:59 +00:00
### Over the air enrollment
This endpoint handles over the air (OTA) MDM enrollments
`POST /api/v1/fleet/ota_enrollment`
#### Parameters
| Name | Type | In | Description |
| ------------------- | ------ | ---- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| enroll_secret | string | url | **Required** Assigns the host to a team with a matching enroll secret |
| XML device response | XML | body | **Required** . The XML response from the device. Fields are documented [here ](https://developer.apple.com/library/archive/documentation/NetworkingInternet/Conceptual/iPhoneOTAConfiguration/ConfigurationProfileExamples/ConfigurationProfileExamples.html#//apple_ref/doc/uid/TP40009505-CH4-SW7 ) |
> Note: enroll secrets can contain special characters. Ensure any special characters are [properly escaped](https://developer.mozilla.org/en-US/docs/Glossary/Percent-encoding).
#### Example
`POST /api/v1/fleet/ota_enrollment?enroll_secret=0Z6IuKpKU4y7xl%2BZcrp2gPcMi1kKNs3p`
##### Default response
`Status: 200`
Per [the spec ](https://developer.apple.com/library/archive/documentation/NetworkingInternet/Conceptual/iPhoneOTAConfiguration/Introduction/Introduction.html#//apple_ref/doc/uid/TP40009505-CH1-SW1 ), the response is different depending on the signature of the XML device response:
- If the body is signed with a certificate that can be validated by our root SCEP certificate, it returns an enrollment profile.
- Otherwise, it returns a SCEP payload.
2023-05-31 13:24:22 +00:00
### Preassign profiles to devices
2025-04-18 20:47:11 +00:00
> The Puppet module API endpoints are deprecated as of Fleet 4.66. They are maintained for backwards compatibility.
2023-05-31 13:24:22 +00:00
_Available in Fleet Premium_
This endpoint stores a profile to be assigned to a host at some point in the future. The actual assignment happens when the [Match preassigned profiles ](#match-preassigned-profiles ) endpoint is called. The reason for this "pre-assign" step is to collect all profiles that are meant to be assigned to a host, and match the list of profiles to an existing team (or create one with that set of profiles if none exist) so that the host can be assigned to that team and inherit its list of profiles.
`POST /api/v1/fleet/mdm/apple/profiles/preassign`
#### Parameters
2023-07-13 18:00:45 +00:00
| Name | Type | In | Description |
| ------------ | ------- | ---- | ----------------------------------------------------------- |
| external_host_identifier | string | body | **Required** . The identifier of the host as generated by the external service (e.g. Puppet). |
| host_uuid | string | body | **Required** . The UUID of the host. |
| profile | string | body | **Required** . The base64-encoded .mobileconfig content of the MDM profile. |
| group | string | body | The group label associated with that profile. This information is used to generate team names if they need to be created. |
| exclude | boolean | body | Whether to skip delivering the profile to this host. |
2023-05-31 13:24:22 +00:00
#### Example
`POST /api/v1/fleet/mdm/apple/profiles/preassign`
##### Request body
```json
{
"external_host_identifier": "id-01234",
"host_uuid": "c0532a64-bec2-4cf9-aa37-96fe47ead814",
"profile": "< base64-encoded profile > ",
2023-07-13 18:00:45 +00:00
"group": "Workstations",
"exclude": false
2023-05-31 13:24:22 +00:00
}
```
##### Default response
`Status: 204`
2023-10-06 22:04:33 +00:00
### Get FileVault statistics
_Available in Fleet Premium_
Get aggregate status counts of disk encryption enforced on macOS hosts.
The summary can optionally be filtered by team id.
`GET /api/v1/fleet/mdm/apple/filevault/summary`
#### Parameters
| Name | Type | In | Description |
| ------------------------- | ------ | ----- | ------------------------------------------------------------------------- |
| team_id | string | query | _Available in Fleet Premium_ The team id to filter the summary. |
#### Example
Get aggregate status counts of Apple disk encryption profiles applying to macOS hosts enrolled to Fleet's MDM that are not assigned to any team.
`GET /api/v1/fleet/mdm/apple/filevault/summary`
##### Default response
`Status: 200`
```json
{
"verified": 123,
"verifying": 123,
"action_required": 123,
"enforcing": 123,
"failed": 123,
"removing_enforcement": 123
}
```
2023-05-31 13:24:22 +00:00
### Match preassigned profiles
2025-04-18 20:47:11 +00:00
> The Puppet module API endpoints are deprecated as of Fleet 4.66. They are maintained for backwards compatibility.
2023-05-31 13:24:22 +00:00
_Available in Fleet Premium_
2023-06-22 16:07:43 +00:00
This endpoint uses the profiles stored by the [Preassign profiles to devices ](#preassign-profiles-to-devices ) endpoint to match the set of profiles to an existing team if possible, creating one if none exists. It then assigns the host to that team so that it receives the associated profiles. It is meant to be called only once all desired profiles have been pre-assigned to the host.
2023-05-31 13:24:22 +00:00
`POST /api/v1/fleet/mdm/apple/profiles/match`
#### Parameters
| Name | Type | In | Description |
| ------------ | ------ | ---- | ----------------------------------------------------------- |
| external_host_identifier | string | body | **Required** . The identifier of the host as generated by the external service (e.g. Puppet). |
#### Example
`POST /api/v1/fleet/mdm/apple/profiles/match`
##### Request body
```json
{
"external_host_identifier": "id-01234"
}
```
##### Default response
`Status: 204`
2024-08-09 17:00:21 +00:00
### Upload VPP content token
`POST /api/v1/fleet/mdm/apple/vpp_token`
#### Parameters
| Name | Type | In | Description |
| ---- | ---- | -- | ----------- |
| token | file | form | *Required* The file containing the content token (.vpptoken) from Apple Business Manager |
#### Example
`POST /api/v1/fleet/mdm/apple/vpp_token`
##### Request header
```http
Content-Length: 850
Content-Type: multipart/form-data; boundary=------------------------f02md47480und42y
```
##### Request body
```http
--------------------------f02md47480und42y
Content-Disposition: form-data; name="token"; filename="sToken_for_Acme.vpptoken"
Content-Type: application/octet-stream
< TOKEN_DATA >
--------------------------f02md47480und42y
```
##### Default response
`Status: 200`
### Disable VPP
`DELETE /api/v1/fleet/mdm/apple/vpp_token`
#### Example
`DELETE /api/v1/fleet/mdm/apple/vpp_token`
##### Default response
`Status: 204`
2024-11-12 17:05:33 +00:00
### SCEP proxy
`/mdm/scep/proxy/{identifier}`
2025-03-31 18:50:05 +00:00
This endpoint is used to proxy SCEP requests to the configured SCEP server. It uses the [SCEP protocol ](https://datatracker.ietf.org/doc/html/rfc8894 ). The `identifier` is in the format `hostUUID,profileUUID,caName` (URL escaped).
2024-08-09 17:00:21 +00:00
2025-03-19 15:03:02 +00:00
### Get Android Enterprise signup URL
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
2025-04-04 19:28:09 +00:00
2025-03-19 15:03:02 +00:00
This endpoint is used to generate a URL, which opens Google's wizard to create Android Enterprise.
`GET /api/v1/fleet/android_enterprise/signup_url`
#### Example
`GET /api/v1/fleet/android_enterprise/signup_url`
##### Default response
`Status: 200`
```json
{
"android_enterprise_signup_url": "https://enterprise.google.com/signup/android/email?origin=android& thirdPartyToken=S7512150D1D59A3BK"
}
```
### Connect Android Enterprise
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
This endpoint is used to connect (bind) Android Enterprise to Fleet and to turn on Android MDM features.
`GET /api/v1/fleet/android_enterprise/connect/:token`
This is callback URL that will be open after user completes Google's signup flow. It will self-close and return user to settings page.
#### Parameters
| Name | Type | In | Description |
| ---- | ------ | ---- | ------------------------------------ |
| token | string | path | **Required.** The signup token associated with Android Enterprise in Fleet. |
| enterpriseToken | string | query | **Required.** The enterprise token that's returned from Google API. |
#### Example
`GET /api/v1/fleet/android_enterprise/connect/6177a9cb410ff61f20015ad?enterpriseToken=FEKXFy427_jz9Nfhq19SGDOKR2nZ4ZqhSAuYqOQw1B1G2OdBkQ5IDfSkLiO0rUqL8ptAXoa5_cZdh5GBRdyLj29m5A8DcZ1dptSp6YMNY6MQv0UiqcQqRC8D`
##### Default response
`Status: 200`
```
< html > <!-- self - closing page --> < / html >
```
### Delete Android Enterprise
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
This endpoint is used to delete Android Enterprise. Once deleted, hosts that belong to Android Enterprise will be un-enrolled and Android MDM features will be turned off.
`DELETE /api/v1/fleet/android_enterprise/`
#### Example
`DELETE /api/v1/fleet/android_enterprise`
##### Default response
`Status: 200`
### Create Android enrollment token
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
This endpoint is used to generate enrollment token and enrollment URL which opens wizard (settings app) to enroll Android host.
`POST /api/v1/fleet/android_enterprise/enrollment_token`
#### Parameters
| Name | Type | In | Description |
|---------------|--------|-------|-----------------------------------------------------|
| enroll_secret | string | query | **Required.** The enroll secret of a team in Fleet. |
#### Example
`POST /api/v1/fleet/android/enterprise/enrollment_token?enroll_secret=0Z6IuKpKU4y7xl%2BZcrp2gPcMi1kKNs3p`
##### Default response
`Status: 200`
```json
{
"android_enrollment_token": "OJDDNCYSEZPAUZZOXHDF",
"android_enrollment_url": "https://enterprise.google.com/android/enroll?et=OJDDNCYSEZPAUZZOXHDF"
}
```
### Get Android Enterprise server-sent event
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
2025-04-04 19:28:09 +00:00
2025-03-19 15:03:02 +00:00
This endpoint is used to get server-sent events (SSE) messages, so that UI know if Android Enterprise is created and bound to Fleet.
`GET /api/v1/fleet/android_enterprise/signup_sse`
#### Example
`GET /api/v1/fleet/android_enterprise/signup_sse`
##### Default response
`Status: 200`
```
Android Enterprise successfully connected
```
### Android Enterprise PubSub push endpoint
> **Experimental feature.** This feature is undergoing rapid improvement, which may result in breaking changes to the API or configuration surface. It is not recommended for use in automated workflows.
2025-04-04 19:28:09 +00:00
2025-03-19 15:03:02 +00:00
This endpoint is used by Google Pub/Sub subscription to push messages to Fleet.
`POST /api/v1/fleet/android_enterprise/pubsub`
2022-09-22 21:41:57 +00:00
## Get or apply configuration files
2022-10-26 23:26:49 +00:00
These API routes are used by the `fleetctl` CLI tool. Users can manage Fleet with `fleetctl` and [configuration files in YAML syntax ](https://fleetdm.com/docs/using-fleet/configuration-files/ ).
2022-09-22 21:41:57 +00:00
- [Get queries ](#get-queries )
- [Get query ](#get-query )
- [Apply queries ](#apply-queries )
2022-12-09 18:23:08 +00:00
- [Apply policies ](#apply-policies )
2022-09-22 21:41:57 +00:00
- [Get packs ](#get-packs )
- [Apply packs ](#apply-packs )
- [Get pack by name ](#get-pack-by-name )
- [Apply team ](#apply-team )
- [Apply labels ](#apply-labels )
- [Get labels ](#get-labels )
2022-12-09 18:23:08 +00:00
- [Get label ](#get-label )
2021-11-01 15:38:34 +00:00
- [Get enroll secrets ](#get-enroll-secrets )
- [Modify enroll secrets ](#modify-enroll-secrets )
2025-01-10 00:04:34 +00:00
- [Store secret variables ](#store-secret-variables )
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
### Get queries
2021-11-01 15:38:34 +00:00
Returns a list of all queries in the Fleet instance. Each item returned includes the name, description, and SQL of the query.
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/queries`
2021-11-01 15:38:34 +00:00
#### Parameters
None.
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/queries`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": [
{
"name": "query1",
"description": "query",
"query": "SELECT * FROM osquery_info"
},
{
"name": "osquery_schedule",
"description": "Report performance stats for each file in the query schedule.",
2022-05-03 14:47:31 +00:00
"query": "SELECT name, interval, executions, output_size, wall_time, (user_time/executions) AS avg_user_time, (system_time/executions) AS avg_system_time, average_memory, last_executed FROM osquery_schedule;"
2021-11-01 15:38:34 +00:00
}
2022-05-03 14:47:31 +00:00
]
2021-11-01 15:38:34 +00:00
}
```
2022-09-22 21:41:57 +00:00
### Get query
2021-11-01 15:38:34 +00:00
Returns the name, description, and SQL of the query specified by name.
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/queries/{name}`
2021-11-01 15:38:34 +00:00
#### Parameters
| Name | Type | In | Description |
| ---- | ------ | ---- | ------------------------------------ |
| name | string | path | **Required.** The name of the query. |
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/queries/query1`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": {
"name": "query1",
"description": "query",
"query": "SELECT * FROM osquery_info"
}
}
```
2022-09-22 21:41:57 +00:00
### Apply queries
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
Creates and/or modifies the queries included in the list. To modify an existing query, the name of the query must already be used by an existing query. If a query with the specified name doesn't exist in Fleet, a new query will be created.
2021-11-01 15:38:34 +00:00
2023-10-17 04:34:32 +00:00
If a query field is not specified in the "spec" then its default value depending on its type will be assumed, e.g. if `interval` is not set then `0` will be assumed, if `discard_data` is omitted then `false` will be assumed, etc.
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/queries`
2021-11-01 15:38:34 +00:00
#### Parameters
| Name | Type | In | Description |
| ----- | ---- | ---- | ---------------------------------------------------------------- |
| specs | list | body | **Required.** The list of the queries to be created or modified. |
2023-10-17 04:34:32 +00:00
For more information about the query fields, please refer to the [Create query endpoint ](https://fleetdm.com/docs/using-fleet/rest-api#create-query ).
2021-11-01 15:38:34 +00:00
#### Example
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/queries`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
"specs": [
{
"name": "new_query",
"description": "This will be a new query because a query with the name 'new_query' doesn't exist in Fleet.",
"query": "SELECT * FROM osquery_info"
},
{
"name": "osquery_schedule",
"description": "This queries description and SQL will be modified because a query with the name 'osquery_schedule' exists in Fleet.",
"query": "SELECT * FROM osquery_info"
}
]
}
```
##### Default response
`Status: 200`
2022-09-22 21:41:57 +00:00
### Get packs
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
Returns all packs in the Fleet instance.
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/packs`
2021-11-01 15:38:34 +00:00
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/packs`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": [
{
"id": 1,
"name": "pack_1",
"description": "Description",
"disabled": false,
"targets": {
2022-02-21 16:18:58 +00:00
"labels": ["All Hosts"],
"teams": null
2021-11-01 15:38:34 +00:00
},
"queries": [
{
"query": "new_query",
"name": "new_query",
"description": "",
"interval": 456,
"snapshot": false,
"removed": true,
"platform": "windows",
"version": "4.5.0"
},
{
"query": "new_title_for_my_query",
"name": "new_title_for_my_query",
"description": "",
"interval": 677,
"snapshot": true,
"removed": false,
"platform": "",
"version": ""
},
{
"query": "osquery_info",
"name": "osquery_info",
"description": "",
"interval": 6667,
"snapshot": true,
"removed": false,
"platform": "",
"version": ""
},
{
"query": "query1",
"name": "query1",
"description": "",
"interval": 7767,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
},
{
"query": "osquery_events",
"name": "osquery_events",
"description": "",
"interval": 454,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
},
{
"query": "osquery_events",
"name": "osquery_events-1",
"description": "",
"interval": 120,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
}
]
},
{
"id": 2,
"name": "pack_2",
"disabled": false,
"targets": {
2022-02-21 16:18:58 +00:00
"labels": null,
"teams": null
2021-11-01 15:38:34 +00:00
},
"queries": [
{
"query": "new_query",
"name": "new_query",
"description": "",
"interval": 333,
"snapshot": false,
"removed": true,
"platform": "windows",
"version": "4.5.0",
"shard": 10,
"denylist": null
}
]
}
]
}
```
2022-09-22 21:41:57 +00:00
### Apply policies
Creates and/or modifies the policies included in the list. To modify an existing policy, the name of the policy included in the list must already be used by an existing policy. If a policy with the specified name doesn't exist in Fleet, a new policy will be created.
NOTE: when updating a policy, team and platform will be ignored.
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
`POST /api/v1/fleet/spec/policies`
#### Parameters
| Name | Type | In | Description |
| ----- | ---- | ---- | ----------------------------------------------------------------- |
| specs | list | body | **Required.** The list of the policies to be created or modified. |
#### Example
`POST /api/v1/fleet/spec/policies`
##### Request body
```json
{
"specs": [
{
"name": "new policy",
"description": "This will be a new policy because a policy with the name 'new policy' doesn't exist in Fleet.",
"query": "SELECT * FROM osquery_info",
2024-09-23 20:56:59 +00:00
"team": "No team",
2022-12-09 18:23:08 +00:00
"resolution": "some resolution steps here",
"critical": false
2022-09-22 21:41:57 +00:00
},
{
"name": "Is FileVault enabled on macOS devices?",
"query": "SELECT 1 FROM disk_encryption WHERE user_uuid IS NOT “” AND filevault_status = ‘ on’ LIMIT 1;",
2024-09-23 20:56:59 +00:00
"team": "Workstations",
2022-09-22 21:41:57 +00:00
"description": "Checks to make sure that the FileVault feature is enabled on macOS devices.",
"resolution": "Choose Apple menu > System Preferences, then click Security & Privacy. Click the FileVault tab. Click the Lock icon, then enter an administrator name and password. Click Turn On FileVault.",
2022-12-09 18:23:08 +00:00
"platform": "darwin",
2024-10-17 22:51:52 +00:00
"critical": true,
"script_id": 123
},
{
"name": "Is Adobe Acrobat installed and up to date?",
"query": "SELECT 1 FROM apps WHERE name = 'Adobe Acrobat Reader.app' AND version_compare(bundle_short_version, '23.001.20687') >= 0;",
"team": "Workstations",
"description": "Checks to make sure that Adobe Acrobat is installed and up to date.",
"platform": "darwin",
"critical": false,
"software_title_id": 12
},
2022-09-22 21:41:57 +00:00
]
}
```
2024-10-17 22:51:52 +00:00
The fields `critical` , `script_id` , and `software_title_id` are available in Fleet Premium.
2022-12-09 18:23:08 +00:00
2022-09-22 21:41:57 +00:00
##### Default response
`Status: 200`
### Apply packs
Creates and/or modifies the packs included in the list.
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/packs`
2021-11-01 15:38:34 +00:00
#### Parameters
| Name | Type | In | Description |
| ----- | ---- | ---- | --------------------------------------------------------------------------------------------- |
| specs | list | body | **Required.** A list that includes the specs for each pack to be added to the Fleet instance. |
#### Example
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/packs`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
"specs": [
{
"id": 1,
"name": "pack_1",
"description": "Description",
"disabled": false,
"targets": {
2022-02-21 16:18:58 +00:00
"labels": ["All Hosts"],
"teams": null
2021-11-01 15:38:34 +00:00
},
"queries": [
{
"query": "new_query",
"name": "new_query",
"description": "",
"interval": 456,
"snapshot": false,
"removed": true
},
{
"query": "new_title_for_my_query",
"name": "new_title_for_my_query",
"description": "",
"interval": 677,
"snapshot": true,
"removed": false
},
{
"query": "osquery_info",
"name": "osquery_info",
"description": "",
"interval": 6667,
"snapshot": true,
"removed": false
},
{
"query": "query1",
"name": "query1",
"description": "",
"interval": 7767,
"snapshot": false,
"removed": true
},
{
"query": "osquery_events",
"name": "osquery_events",
"description": "",
"interval": 454,
"snapshot": false,
"removed": true
},
{
"query": "osquery_events",
"name": "osquery_events-1",
"description": "",
"interval": 120,
"snapshot": false,
"removed": true
}
]
},
{
"id": 2,
"name": "pack_2",
"disabled": false,
"targets": {
2022-02-21 16:18:58 +00:00
"labels": null,
"teams": null
2021-11-01 15:38:34 +00:00
},
"queries": [
{
"query": "new_query",
"name": "new_query",
"description": "",
"interval": 333,
"snapshot": false,
"removed": true,
"platform": "windows"
}
]
}
]
}
```
##### Default response
`Status: 200`
2022-09-22 21:41:57 +00:00
### Get pack by name
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
Returns a pack.
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/packs/{name}`
2021-11-01 15:38:34 +00:00
#### Parameters
| Name | Type | In | Description |
| ---- | ------ | ---- | ------------------------------ |
| name | string | path | **Required.** The pack's name. |
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/packs/pack_1`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": {
"id": 15,
"name": "pack_1",
"description": "Description",
"disabled": false,
"targets": {
2022-02-21 16:18:58 +00:00
"labels": ["All Hosts"],
"teams": null
2021-11-01 15:38:34 +00:00
},
"queries": [
{
"query": "new_title_for_my_query",
"name": "new_title_for_my_query",
"description": "",
"interval": 677,
"snapshot": true,
"removed": false,
"platform": "",
"version": ""
},
{
"query": "osquery_info",
"name": "osquery_info",
"description": "",
"interval": 6667,
"snapshot": true,
"removed": false,
"platform": "",
"version": ""
},
{
"query": "query1",
"name": "query1",
"description": "",
"interval": 7767,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
},
{
"query": "osquery_events",
"name": "osquery_events",
"description": "",
"interval": 454,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
},
{
"query": "osquery_events",
"name": "osquery_events-1",
"description": "",
"interval": 120,
"snapshot": false,
"removed": true,
"platform": "",
"version": ""
}
]
}
}
```
2022-09-22 21:41:57 +00:00
### Apply team
2021-11-01 15:38:34 +00:00
_Available in Fleet Premium_
If the `name` specified is associated with an existing team, this API route, completely replaces this team's existing `agent_options` and `secrets` with those that are specified.
If the `name` is not already associated with an existing team, this API route creates a new team with the specified `name` , `agent_options` , and `secrets` .
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/teams`
2021-11-01 15:38:34 +00:00
#### Parameters
2023-02-28 20:34:46 +00:00
| Name | Type | In | Description |
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
| ----------------------------------------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
2023-02-28 20:34:46 +00:00
| name | string | body | **Required.** The team's name. |
| agent_options | object | body | The agent options spec that is applied to the hosts assigned to the specified to team. These agent options completely override the global agent options specified in the [`GET /api/v1/fleet/config API route` ](#get-configuration ) |
| features | object | body | The features that are applied to the hosts assigned to the specified to team. These features completely override the global features specified in the [`GET /api/v1/fleet/config API route` ](#get-configuration ) |
2025-01-10 00:04:34 +00:00
| secrets | array | body | A list of plain text strings is used as the enroll secrets. Existing secrets are replaced with this list, or left unmodified if this list is empty. Note that there is a limit of 50 secrets allowed. |
2023-02-28 20:34:46 +00:00
| mdm | object | body | The team's MDM configuration options. |
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
| mdm.macos_updates | object | body | The OS updates macOS configuration options for Nudge. |
| mdm.macos_updates.minimum_version | string | body | The required minimum operating system version. |
| mdm.macos_updates.deadline | string | body | The required installation date for Nudge to enforce the operating system version. |
| mdm.macos_settings | object | body | The macOS-specific MDM settings. |
2025-01-10 00:04:34 +00:00
| mdm.macos_settings.custom_settings | array | body | The list of objects consists of a `path` to .mobileconfig or JSON file and `labels_include_all` , `labels_include_any` , or `labels_exclude_any` list of label names. |
2024-01-24 17:13:56 +00:00
| mdm.windows_settings | object | body | The Windows-specific MDM settings. |
2025-01-10 00:04:34 +00:00
| mdm.windows_settings.custom_settings | array | body | The list of objects consists of a `path` to XML files and `labels_include_all` , `labels_include_any` , or `labels_exclude_any` list of label names. |
| scripts | array | body | A list of script files to add to this team so they can be executed at a later time. |
2024-08-09 17:00:21 +00:00
| software | object | body | The team's software that will be available for install. |
2025-02-04 18:38:54 +00:00
| software.app_store_apps | array | body | An array of objects with values below. |
| software.app_store_apps.app_store_id | string | body | ID of the App Store app. |
| software.app_store_apps.self_service | boolean | body | Specifies whether or not end users can install self-service. |
| software.app_store_apps.labels_include_any | array | body | Specifies whether the app will only be available for install on hosts that **have any** of these labels. Only one of either `labels_include_any` or `labels_exclude_any` can be specified. |
| software.app_store_apps.labels_exclude_any | array | body | Specifies whether the app will only be available for install on hosts that **don't have any** of these labels. Only one of either `labels_include_any` or `labels_exclude_any` can be specified. |
2025-01-10 00:04:34 +00:00
| software.packages | array | body | An array of objects with values below. |
| software.packages.url | string | body | URL to the software package (PKG, MSI, EXE or DEB). |
| software.packages.install_script | string | body | Command that Fleet runs to install software. |
| software.packages.pre_install_query | string | body | Condition query that determines if the install will proceed. |
| software.packages.post_install_script | string | body | Script that runs after software install. |
| software.packages.uninstall_script | string | body | Command that Fleet runs to uninstall software. |
| software.packages.self_service | boolean | body | If `true` lists software in the self-service. |
| software.packages.labels_include_any | array | body | Target hosts that have any label in the array. Only one of `labels_include_any` or `labels_exclude_any` can be included. If neither are included, all hosts are targeted. |
| software.packages.labels_exclude_any | array | body | Target hosts that don't have any label in the array. Only one of `labels_include_any` or `labels_exclude_any` can be included. If neither are included, all hosts are targeted. |
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
| mdm.macos_settings.enable_disk_encryption | bool | body | Whether disk encryption should be enabled for hosts that belong to this team. |
2023-02-28 20:34:46 +00:00
| force | bool | query | Force apply the spec even if there are (ignorable) validation errors. Those are unknown keys and agent options-related validations. |
| dry_run | bool | query | Validate the provided JSON for unknown keys and invalid value types and return any validation errors, but do not apply the changes. |
2021-11-01 15:38:34 +00:00
#### Example
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/teams`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
"specs": [
{
"name": "Client Platform Engineering",
2022-08-30 11:13:09 +00:00
"features": {
"enable_host_users": false,
"enable_software_inventory": true,
"additional_queries": {
"foo": "SELECT * FROM bar;"
}
},
2021-11-01 15:38:34 +00:00
"agent_options": {
"spec": {
"config": {
"options": {
"pack_delimiter": "/",
"logger_tls_period": 10,
"distributed_plugin": "tls",
"disable_distributed": false,
2022-04-14 21:33:52 +00:00
"logger_tls_endpoint": "/api/v1/osquery/log",
2021-11-01 15:38:34 +00:00
"distributed_interval": 10,
"distributed_tls_max_attempts": 3
},
"decorators": {
"load": [
"SELECT uuid AS host_uuid FROM system_info;",
"SELECT hostname AS hostname FROM system_info;"
]
}
},
"overrides": {}
}
},
"secrets": [
{
"secret": "fTp52/twaxBU6gIi0J6PHp8o5Sm1k1kn"
},
{
"secret": "bhD5kiX2J+KBgZSk118qO61ZIdX/v8On"
}
2023-02-28 20:34:46 +00:00
],
"mdm": {
"macos_updates": {
"minimum_version": "12.3.1",
"deadline": "2023-12-01"
},
"macos_settings": {
2024-07-26 10:32:02 +00:00
"custom_settings": [
{
"path": "path/to/profile1.mobileconfig"
"labels_include_all": ["Label 1", "Label 2"]
},
{
"path": "path/to/profile2.json"
"labels_exclude_any": ["Label 3", "Label 4"]
},
],
2023-02-28 20:34:46 +00:00
"enable_disk_encryption": true
2023-12-04 13:41:37 +00:00
},
"windows_settings": {
2024-07-26 10:32:02 +00:00
"custom_settings": [
{
"path": "path/to/profile3.xml"
"labels_include_all": ["Label 1", "Label 2"]
}
]
2023-02-28 20:34:46 +00:00
}
2023-10-10 22:00:45 +00:00
},
"scripts": ["path/to/script.sh"],
2024-08-09 17:00:21 +00:00
"software": {
"packages": [
{
"url": "https://cdn.zoom.us/prod/5.16.10.26186/x64/ZoomInstallerFull.msi",
"pre_install_query": "SELECT 1 FROM macos_profiles WHERE uuid='c9f4f0d5-8426-4eb8-b61b-27c543c9d3db';",
"post_install_script": "sudo /Applications/Falcon.app/Contents/Resources/falconctl license 0123456789ABCDEFGHIJKLMNOPQRSTUV-WX",
"self_service": true,
}
],
"app_store_apps": [
{
"app_store_id": "12464567",
2024-10-01 21:09:33 +00:00
"self_service": true
2024-08-09 17:00:21 +00:00
}
]
}
2022-12-09 18:23:08 +00:00
}
2021-11-01 15:38:34 +00:00
]
}
```
#### Default response
`Status: 200`
2023-06-07 17:29:36 +00:00
```json
{
"team_ids_by_name": {
"Client Platform Engineering": 123
}
}
```
2022-09-22 21:41:57 +00:00
### Apply labels
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
Adds the supplied labels to Fleet. Each label requires the `name` , and `label_membership_type` properties.
2021-11-01 15:38:34 +00:00
If the `label_membership_type` is set to `dynamic` , the `query` property must also be specified with the value set to a query in SQL syntax.
If the `label_membership_type` is set to `manual` , the `hosts` property must also be specified with the value set to a list of hostnames.
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/labels`
2021-11-01 15:38:34 +00:00
#### Parameters
| Name | Type | In | Description |
| ----- | ---- | ---- | ------------------------------------------------------------------------------------------------------------- |
| specs | list | path | A list of the label to apply. Each label requires the `name` , `query` , and `label_membership_type` properties |
#### Example
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/labels`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
"specs": [
{
"name": "Ubuntu",
2022-05-12 16:32:54 +00:00
"description": "Filters Ubuntu hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'ubuntu';",
2021-11-01 15:38:34 +00:00
"label_membership_type": "dynamic"
},
{
"name": "local_machine",
"description": "Includes only my local machine",
"label_membership_type": "manual",
"hosts": ["snacbook-pro.local"]
}
]
}
```
##### Default response
`Status: 200`
2022-09-22 21:41:57 +00:00
### Get labels
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/labels`
2021-11-01 15:38:34 +00:00
#### Parameters
None.
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/labels`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": [
{
"id": 6,
"name": "All Hosts",
"description": "All hosts which have enrolled in Fleet",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1;",
2021-11-01 15:38:34 +00:00
"label_type": "builtin",
"label_membership_type": "dynamic"
},
{
"id": 7,
"name": "macOS",
"description": "All macOS hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'darwin';",
2021-11-01 15:38:34 +00:00
"platform": "darwin",
"label_type": "builtin",
"label_membership_type": "dynamic"
},
{
"id": 8,
"name": "Ubuntu Linux",
"description": "All Ubuntu hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'ubuntu';",
2021-11-01 15:38:34 +00:00
"platform": "ubuntu",
"label_type": "builtin",
"label_membership_type": "dynamic"
},
{
"id": 9,
"name": "CentOS Linux",
"description": "All CentOS hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'centos' OR name LIKE '%centos%'",
2021-11-01 15:38:34 +00:00
"label_type": "builtin",
"label_membership_type": "dynamic"
},
{
"id": 10,
"name": "MS Windows",
"description": "All Windows hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'windows';",
2021-11-01 15:38:34 +00:00
"platform": "windows",
"label_type": "builtin",
"label_membership_type": "dynamic"
},
{
"id": 11,
"name": "Ubuntu",
2022-05-12 16:32:54 +00:00
"description": "Filters Ubuntu hosts",
2022-05-03 14:47:31 +00:00
"query": "SELECT 1 FROM os_version WHERE platform = 'ubuntu';",
2021-11-01 15:38:34 +00:00
"label_membership_type": "dynamic"
}
]
}
```
2022-09-22 21:41:57 +00:00
### Get label
2021-11-01 15:38:34 +00:00
2022-09-22 21:41:57 +00:00
Returns the label specified by name.
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/labels/{name}`
2021-11-01 15:38:34 +00:00
#### Parameters
None.
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/labels/local_machine`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"specs": {
"id": 12,
"name": "local_machine",
"description": "Includes only my local machine",
"query": "",
"label_membership_type": "manual"
}
}
```
### Get enroll secrets
Returns the valid global enroll secrets.
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/enroll_secret`
2021-11-01 15:38:34 +00:00
#### Parameters
None.
#### Example
2022-04-14 21:33:52 +00:00
`GET /api/v1/fleet/spec/enroll_secret`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
```json
{
"spec": {
"secrets": [
{
"secret": "fTp52/twaxBU6gIi0J6PHp8o5Sm1k1kn",
"created_at": "2021-01-07T19:40:04Z"
},
{
"secret": "bhD5kiX2J+KBgZSk118qO61ZIdX/v8On",
"created_at": "2021-01-04T21:18:07Z"
}
]
}
}
```
### Modify enroll secrets
2022-05-12 16:32:54 +00:00
This replaces the active global enroll secrets with the secrets specified.
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/enroll_secret`
2021-11-01 15:38:34 +00:00
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ------- | ---- | ---- | ---------------------------------------------------------------------------------------------------------------- |
| secrets | list | body | **Required.** The plain text string used as the enroll secret. Note that there is a limit of 50 secrets allowed. |
2021-11-01 15:38:34 +00:00
#### Example
##### Request body
```json
{
"spec": {
"secrets": [
{
"secret": "fTp52/twaxBU6gIi0J6PHp8o5Sm1k1kn"
}
]
}
}
```
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/spec/enroll_secret`
2021-11-01 15:38:34 +00:00
##### Default response
`Status: 200`
2025-01-10 00:04:34 +00:00
### Store secret variables
Stores secret variables prefixed with `$FLEET_SECRET_` to Fleet.
`PUT /api/v1/fleet/spec/secret_variables`
#### Parameters
| Name | Type | In | Description |
| ------- | ---- | ---- | ---------------------------------------------------------------------------------------------------------------- |
| secrets | list | body | **Required.** List of objects consisting of fields: `name` and `value`
| dry_run | boolean | body | **Optional.** If true, validates the provided secrets and returns any validation errors, but does not apply the changes.
#### Example
`PUT /api/v1/fleet/spec/secret_variables`
##### Request body
```json
{
"secrets": [
{
"name": "FLEET_SECRET_SOME_API_TOKEN",
"value": "971ef02b93c74ca9b22b694a9251f1d6"
}
]
}
```
##### Default response
`Status: 200`
2022-09-22 21:41:57 +00:00
---
## Live query
These API routes are used by the Fleet UI.
- [Check live query status ](#check-live-query-status )
- [Check result store status ](#check-result-store-status )
- [Search targets ](#search-targets )
- [Count targets ](#count-targets )
- [Run live query ](#run-live-query )
- [Run live query by name ](#run-live-query-by-name )
2022-12-09 18:23:08 +00:00
- [Retrieve live query results (standard WebSocket API) ](#retrieve-live-query-results-standard-websocket-api )
- [Retrieve live query results (SockJS) ](#retrieve-live-query-results-sockjs )
2022-09-22 21:41:57 +00:00
### Check live query status
2022-11-30 17:57:42 +00:00
This checks the status of Fleet's ability to run a live query. If an error is present in the response, Fleet won't be able to run a live query successfully. The Fleet UI uses this endpoint to make sure that the Fleet instance is correctly configured to run live queries.
2022-09-22 21:41:57 +00:00
`GET /api/v1/fleet/status/live_query`
#### Parameters
None.
#### Example
`GET /api/v1/fleet/status/live_query`
##### Default response
`Status: 200`
### Check result store status
This checks Fleet's result store status. If an error is present in the response, Fleet won't be able to run a live query successfully. The Fleet UI uses this endpoint to make sure that the Fleet instance is correctly configured to run live queries.
`GET /api/v1/fleet/status/result_store`
#### Parameters
None.
#### Example
`GET /api/v1/fleet/status/result_store`
##### Default response
`Status: 200`
2022-06-10 18:29:45 +00:00
### Search targets
2022-06-14 18:55:59 +00:00
Accepts a search query and a list of host IDs to omit and returns a set of up to ten matching hosts. If
2022-06-10 18:29:45 +00:00
a query ID is provided and the referenced query allows observers to run, targets will include hosts
for which the user has an observer role.
`POST /api/latest/fleet/hosts/search`
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----------------- | ------- | ---- | ----------------------------------------------------------------------------------------------------------------------------------------------- |
2024-04-12 21:34:38 +00:00
| query | string | body | The query used to identify hosts to target. Searchable items include a `display_name` , `hostname` , `hardware_serial` , `uuid` or `primary_ip` . |
2023-01-30 21:35:56 +00:00
| query_id | integer | body | The saved query (if any) that will be run. The `observer_can_run` property on the query and the user's roles affect which targets are included. |
| excluded_host_ids | array | body | The list of host ids to omit from the search results. |
2022-06-10 18:29:45 +00:00
#### Example
2023-01-30 21:35:56 +00:00
`POST /api/v1/fleet/hosts/search`
2022-06-10 18:29:45 +00:00
##### Request body
```json
{
"query": "foo",
"query_id": 42,
"selected": {
"hosts": [],
"labels": [],
"teams": [1]
}
}
```
##### Default response
```json
{
"targets": {
"hosts": [
{
"created_at": "2021-02-03T16:11:43Z",
"updated_at": "2021-02-03T21:58:19Z",
"id": 1337,
"detail_updated_at": "2021-02-03T21:58:10Z",
"label_updated_at": "2021-02-03T21:58:10Z",
"last_enrolled_at": "2021-02-03T16:11:43Z",
"seen_time": "2021-02-03T21:58:20Z",
"hostname": "foof41482833",
"uuid": "a2064cef-0000-0000-afb9-283e3c1d487e",
"platform": "rhel",
"osquery_version": "4.5.1",
"os_version": "CentOS 6.10.0",
"build": "",
"platform_like": "rhel",
"code_name": "",
"uptime": 32688000000000,
"memory": 2086899712,
"cpu_type": "x86_64",
"cpu_subtype": "142",
"cpu_brand": "Intel(R) Core(TM) i5-8279U CPU @ 2.40GHz",
"cpu_physical_cores": 4,
"cpu_logical_cores": 4,
"hardware_vendor": "",
"hardware_model": "",
"hardware_version": "",
"hardware_serial": "",
"computer_name": "foof41482833",
2022-10-08 12:57:46 +00:00
"display_name": "foof41482833",
2022-06-10 18:29:45 +00:00
"primary_ip": "172.20.0.3",
"primary_mac": "02:42:ac:14:00:03",
"distributed_interval": 10,
"config_tls_refresh": 10,
"logger_tls_period": 10,
"additional": {},
"status": "offline",
"display_text": "foof41482833"
}
]
}
}
```
### Count targets
Counts the number of online and offline hosts included in a given set of selected targets.
`POST /api/latest/fleet/targets/count`
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| -------- | ------- | ---- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| query_id | integer | body | The saved query (if any) that will be run. The `observer_can_run` property on the query and the user's roles determine which targets are included. |
2024-03-07 20:23:18 +00:00
| selected | object | body | The object includes lists of selected host IDs (`selected.hosts`), label IDs (`selected.labels`), and team IDs (`selected.teams`). When provided, builtin label IDs, custom label IDs and team IDs become `AND` filters. Within each selector, selecting two or more teams, two or more builtin labels, or two or more custom labels, behave as `OR` filters. There's one special case for the builtin label "All hosts", if such label is selected, then all other label and team selectors are ignored (and all hosts will be selected). If a host ID is explicitly included in `selected.hosts` , then it is assured that the query will be selected to run on it (no matter the contents of `selected.labels` and `selected.teams` ). Use `0` team ID to filter by hosts assigned to "No team". See examples below. |
2022-06-10 18:29:45 +00:00
#### Example
`POST /api/latest/fleet/targets/count`
##### Request body
```json
{
"query_id": 1337,
"selected": {
"hosts": [],
"labels": [42],
"teams": []
}
}
```
##### Default response
```json
{
"targets_count": 813,
"targets_offline": 813,
"targets_online": 0
}
```
2021-11-01 15:38:34 +00:00
### Run live query
2022-05-31 13:54:03 +00:00
Runs the specified query as a live query on the specified hosts or group of hosts and returns a new live query campaign. Individual hosts must be specified with the host's ID. Label IDs also specify groups of hosts.
2021-11-01 15:38:34 +00:00
2022-05-31 13:54:03 +00:00
After you initiate the query, [get results via WebSocket ](#retrieve-live-query-results-standard-websocket-api ).
2021-11-01 15:38:34 +00:00
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/queries/run`
2021-11-01 15:38:34 +00:00
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| -------- | ------- | ---- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| query | string | body | The SQL if using a custom query. |
| query_id | integer | body | The saved query (if any) that will be run. Required if running query as an observer. The `observer_can_run` property on the query effects which targets are included. |
2024-03-07 20:23:18 +00:00
| selected | object | body | **Required.** The object includes lists of selected host IDs (`selected.hosts`), label IDs (`selected.labels`), and team IDs (`selected.teams`). When provided, builtin label IDs, custom label IDs and team IDs become `AND` filters. Within each selector, selecting two or more teams, two or more builtin labels, or two or more custom labels, behave as `OR` filters. There's one special case for the builtin label "All hosts", if such label is selected, then all other label and team selectors are ignored (and all hosts will be selected). If a host ID is explicitly included in `selected.hosts` , then it is assured that the query will be selected to run on it (no matter the contents of `selected.labels` and `selected.teams` ). Use `0` team ID to filter by hosts assigned to "No team". See examples below. |
2021-11-01 15:38:34 +00:00
One of `query` and `query_id` must be specified.
#### Example with one host targeted by ID
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/queries/run`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
2022-05-03 14:47:31 +00:00
"query": "SELECT instance_id FROM system_info",
2021-11-01 15:38:34 +00:00
"selected": {
"hosts": [171]
}
}
```
##### Default response
`Status: 200`
```json
{
"campaign": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"Metrics": {
"TotalHosts": 1,
"OnlineHosts": 0,
"OfflineHosts": 1,
"MissingInActionHosts": 0,
"NewHosts": 1
},
"id": 1,
"query_id": 3,
"status": 0,
"user_id": 1
}
}
```
#### Example with multiple hosts targeted by label ID
2022-04-14 21:33:52 +00:00
`POST /api/v1/fleet/queries/run`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
2022-05-03 14:47:31 +00:00
"query": "SELECT instance_id FROM system_info;",
2021-11-01 15:38:34 +00:00
"selected": {
"labels": [7]
}
}
```
##### Default response
`Status: 200`
```json
{
"campaign": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"Metrics": {
"TotalHosts": 102,
"OnlineHosts": 0,
"OfflineHosts": 24,
"MissingInActionHosts": 0,
"NewHosts": 0
},
"id": 2,
"query_id": 3,
"status": 0,
"user_id": 1
}
}
```
### Run live query by name
Runs the specified saved query as a live query on the specified targets. Returns a new live query campaign. Individual hosts must be specified with the host's hostname. Groups of hosts are specified by label name.
After the query has been initiated, [get results via WebSocket ](#retrieve-live-query-results-standard-websocket-api ).
2024-12-17 21:39:24 +00:00
`POST /api/v1/fleet/queries/run_by_identifiers`
2021-11-01 15:38:34 +00:00
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| -------- | ------- | ---- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| query | string | body | The SQL of the query. |
| query_id | integer | body | The saved query (if any) that will be run. The `observer_can_run` property on the query effects which targets are included. |
2024-12-17 21:39:24 +00:00
| selected | object | body | **Required.** The object includes lists of selected hostnames (`selected.hosts`), label names (`labels`). When provided, builtin label names and custom label names become `AND` filters. Within each selector, selecting two or more builtin labels, or two or more custom labels, behave as `OR` filters. If a label provided could not be found in the database, a 400 bad request will be returned specifying which label is invalid. There's one special case for the builtin label `"All hosts"` , if such label is selected, then all other label and team selectors are ignored (and all hosts will be selected). If a host's hostname is explicitly included in `selected.hosts` , then it is assured that the query will be selected to run on it (no matter the contents of `selected.labels` ). See examples below. |
2021-11-01 15:38:34 +00:00
One of `query` and `query_id` must be specified.
#### Example with one host targeted by hostname
2024-12-12 19:55:49 +00:00
`POST /api/v1/fleet/queries/run_by_identifiers`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
"query_id": 1,
"selected": {
"hosts": ["macbook-pro.local"]
}
}
```
##### Default response
`Status: 200`
```json
{
"campaign": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"Metrics": {
"TotalHosts": 1,
"OnlineHosts": 0,
"OfflineHosts": 1,
"MissingInActionHosts": 0,
"NewHosts": 1
},
"id": 1,
"query_id": 3,
"status": 0,
"user_id": 1
}
}
```
#### Example with multiple hosts targeted by label name
2024-12-12 19:55:49 +00:00
`POST /api/v1/fleet/queries/run_by_identifiers`
2021-11-01 15:38:34 +00:00
##### Request body
```json
{
2022-05-03 14:47:31 +00:00
"query": "SELECT instance_id FROM system_info",
2021-11-01 15:38:34 +00:00
"selected": {
"labels": ["All Hosts"]
}
}
```
##### Default response
`Status: 200`
```json
{
"campaign": {
"created_at": "0001-01-01T00:00:00Z",
"updated_at": "0001-01-01T00:00:00Z",
"Metrics": {
"TotalHosts": 102,
"OnlineHosts": 0,
"OfflineHosts": 24,
"MissingInActionHosts": 0,
"NewHosts": 1
},
"id": 2,
"query_id": 3,
"status": 0,
"user_id": 1
}
}
```
2024-12-12 19:55:49 +00:00
#### Example with invalid label
`POST /api/v1/fleet/queries/run_by_identifiers`
##### Request body
```json
{
"query": "SELECT instance_id FROM system_info",
"selected": {
"labels": ["Windows", "Banana", "Apple"]
}
}
```
##### Default response
`Status: 400`
```json
{
"message": "Bad request",
"errors": [
{
"name": "base",
"reason": "Invalid label name(s): Banana, Apple."
}
],
"uuid": "303649f4-5e45-4379-bae9-64ec0ef56287"
}
```
2021-11-01 15:38:34 +00:00
### Retrieve live query results (standard WebSocket API)
You can retrieve the results of a live query using the [standard WebSocket API ](#https://developer.mozilla.org/en-US/docs/Web/API/WebSockets_API/Writing_WebSocket_client_applications ).
Before you retrieve the live query results, you must create a live query campaign by running the live query. Use the [Run live query ](#run-live-query ) or [Run live query by name ](#run-live-query-by-name ) endpoints to create a live query campaign.
Note that live queries are automatically cancelled if this method is not called to start retrieving the results within 60 seconds of initiating the query.
2022-05-16 15:20:54 +00:00
`/api/v1/fleet/results/websocket`
2021-11-01 15:38:34 +00:00
### Parameters
| Name | Type | In | Description |
| ---------- | ------- | --- | ---------------------------------------------------------------- |
| token | string | | **Required.** The token used to authenticate with the Fleet API. |
| campaignID | integer | | **Required.** The ID of the live query campaign. |
### Example
#### Example script to handle request and response
2023-09-22 21:57:40 +00:00
```js
2022-05-16 15:20:54 +00:00
const socket = new WebSocket('wss://< your-base-url > /api/v1/fleet/results/websocket');
2021-11-01 15:38:34 +00:00
socket.onopen = () => {
socket.send(JSON.stringify({ type: 'auth', data: { token: < auth-token > } }));
socket.send(JSON.stringify({ type: 'select_campaign', data: { campaign_id: < campaign-id > } }));
};
socket.onmessage = ({ data }) => {
console.log(data);
const message = JSON.parse(data);
if (message.type === 'status' & & message.data.status === 'finished') {
socket.close();
}
}
```
### Detailed request and response walkthrough with example data
#### webSocket.onopen()
##### Response data
```json
o
```
#### webSocket.send()
##### Request data
```json
[
{
"type": "auth",
"data": { "token": < insert_token_here > }
}
]
```
```json
[
{
"type": "select_campaign",
"data": { "campaign_id": 12 }
}
]
```
#### webSocket.onmessage()
##### Response data
```json
// Sends the total number of hosts targeted and segments them by status
[
{
"type": "totals",
"data": {
"count": 24,
"online": 6,
"offline": 18,
"missing_in_action": 0
}
}
]
```
```json
// Sends the expected results, actual results so far, and the status of the live query
[
{
"type": "status",
"data": {
"expected_results": 6,
"actual_results": 0,
"status": "pending"
}
}
]
```
```json
// Sends the result for a given host
[
{
"type": "result",
"data": {
"distributed_query_execution_id": 39,
"host": {
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
"id": 42,
"hostname": "foobar",
"display_name": "foobar"
2021-11-01 15:38:34 +00:00
},
"rows": [
// query results data for the given host
],
"error": null
}
}
]
```
```json
// Sends the status of "finished" when messages with the results for all expected hosts have been sent
[
{
"type": "status",
"data": {
"expected_results": 6,
"actual_results": 6,
"status": "finished"
}
}
]
```
### Retrieve live query results (SockJS)
You can also retrieve live query results with a [SockJS client ](https://github.com/sockjs/sockjs-client ). The script to handle the request and response messages will look similar to the standard WebSocket API script with slight variations. For example, the constructor used for SockJS is `SockJS` while the constructor used for the standard WebSocket API is `WebSocket` .
Note that SockJS has been found to be substantially less reliable than the [standard WebSockets approach ](#retrieve-live-query-results-standard-websocket-api ).
`/api/v1/fleet/results/`
### Parameters
| Name | Type | In | Description |
| ---------- | ------- | --- | ---------------------------------------------------------------- |
| token | string | | **Required.** The token used to authenticate with the Fleet API. |
| campaignID | integer | | **Required.** The ID of the live query campaign. |
### Example
#### Example script to handle request and response
2023-09-22 21:57:40 +00:00
```js
2021-11-01 15:38:34 +00:00
const socket = new SockJS(`< your-base-url > /api/v1/fleet/results`, undefined, {});
socket.onopen = () => {
socket.send(JSON.stringify({ type: 'auth', data: { token: < token > } }));
socket.send(JSON.stringify({ type: 'select_campaign', data: { campaign_id: < campaignID > } }));
};
socket.onmessage = ({ data }) => {
console.log(data);
const message = JSON.parse(data);
if (message.type === 'status' & & message.data.status === 'finished') {
socket.close();
}
}
```
##### Detailed request and response walkthrough
#### socket.onopen()
##### Response data
```json
o
```
#### socket.send()
##### Request data
```json
[
{
"type": "auth",
"data": { "token": < insert_token_here > }
}
]
```
```json
[
{
"type": "select_campaign",
"data": { "campaign_id": 12 }
}
]
```
#### socket.onmessage()
##### Response data
```json
// Sends the total number of hosts targeted and segments them by status
[
{
"type": "totals",
"data": {
"count": 24,
"online": 6,
"offline": 18,
"missing_in_action": 0
}
}
]
```
```json
// Sends the expected results, actual results so far, and the status of the live query
[
{
"type": "status",
"data": {
"expected_results": 6,
"actual_results": 0,
"status": "pending"
}
}
]
```
```json
// Sends the result for a given host
[
{
"type": "result",
"data": {
"distributed_query_execution_id": 39,
"host": {
Reduce size of `DistributedQueryResult` to improve live query performance (#11882)
This was found while working on #10957.
When running a live query, a lot of unused host data is stored in Redis
and sent on every live query result message via websockets. The frontend
and fleetctl just need `id`, `hostname` and `display_name`. (This
becomes worse every time we add new fields to the `Host` struct.)
Sample of one websocket message result when running `SELECT * from
osquery_info;`:
size in `main`: 2234 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":57,\"host\":
{\"created_at\":\"2023-05-22T12:14:11Z\",\"updated_at\":\"2023-05-23T12:31:51Z\",
\"software_updated_at\":\"0001-01-01T00:00:00Z\",\"id\":106,\"detail_updated_at\":\"2023-05-23T11:50:04Z\",
\"label_updated_at\":\"2023-05-23T11:50:04Z\",\"policy_updated_at\":\"1970-01-02T00:00:00Z\",
\"last_enrolled_at\":\"2023-05-22T12:14:12Z\",
\"seen_time\":\"2023-05-23T09:52:23.876311-03:00\",\"refetch_requested\":false,
\"hostname\":\"lucass-macbook-pro.local\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"platform\":\"darwin\",\"osquery_version\":\"5.8.2\",\"os_version\":\"macOS 13.3.1\",\"build\":\"22E261\",\"platform_like\":\"darwin\",\"code_name\":\"\",
\"uptime\":91125000000000,\"memory\":34359738368,\"cpu_type\":\"x86_64h\",\"cpu_subtype\":\"Intel x86-64h Haswell\",\"cpu_brand\":\"Intel(R) Core(TM) i7-1068NG7 CPU @ 2.30GHz\",\"cpu_physical_cores\":4,\"cpu_logical_cores\":8,\"hardware_vendor\":\"Apple Inc.\",\"hardware_model\":\"MacBookPro16,2\",\"hardware_version\":\"1.0\",
\"hardware_serial\":\"0DPQR4HMD1FZ\",
\"computer_name\":\"Lucas’s MacBook Pro\",\"public_ip\":\"\",
\"primary_ip\":\"192.168.0.230\",\"primary_mac\":\"68:2f:67:8e:b6:1f\",
\"distributed_interval\":1,\"config_tls_refresh\":60,\"logger_tls_period\":10,\"team_id\":null,
\"pack_stats\":null,\"team_name\":null,
\"gigs_disk_space_available\":386.23,\"percent_disk_space_available\":40,
\"issues\":{\"total_issues_count\":0,\"failing_policies_count\":0},
\"mdm\":{\"enrollment_status\":null,\"server_url\":null,\"name\":\"\",\"encryption_key_available\":false},
\"status\":\"online\",\"display_text\":\"lucass-macbook-pro.local\",\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"b7ee9363a7c686e76e99ffb122e9c5241a791e69\",\"config_valid\":\"1\",
\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",\"start_time\":\"1684757652\",
\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",
\"version\":\"5.8.2\",\"watcher\":\"8364\"}],\"error\":null}}"]
```
vs. size of the message result on this branch: 675 bytes
```
a["{\"type\":\"result\",\"data\":{\"distributed_query_execution_id\":59,
\"host\":{\"id\":106,\"hostname\":\"lucass-macbook-pro.local\",
\"display_name\":\"Lucas’s MacBook Pro\"},
\"rows\":[{\"build_distro\":\"10.14\",\"build_platform\":\"darwin\",
\"config_hash\":\"f80dee827635db39077a458243379b3ad63311fd\",
\"config_valid\":\"1\",\"extensions\":\"active\",\"host_display_name\":\"Lucas’s MacBook Pro\",
\"host_hostname\":\"lucass-macbook-pro.local\",
\"instance_id\":\"cde5de81-344b-4c76-b1c5-dae964fdd4f2\",\"pid\":\"8370\",\"platform_mask\":\"21\",
\"start_time\":\"1684757652\",\"uuid\":\"BD4DFA10-E334-41D9-8136-D2163A8FE588\",\"version\":\"5.8.2\",
\"watcher\":\"8364\"}]}}"]
```
Manual tests included running with an old fleetctl running with a new
fleet server, and vice-versa, a new fleetctl running against an old
fleet server.
- [X] Changes file added for user-visible changes in `changes/` or
`orbit/changes/`.
See [Changes
files](https://fleetdm.com/docs/contributing/committing-changes#changes-files)
for more information.
- [X] Documented any API changes (docs/Using-Fleet/REST-API.md or
docs/Contributing/API-for-contributors.md)
- ~[ ] Documented any permissions changes~
- ~[ ] Input data is properly validated, `SELECT *` is avoided, SQL
injection is prevented (using placeholders for values in statements)~
- ~[ ] Added support on fleet's osquery simulator `cmd/osquery-perf` for
new osquery data ingestion features.~
- [X] Added/updated tests
- [X] Manual QA for all new/changed functionality
- ~For Orbit and Fleet Desktop changes:~
- ~[ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.~
- ~[ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).~
2023-05-25 11:11:53 +00:00
"id": 42,
"hostname": "foobar",
"display_name": "foobar"
2021-11-01 15:38:34 +00:00
},
"rows": [
// query results data for the given host
],
"error": null
}
}
]
```
```json
// Sends the status of "finished" when messages with the results for all expected hosts have been sent
[
{
"type": "status",
"data": {
"expected_results": 6,
"actual_results": 6,
"status": "finished"
}
}
]
```
2022-01-11 14:04:29 +00:00
2022-11-30 17:57:42 +00:00
---
## Trigger cron schedule
This API is used by the `fleetctl` CLI tool to make requests to trigger an ad hoc run of all jobs in
a specified cron schedule.
### Trigger
This makes a request to trigger the specified cron schedule. Upon receiving the request, the Fleet
server first checks the current status of the schedule, and it returns an error if a run is
currently pending.
`POST /api/latest/fleet/trigger`
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ---- | ------ | ----- | ----------------------------------------- |
2023-07-28 18:18:45 +00:00
| name | string | query | The name of the cron schedule to trigger. Supported trigger names are `apple_mdm_dep_profile_assigner` , `automations` , `cleanups_then_aggregation` , `integrations` , `mdm_apple_profile_manager` , `usage_statistics` , and `vulnerabilities` |
2024-01-10 22:29:51 +00:00
2022-11-30 17:57:42 +00:00
#### Example
`POST /api/latest/fleet/trigger?name=automations`
##### Default response
`Status: 200`
---
2022-12-05 16:35:45 +00:00
## Device-authenticated routes
2022-06-29 12:12:20 +00:00
2022-07-05 21:42:08 +00:00
Device-authenticated routes are routes used by the Fleet Desktop application. Unlike most other routes, Fleet user's API token does not authenticate them. They use a device-specific token.
2022-06-29 12:12:20 +00:00
2022-09-22 21:41:57 +00:00
- [Refetch device's host ](#refetch-devices-host )
- [Get device's Google Chrome profiles ](#get-devices-google-chrome-profiles )
- [Get device's mobile device management (MDM) and Munki information ](#get-devices-mobile-device-management-mdm-and-munki-information )
2024-05-23 21:07:07 +00:00
- [Get Fleet Desktop information ](#get-fleet-desktop-information )
- [Get device's software ](#get-devices-software )
2025-04-24 17:20:21 +00:00
- [Get device's software install results ](#get-devices-software-install-results )
- [Get device's software MDM command results ](#get-devices-software-mdm-command-results )
2025-05-13 17:14:45 +00:00
- [Install self-service software ](#install-self-service-software )
- [Uninstall software via self-service ](#uninstall-software-via-self-service )
2022-09-22 21:41:57 +00:00
- [Get device's policies ](#get-devices-policies )
2025-03-14 19:54:48 +00:00
- [Get device's certificate ](#get-devices-certificate )
2022-09-22 21:41:57 +00:00
- [Get device's API features ](#get-devices-api-features )
- [Get device's transparency URL ](#get-devices-transparency-url )
2023-01-16 15:22:12 +00:00
- [Download device's MDM manual enrollment profile ](#download-devices-mdm-manual-enrollment-profile )
2023-05-17 14:16:26 +00:00
- [Migrate device to Fleet from another MDM solution ](#migrate-device-to-fleet-from-another-mdm-solution )
2024-11-23 02:51:57 +00:00
- [Trigger Linux disk encryption escrow ](#trigger-linux-disk-encryption-escrow )
2023-08-24 16:04:27 +00:00
- [Report an agent error ](#report-an-agent-error )
2022-09-22 21:41:57 +00:00
2022-06-29 12:12:20 +00:00
#### Refetch device's host
2022-10-26 23:26:49 +00:00
Same as [Refetch host route ](https://fleetdm.com/docs/using-fleet/rest-api#refetch-host ) for the current device.
2022-06-29 12:12:20 +00:00
`POST /api/v1/fleet/device/{token}/refetch`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
#### Get device's Google Chrome profiles
2022-10-26 23:26:49 +00:00
Same as [Get host's Google Chrome profiles ](https://fleetdm.com/docs/using-fleet/rest-api#get-hosts-google-chrome-profiles ) for the current device.
2022-06-29 12:12:20 +00:00
`GET /api/v1/fleet/device/{token}/device_mapping`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
#### Get device's mobile device management (MDM) and Munki information
2022-10-26 23:26:49 +00:00
Same as [Get host's mobile device management and Munki information ](https://fleetdm.com/docs/using-fleet/rest-api#get-hosts-mobile-device-management-mdm-and-munki-information ) for the current device.
2022-06-29 12:12:20 +00:00
`GET /api/v1/fleet/device/{token}/macadmins`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
2023-12-13 20:31:48 +00:00
#### Ping Server with Device Token
Ping the server. OK response expected if the device token is still valid.
`HEAD /api/v1/fleet/device/{token}/ping`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
##### Example
`HEAD /api/v1/fleet/device/abcdef012456789/ping`
##### Default response
`Status: 200`
2022-09-12 19:37:38 +00:00
#### Get Fleet Desktop information
_Available in Fleet Premium_
2023-05-15 20:00:52 +00:00
Gets all information required by Fleet Desktop, this includes things like the number of failed policies or notifications to show/hide menu items.
2022-09-12 19:37:38 +00:00
`GET /api/v1/fleet/device/{token}/desktop`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-09-12 19:37:38 +00:00
##### Example
`GET /api/v1/fleet/device/abcdef012456789/desktop`
##### Default response
`Status: 200`
```json
{
2023-05-15 20:00:52 +00:00
"failing_policies_count": 3,
2024-08-02 19:49:20 +00:00
"self_service": true,
2023-05-15 20:00:52 +00:00
"notifications": {
2023-10-06 22:04:33 +00:00
"needs_mdm_migration": true,
"renew_enrollment_profile": false,
2023-06-20 18:06:45 +00:00
},
"config": {
"org_info": {
"org_name": "Fleet",
"org_logo_url": "https://example.com/logo.jpg",
2023-09-22 22:15:37 +00:00
"org_logo_url_light_background": "https://example.com/logo-light.jpg",
2023-06-20 18:06:45 +00:00
"contact_url": "https://fleetdm.com/company/contact"
},
"mdm": {
"macos_migration": {
"mode": "forced"
}
}
2023-05-15 20:00:52 +00:00
}
2022-09-12 19:37:38 +00:00
}
```
2023-05-15 20:00:52 +00:00
In regards to the `notifications` key:
- `needs_mdm_migration` means that the device fits all the requirements to allow the user to initiate an MDM migration to Fleet.
2023-05-23 18:01:04 +00:00
- `renew_enrollment_profile` means that the device is currently unmanaged from MDM but should be DEP enrolled into Fleet.
2022-09-12 19:37:38 +00:00
2024-05-23 21:07:07 +00:00
#### Get device's software
Lists the software installed on the current device.
`GET /api/v1/fleet/device/{token}/software`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2024-07-09 16:23:44 +00:00
| self_service | bool | query | Filter `self_service` software. |
2024-05-23 21:07:07 +00:00
| query | string | query | Search query keywords. Searchable fields include `name` . |
| page | integer | query | Page number of the results to fetch.|
| per_page | integer | query | Results per page.|
##### Example
`GET /api/v1/fleet/device/bbb7cdcc-f1d9-4b39-af9e-daa0f35728e8/software`
##### Default response
`Status: 200`
```json
{
"count": 2,
"software": [
{
"id": 121,
"name": "Google Chrome.app",
2024-08-09 17:00:21 +00:00
"software_package": {
"name": "GoogleChrome.pkg"
"version": "125.12.2"
"self_service": true,
2025-05-22 21:20:56 +00:00
"categories": ["Browsers"],
2024-08-09 17:00:21 +00:00
"last_install": {
"install_uuid": "8bbb8ac2-b254-4387-8cba-4d8a0407368b",
"installed_at": "2024-05-15T15:23:57Z"
},
},
"app_store_app": null,
2024-05-23 21:07:07 +00:00
"source": "apps",
"status": "failed",
"installed_versions": [
2024-08-20 14:51:36 +00:00
{
2024-05-23 21:07:07 +00:00
"version": "121.0",
"last_opened_at": "2024-04-01T23:03:07Z",
"vulnerabilities": ["CVE-2023-1234","CVE-2023-4321","CVE-2023-7654"],
"installed_paths": ["/Applications/Google Chrome.app"]
}
2024-08-20 14:51:36 +00:00
],
"software_package": {
"name": "google-chrome-124-0-6367-207.pkg",
"version": "121.0",
"self_service": true,
"icon_url": null,
"last_install": null
},
"app_store_app": null
2024-05-23 21:07:07 +00:00
},
{
"id": 143,
"name": "Firefox.app",
2024-08-09 17:00:21 +00:00
"software_package": null,
"app_store_app": null,
2024-05-23 21:07:07 +00:00
"source": "apps",
"status": null,
"installed_versions": [
2024-08-20 14:51:36 +00:00
{
2024-05-23 21:07:07 +00:00
"version": "125.6",
"last_opened_at": "2024-04-01T23:03:07Z",
"vulnerabilities": ["CVE-2023-1234","CVE-2023-4321","CVE-2023-7654"],
"installed_paths": ["/Applications/Firefox.app"]
}
2024-08-20 14:51:36 +00:00
],
"software_package": null,
"app_store_app": {
"app_store_id": "12345",
2025-05-22 21:20:56 +00:00
"categories": ["Browsers"],
2024-08-20 14:51:36 +00:00
"version": "125.6",
"self_service": false,
"icon_url": "https://example.com/logo-light.jpg",
"last_install": null
},
2024-05-23 21:07:07 +00:00
}
],
"meta": {
"has_next_results": false,
"has_previous_results": false
}
}
```
2025-04-24 17:20:21 +00:00
### Get device's software install results
_Available in Fleet Premium._
`GET /api/v1/fleet/device/{token}/software/install/{install_uuid}/results`
Get the results of a Fleet-maintained app or custom package install if it was performed on the authenticated host.
| Name | Type | In | Description |
| ---- | ------- | ---- | -------------------------------------------- |
| token | string | path | **Required** . The device's authentication token. |
| install_uuid | string | path | **Required** . The software installation UUID. |
#### Example
`GET /api/v1/fleet/device/22aada07-dc73-41f2-8452-c0987543fd29/software/install/b15ce221-e22e-4c6a-afe7-5b3400a017da/results`
##### Default response
`Status: 200`
```json
{
"install_uuid": "b15ce221-e22e-4c6a-afe7-5b3400a017da",
"software_title": "Falcon.app",
"software_title_id": 8353,
"software_package": "FalconSensor-6.44.pkg",
"host_id": 123,
"host_display_name": "Marko's MacBook Pro",
"status": "failed_install",
"output": "Installing software...\nError: The operation can’ t be completed because the item “Falcon” is in use.",
"pre_install_query_output": "Query returned result\nSuccess",
"post_install_script_output": "Running script...\nExit code: 1 (Failed)\nRolling back software install...\nSuccess"
}
```
### Get device's software MDM command results
This endpoint returns the results for a specific MDM command associated with a software (un)install.
`GET /api/v1/fleet/device/{token}/software/commands/{command_uuid}/results`
#### Parameters
| Name | Type | In | Description |
| ------------------------- | ------ | ----- | ------------------------------------------------------------------------- |
| token | string | path | **Required** . The device's authentication token. |
| command_uuid | string | path | The unique identifier of the command. |
#### Example
`GET /api/v1/fleet/device/22aada07-dc73-41f2-8452-c0987543fd29/software/commands/a2064cef-0000-1234-afb9-283e3c1d487e/results`
##### Default response
`Status: 200`
```json
{
"results": [
{
"host_uuid": "145cafeb-87c7-4869-84d5-e4118a927746",
"command_uuid": "a2064cef-0000-1234-afb9-283e3c1d487e",
"status": "Acknowledged",
"updated_at": "2023-04-04:00:00Z",
"request_type": "InstallApplication",
"hostname": "mycomputer",
"payload": "[base64]",
"result": "[base64]"
}
]
}
```
> Note: If the server has not yet received a result for a command, it will return an empty object (`{}`).
2024-07-09 16:23:44 +00:00
#### Install self-service software
Install self-service software on macOS, Windows, or Linux (Ubuntu) host. The software must have a `self_service` flag `true` to be installed.
`POST /api/v1/fleet/device/{token}/software/install/:software_title_id`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | **Required** . The device's authentication token. |
| software_title_id | string | path | **Required** . The software title's ID. |
##### Example
`POST /api/v1/fleet/device/22aada07-dc73-41f2-8452-c0987543fd29/software/install/123`
##### Default response
`Status: 202`
2024-05-23 21:07:07 +00:00
2025-05-13 17:14:45 +00:00
### Uninstall software via self-service
Uninstalls software from a host via the My device page.
`POST /api/v1/fleet/device/{token}/software/uninstall/:software_title_id`
#### Parameters
| Name | Type | In | Description |
| --------- | ---------- | ---- | -------------------------------------------- |
| token | string | path | **Required** . The device's authentication token. |
| software_title_id | integer | path | **Required** . The software title's ID. |
#### Example
`POST /api/v1/fleet/device/22aada07-dc73-41f2-8452-c0987543fd29/software/uninstall/123`
##### Default response
`Status: 202`
2022-06-29 12:12:20 +00:00
#### Get device's policies
_Available in Fleet Premium_
Lists the policies applied to the current device.
`GET /api/v1/fleet/device/{token}/policies`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
##### Example
`GET /api/v1/fleet/device/abcdef012456789/policies`
##### Default response
`Status: 200`
```json
{
"policies": [
{
"id": 1,
"name": "SomeQuery",
"query": "SELECT * FROM foo;",
"description": "this is a query",
"resolution": "fix with these steps...",
"platform": "windows,linux",
"response": "pass"
},
{
"id": 2,
"name": "SomeQuery2",
"query": "SELECT * FROM bar;",
"description": "this is another query",
"resolution": "fix with these other steps...",
"platform": "darwin",
"response": "fail"
},
{
"id": 3,
"name": "SomeQuery3",
"query": "SELECT * FROM baz;",
"description": "",
"resolution": "",
"platform": "",
"response": ""
}
]
}
2025-03-14 19:54:48 +00:00
```
#### Get device's certificates
Available for macOS, iOS, and iPadOS hosts only. Requires Fleet's MDM properly [enabled and configured ](https://fleetdm.com/docs/using-fleet/mdm-setup ).
Lists the certificates installed on the current device.
`GET /api/v1/fleet/device/{token}/certificates`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
| page | integer | query | Page number of the results to fetch.|
| per_page | integer | query | Results per page.|
| order_key | string | query | What to order results by. Options include `common_name` and `not_valid_after` . Default is `common_name` . |
| order_direction | string | query | **Requires `order_key`** . The direction of the order given the order key. Options include `asc` and `desc` . Default is `asc` . |
##### Example
`GET /api/v1/fleet/device/bbb7cdcc-f1d9-4b39-af9e-daa0f35728e8/certificates`
#### Default response
`Status: 200`
```json
{
"certificates": [
{
"id": 3,
"not_valid_after": "2021-08-19T02:02:17Z",
"not_valid_before": "2021-08-19T02:02:17Z",
"certificate_authority": true,
"common_name": "FleetDM",
"key_algorithm": "rsaEncryption",
"key_strength": 2048,
"key_usage": "CRL Sign, Key Cert Sign",
"serial": 1,
"signing_algorithm": "sha256WithRSAEncryption",
"subject": {
"country": "US",
"organization": "Fleet Device Management Inc.",
"organizational_unit": "Fleet Device Management Inc.",
"common_name": "FleetDM"
},
"issuer": {
"country": "US",
"organization": "Fleet Device Management Inc.",
"organizational_unit": "Fleet Device Management Inc.",
"common_name": "FleetDM"
}
}
],
"meta": {
"has_next_results": false,
"has_previous_results": false
}
}
2022-06-29 12:12:20 +00:00
```
#### Get device's API features
2022-07-05 21:42:08 +00:00
This supports the dynamic discovery of API features supported by the server for device-authenticated routes. This allows supporting different versions of Fleet Desktop and Fleet server instances (older or newer) while supporting the evolution of the API features. With this mechanism, an older Fleet Desktop can ignore features it doesn't know about, and a newer one can avoid requesting features about which the server doesn't know.
2022-06-29 12:12:20 +00:00
`GET /api/v1/fleet/device/{token}/api_features`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
##### Example
`GET /api/v1/fleet/device/abcdef012456789/api_features`
##### Default response
`Status: 200`
```json
{
"features": {}
}
```
#### Get device's transparency URL
2024-11-23 02:51:57 +00:00
Returns the URL to open when clicking the "About Fleet" menu item in Fleet Desktop. Note that _Fleet Premium_ is required to configure a custom transparency URL.
2022-06-29 12:12:20 +00:00
`GET /api/v1/fleet/device/{token}/transparency`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2022-06-29 12:12:20 +00:00
##### Example
`GET /api/v1/fleet/device/abcdef012456789/transparency`
##### Default response
`Status: 307`
Redirects to the transparency URL.
2023-01-16 15:22:12 +00:00
#### Download device's MDM manual enrollment profile
Downloads the Mobile Device Management (MDM) enrollment profile to install on the device for a manual enrollment into Fleet MDM.
`GET /api/v1/fleet/device/{token}/mdm/apple/manual_enrollment_profile`
##### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
2023-01-16 15:22:12 +00:00
##### Example
`GET /api/v1/fleet/device/abcdef012456789/mdm/apple/manual_enrollment_profile`
##### Default response
`Status: 200`
```xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
< plist version = "1.0" >
<!-- ... -->
< / plist >
```
2022-09-22 21:41:57 +00:00
---
2023-05-17 14:16:26 +00:00
#### Migrate device to Fleet from another MDM solution
Signals the Fleet server to send a webbook request with the device UUID and serial number to the webhook URL configured for MDM migration. **Requires Fleet Premium license**
`POST /api/v1/fleet/device/{token}/migrate_mdm`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
##### Example
`POST /api/v1/fleet/device/abcdef012456789/migrate_mdm`
##### Default response
`Status: 204`
---
2024-11-23 02:51:57 +00:00
### Trigger Linux disk encryption escrow
_Available in Fleet Premium_
Signals the fleet server to queue up the LUKS disk encryption escrow process (LUKS passphrase and slot key). If validation succeeds (disk encryption must be enforced for the team, the host's platform must be supported, the host's disk must already be encrypted, and the host's Orbit version must be new enough), this adds a notification flag for Orbit that, triggers escrow from the Orbit side.
`POST /api/v1/fleet/device/{token}/mdm/linux/trigger_escrow`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
| token | string | path | The device's authentication token. |
##### Example
`POST /api/v1/fleet/device/abcdef012456789/mdm/linux/trigger_escrow`
##### Default response
`Status: 204`
---
2023-08-24 16:04:27 +00:00
### Report an agent error
Notifies the server about an agent error, resulting in two outcomes:
- The error gets saved in Redis and can later be accessed using `fleetctl debug archive` .
2024-11-14 19:43:16 +00:00
> Note: to allow `fleetd` agents to use this endpoint, you need to set a [custom environment variable](./Configuration-for-contributors.md#fleet_enable_post_client_debug_errors). `fleetd` agents will always report vital errors to Fleet.
2023-10-20 04:49:18 +00:00
2023-08-24 16:04:27 +00:00
`POST /api/v1/fleet/device/{token}/debug/errors`
#### Parameters
2024-11-14 19:43:16 +00:00
| Name | Type | Description |
|-----------------------|----------|-------------------------------------------------------------------------------------------------------------------------------------------|
| error_source | string | Process name that error originated from ex. orbit, fleet-desktop |
| error_source_version | string | version of error_source |
2025-03-26 21:45:23 +00:00
| error_timestamp | datetime | Time in UTC that error occurred |
2024-11-14 19:43:16 +00:00
| error_message | string | error message |
| error_additional_info | obj | Any additional identifiers to assist debugging |
| vital | boolean | Whether the error is vital and should also be reported to Fleet via usage statistics. Do not put sensitive information into vital errors. |
2023-08-24 16:04:27 +00:00
##### Default response
2024-11-14 19:43:16 +00:00
`Status: 200`
2023-08-24 16:04:27 +00:00
2023-07-12 23:30:52 +00:00
---
2024-11-12 17:05:45 +00:00
## Orbit-authenticated routes
2024-11-23 02:51:57 +00:00
- [Escrow LUKS data ](#escrow-luks-data )
2024-11-12 17:05:45 +00:00
- [Get the status of a device in the setup experience ](#get-the-status-of-a-device-in-the-setup-experience )
2024-12-05 17:22:13 +00:00
- [Set or update device token ](#set-or-update-device-token )
- [Get orbit script ](#get-orbit-script )
- [Post orbit script result ](#post-orbit-script-result )
- [Put orbit device mapping ](#put-orbit-device-mapping )
- [Post orbit software install result ](#post-orbit-software-install-result )
- [Download software installer ](#download-software-installer )
- [Get orbit software install details ](#get-orbit-software-install-details )
- [Post disk encryption key ](#post-disk-encryption-key )
2024-11-12 17:05:45 +00:00
2024-11-23 02:51:57 +00:00
---
### Escrow LUKS data
`POST /api/fleet/orbit/luks_data`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
2024-12-05 17:22:13 +00:00
| orbit_node_key | string | body | The Orbit node key for authentication. |
2024-11-23 02:51:57 +00:00
| client_error | string | body | An error description if the LUKS key escrow process fails client-side. If provided, passphrase/salt/key slot request parameters are ignored and may be omitted. |
| passphrase | string | body | The LUKS passphrase generated for Fleet (the end user's existing passphrase is not transmitted) |
| key_slot | int | body | The LUKS key slot ID corresponding to the provided passphrase |
| salt | string | body | The salt corresponding to the specified LUKS key slot. Provided to track cases where an end user rotates LUKS credentials (at which point we'll no longer be able to decrypt data with the escrowed passphrase). |
##### Example
`POST /api/v1/fleet/orbit/luks_data`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"passphrase": "6e657665-7220676f-6e6e6120-67697665-20796f75-207570",
"salt": "d34db33f",
"key_slot": 1,
"client_error": ""
}
```
##### Default response
`Status: 204`
---
2024-11-12 17:05:45 +00:00
### Get the status of a device in the setup experience
`POST /api/fleet/orbit/setup_experience/status`
##### Parameters
| Name | Type | In | Description |
| ----- | ------ | ---- | ---------------------------------- |
2024-12-05 17:22:13 +00:00
| orbit_node_key | string | body | The Orbit node key for authentication. |
2024-11-12 17:05:45 +00:00
| force_release | boolean | body | Force a host release from ADE flow, in case the setup is taking too long. |
##### Example
2024-12-05 17:22:13 +00:00
`POST /api/v1/fleet/orbit/setup_experience/status`
2024-11-12 17:05:45 +00:00
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"force_release":false
}
```
##### Default response
`Status: 200`
```json
{
"setup_experience_results": {
"script": {
"name": "setup_script.sh",
"status": "success",
"execution_id": "b16fdd31-71cc-4258-ab27-744490809ebd"
},
"software": [
{
"name": "Zoom Workplace",
"status": "success",
"software_title_id": 957
},
{
"name": "Bear: Markdown Notes",
"status": "success",
"software_title_id": 287
},
{
"name": "Evernote",
"status": "success",
"software_title_id": 1313
}
],
"configuration_profiles": [
{
"profile_uuid": "ae6a9efd5-9166-11ef-83af-0242ac12000b",
"name": "Fleetd configuration",
"status": "verified"
},
{
"profile_uuid": "ae6aa8108-9166-11ef-83af-0242ac12000b",
"name": "Fleet root certificate authority (CA)",
"status": "verified"
}
],
"org_logo_url": ""
}
}
```
2023-05-17 14:16:26 +00:00
2024-12-05 17:22:13 +00:00
### Set or update device token
`POST /api/fleet/orbit/device_token`
##### Parameters
| Name | Type | In | Description |
| ----------------- | ------ | ---- | ------------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| device_auth_token | string | body | The device auth token to set for this host. |
##### Example
`POST /api/v1/fleet/orbit/device_token`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"device_auth_token": "2267a440-4cfb-48af-804b-d52224a05e1b"
}
```
##### Default response
`Status: 200`
### Get Orbit config
`POST /api/fleet/orbit/config`
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ---- | -------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
##### Example
`POST /api/fleet/orbit/config`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/"
}
```
##### Default response
`Status: 200`
```json
{
"script_execution_timeout": 3600,
"command_line_startup_flags": {
"--verbose": true
},
"extensions": {
"hello_world_linux": {
"channel": "stable",
"platform": "linux"
}
},
"nudge_config": {
"osVersionRequirements": [
{
"requiredInstallationDate": "2024-12-04T20:00:00Z",
"requiredMinimumOSVersion": "15.1.1",
"aboutUpdateURLs": [
{
"_language": "en",
"aboutUpdateURL": "https://fleetdm.com/learn-more-about/os-updates"
}
]
}
],
"userInterface": {
"simpleMode": true,
"showDeferralCount": false,
"updateElements": [
{
"_language": "en",
"actionButtonText": "Update",
"mainHeader": "Your device requires an update"
}
]
},
"userExperience": {
"initialRefreshCycle": 86400,
"approachingRefreshCycle": 86400,
"imminentRefreshCycle": 7200,
"elapsedRefreshCycle": 3600
}
},
"notifications": {
"renew_enrollment_profile": true,
"rotate_disk_encryption_key": true,
"needs_mdm_migration": true,
"needs_programmatic_windows_mdm_enrollment": true,
"windows_mdm_discovery_endpoint": "/some/path/here",
"needs_programmatic_windows_mdm_unenrollment": true,
"pending_script_execution_ids": [
"a129a440-4cfb-48af-804b-d52224a05e1b"
],
"enforce_bitlocker_encryption": true,
"pending_software_installer_ids": [
"2267a440-4cfb-48af-804b-d52224a05e1b"
],
"run_setup_experience": true,
"run_disk_encryption_escrow": true
},
"update_channels": {
"orbit": "stable",
"osqueryd": "stable",
"desktop": "stable"
}
}
```
### Get script execution result by execution ID
`POST /api/fleet/orbit/scripts/request`
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ---- | -------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| execution_id | string | body | The UUID of the script execution. |
##### Example
`POST /api/fleet/orbit/scripts/request`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"execution_id": "006112E7-7383-4F21-999C-8FA74BB3F573"
}
```
##### Default response
`Status: 200`
```json
{
"host_id": 12,
"execution_id": "006112E7-7383-4F21-999C-8FA74BB3F573",
"script_contents": "echo hello",
"output": "hello",
"runtime": 1,
"exit_code": 0,
"timeout": 30,
"script_id": 42,
"policy_id": 10,
"team_id": 1,
"message": ""
}
```
### Upload Orbit script result
`POST /api/fleet/orbit/scripts/result`
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ------ | ----------------------------------------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| host_id | number | body | The ID of the host on which the script ran. |
| execution_id | string | body | The UUID of the script execution. |
| output | string | body | The output of the script. |
| runtime | string | number | The amount of time the script ran for (in seconds). |
| exit_code | string | number | The exit code of the script. |
| timeout | string | number | The maximum amount of time this script was allowed to run (in seconds). |
##### Example
`POST /api/fleet/orbit/scripts/result`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"host_id": 12,
"execution_id": "006112E7-7383-4F21-999C-8FA74BB3F573",
"output": "hello",
"runtime": 1,
"exit_code": 0,
"timeout": 30
}
```
##### Default response
`Status: 200`
### Set Orbit device mapping
2025-04-07 23:47:11 +00:00
`PUT /api/fleet/orbit/device_mapping`
2024-12-05 17:22:13 +00:00
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ---- | ---------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| email | string | body | The email to use for the device mapping. |
##### Example
2025-04-07 23:47:11 +00:00
`PUT /api/fleet/orbit/device_mapping`
2024-12-05 17:22:13 +00:00
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"email": "test@example.com"
}
```
##### Default response
`Status: 200`
### Upload Orbit software install result
`POST /api/fleet/orbit/software_install/result`
##### Parameters
| Name | Type | In | Description |
| ----------------------------- | ------ | ---- | ------------------------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| host_id | number | body | The ID of the host on which the software was installed. |
| install_uuid | string | body | The UUID of the installation attempt. |
| pre_install_condition_output | string | body | The output from the pre-install condition query. |
| install_script_exit_code | number | body | The exit code from the install script. |
| install_script_output | string | body | The output from the install script. |
| post_install_script_exit_code | number | body | The exit code from the post-install script. |
| post_install_script_output | string | body | The output from the post-install script. |
##### Example
`POST /api/fleet/orbit/software_install/result`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"host_id ": 12,
"install_uuid ": "4D91F9C3-919B-4D5B-ABFC-528D648F27D1",
"pre_install_condition_output ": "example",
"install_script_exit_code ": 0,
"install_script_output ": "software installed",
"post_install_script_exit_code ": 1,
"post_install_script_output ": "error: post-install script failed"
}
```
##### Default response
`Status: 204`
### Download software installer
`POST /api/fleet/orbit/software_install/package`
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ----- | -------------------------------------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| installer_id | number | body | The ID of the software installer to download. |
| alt | string | query | Indicates whether to download the package. Must be set to `"media"` . |
##### Example
`POST /api/fleet/orbit/software_install/package`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"installer_id": 15
}
```
##### Default response
`Status: 200`
```http
Status: 200
Content-Type: application/octet-stream
Content-Disposition: attachment
Content-Length: < length >
Body: < blob >
```
### Get orbit software install details
`POST /api/fleet/orbit/software_install/details`
2025-02-04 18:38:54 +00:00
> Note: The `installer_url` in the response will only be populated if AWS S3 and CloudFront URL signing are configured.
2024-12-05 17:22:13 +00:00
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ---- | ---------------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| install_uuid | string | body | The UUID of the software installation attempt. |
##### Example
`POST /api/fleet/orbit/software_install/details`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"install_uuid": "1652210E-619E-43BA-B3CC-17F4247823F3"
}
```
##### Default response
`Status: 200`
```json
{
"install_id": "1652210E-619E-43BA-B3CC-17F4247823F3",
"installer_id": 12,
"pre_install_condition": "SELECT * FROM osquery_info;",
"install_script": "sudo run-installer",
"uninstall_script": "sudo run-uninstaller",
"post_install_script": "echo done",
"self_service": true,
2025-02-04 18:38:54 +00:00
"installer_url": {
"url": "https://d1nsa5964r3p4i.cloudfront.net/software-installers/98330e7e6db3507b444d576dc437a9ac4d82333a88a6bb6ef36a91fe3d85fa92?Expires=1736178766& Signature=HpcpyniNSBkS695mZhkZRjXo6UQ5JtXQ2sk0poLEMDMeF063IjsBj2O56rruzk3lomYFjqoxc3BdnFqEjrEXQSieSALiCufZ2LjTfWffs7f7qnNVZwlkg-upZd5KBfrCHSIyzMYSPhgWFPOpNRVqOc4NFXx8fxRLagK7NBKFAEfCAwo0~KMCSJiof0zWOdY0a8p0NNAbBn0uLqK7vZLwSttVpoK6ytWRaJlnemofWNvLaa~Et3p5wJJRfYGv73AK-pe4FMb8dc9vqGNSZaDAqw2SOdXrLhrpvSMjNmMO3OvTcGS9hVHMtJvBmgqvCMAWmHBK6v5C9BobSh4TCNLIuA__& Key-Pair-Id=K1HFGXOMBB6TFF",
"filename": "my-installer.pkg"
}
2024-12-05 17:22:13 +00:00
}
```
### Upload disk encryption key
`POST /api/fleet/orbit/disk_encryption_key`
##### Parameters
| Name | Type | In | Description |
| -------------- | ------ | ---- | ----------------------------------------- |
| orbit_node_key | string | body | The Orbit node key for authentication. |
| encryption_key | string | body | The encryption key bytes. |
| client_error | string | body | The error reported by the client, if any. |
##### Example
`POST /api/fleet/orbit/disk_encryption_key`
##### Request body
```json
{
"orbit_node_key":"FbvSsWfTRwXEecUlCBTLmBcjGFAdzqd/",
"encryption_key": "Zm9vYmFyem9vYmFyZG9vYmFybG9vYmFy",
"client_error": "example error",
}
```
##### Default response
`Status: 204`
---
2022-12-05 16:35:45 +00:00
## Setup
2022-08-18 18:58:39 +00:00
Sets up a new Fleet instance with the given parameters.
2022-09-22 21:41:57 +00:00
`POST /api/v1/setup`
2022-08-18 18:58:39 +00:00
#### Parameters
2023-01-30 21:35:56 +00:00
| Name | Type | In | Description |
| ---------- | ------ | ---- | --------------------------------------------------------------------------------------------------------------------------- |
| admin | object | body | **Required.** Contains the following admin user details: `admin` , `email` , `name` , `password` , and `password_confirmation` . |
| org_info | object | body | **Required.** Contains the following organizational details: `org_name` . |
| server_url | string | body | **Required.** The URL of the Fleet instance. |
2022-08-18 18:58:39 +00:00
##### Request body
```json
{
"admin": {
"admin": true,
"email": "janedoe@example.com",
"name": "Jane Doe",
"password": "password!234",
"password_confirmation": "password!234"
},
"org_info": {
"org_name": "Fleet Device Management"
},
"server_url": "https://localhost:8080"
}
```
##### Default response
`Status: 200`
If the Fleet instance is provided required parameters to complete setup.
```json
{
"admin": {
"created_at": "2021-01-07T19:40:04Z",
"updated_at": "2021-01-07T19:40:04Z",
"id": 1,
"name": "Jane Doe",
"email": "janedoe@example.com",
"force_password_reset": false,
"gravatar_url": "",
"sso_enabled": false,
"global_role": "admin",
"api_only": false,
"teams": []
},
"org_info": {
"org_name": "Fleet Device Management",
"org_logo_url": "https://fleetdm.com/logo.png"
},
"server_url": "https://localhost:8080",
"osquery_enroll_secret": null,
"token": "ur4RWGBeiNmNzer/dnGzgUQ+jxrJe19xuHg/LhLkbhuZMQu35scyBHUHs68+RJxZynxQnuTz4WTHXayAJJaGgg=="
}
```
2022-08-16 15:54:41 +00:00
2023-10-10 22:00:45 +00:00
## Scripts
2024-08-20 14:51:36 +00:00
### Batch-apply scripts
2023-10-10 22:00:45 +00:00
_Available in Fleet Premium_
`POST /api/v1/fleet/scripts/batch`
#### Parameters
| Name | Type | In | Description |
| --------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
2024-01-10 22:29:51 +00:00
| team_id | number | query | The ID of the team to add the scripts to. Only one team identifier (`team_id` or `team_name` ) can be included in the request, omit this parameter if using `team_name` .
| team_name | string | query | The name of the team to add the scripts to. Only one team identifier (`team_id` or `team_name` ) can be included in the request, omit this parameter if using `team_id` .
2023-10-10 22:00:45 +00:00
| dry_run | bool | query | Validate the provided scripts and return any validation errors, but do not apply the changes. |
| scripts | array | body | An array of objects with the scripts payloads. Each item must contain `name` with the script name and `script_contents` with the script contents encoded in base64 |
2025-01-10 00:04:34 +00:00
If both `team_id` and `team_name` parameters are included, this endpoint will respond with an error.
If no `team_name` or `team_id` is provided, the scripts will be applied for **all hosts** .
Script contents are uploaded verbatim, without CRLF -> LF conversion.
2023-10-10 22:00:45 +00:00
> Note that this endpoint replaces all the active scripts for the specified team (or no team). Any existing script that is not included in the list will be removed, and existing scripts with the same name as a new script will be edited. Providing an empty list of scripts will remove existing scripts.
#### Example
2024-05-23 21:07:07 +00:00
`POST /api/v1/fleet/scripts/batch`
2023-10-10 22:00:45 +00:00
##### Default response
2024-10-17 22:51:52 +00:00
`Status: 200`
```json
{
"scripts": [
{
"team_id": 3,
"id": 6690,
"name": "Ensure shields are up"
},
{
"team_id": 3,
"id": 10412,
"name": "Ensure flux capacitor is charged"
}
]
}
```
2024-05-23 21:07:07 +00:00
2024-09-23 23:09:31 +00:00
### Run live script
2024-07-25 22:25:48 +00:00
Run a live script and get results back (5 minute timeout). Live scripts only runs on the host if it has no other scripts running.
`POST /api/v1/fleet/scripts/run/sync`
#### Parameters
2024-12-10 19:39:24 +00:00
| Name | Type | In | Description |
| ---- | ------- | ---- | -------------------------------------------- |
| host_id | integer | body | **Required** . The ID of the host to run the script on. |
| script_id | integer | body | The ID of the existing saved script to run. Only one of either `script_id` , `script_contents` , or `script_name` can be included. |
| script_contents | string | body | The contents of the script to run. Only one of either `script_id` , `script_contents` , or `script_name` can be included. |
| script_name | integer | body | The name of the existing saved script to run. If specified, requires `team_id` . Only one of either `script_id` , `script_contents` , or `script_name` can be included. |
| team_id | integer | body | The ID of the existing saved script to run. If specified, requires `script_name` . Only one of either `script_id` , `script_contents` , or `script_name` can be included in the request. |
> Note that if any combination of `script_id`, `script_contents`, and `script_name` are included in the request, this endpoint will respond with an error.
2024-07-25 22:25:48 +00:00
#### Example
`POST /api/v1/fleet/scripts/run/sync`
##### Default response
`Status: 200`
```json
{
"host_id": 1227,
"execution_id": "e797d6c6-3aae-11ee-be56-0242ac120002",
"script_contents": "echo 'hello'",
"output": "hello",
"message": "",
"runtime": 1,
"host_timeout": false,
"exit_code": 0
}
```
2024-10-01 21:09:33 +00:00
## Software
2025-03-07 17:36:17 +00:00
### Update software title name
`PATCH /api/v1/fleet/software/titles/:software_title_id/name`
Only available for software titles that have a non-empty bundle ID, as titles without a bundle
ID will be added back as new rows on the next software ingest with the same name. Endpoint authorization limited
to global admins as this changes the software title's name across all teams.
> **Experimental endpoint**. This endpoint is not guaranteed to continue to exist on future minor releases of Fleet.
#### Parameters
| Name | Type | In | Description |
|-------------------|---------|------|----------------------------------------------------|
| software_title_id | integer | path | **Required** . The ID of the software title to modify. |
| name | string | body | **Required** . The new name of the title. |
#### Example
`PATCH /api/v1/fleet/software/titles/1/name`
```json
{
"name": "2 Chrome 2 Furious.app"
}
```
##### Default response
`Status: 205`
2024-10-01 21:09:33 +00:00
### Batch-apply software
_Available in Fleet Premium._
`POST /api/v1/fleet/software/batch`
This endpoint is asynchronous, meaning it will start a background process to download and apply the software and return a `request_uuid` in the JSON response that can be used to query the status of the batch-apply (using the `GET /api/v1/fleet/software/batch/:request_uuid` endpoint defined below).
#### Parameters
| Name | Type | In | Description |
| --------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| team_name | string | query | The name of the team to add the software package to. Ommitting these parameters will add software to 'No Team'. |
| dry_run | bool | query | If `true` , will validate the provided software packages and return any validation errors, but will not apply the changes. |
| software | object | body | The team's software that will be available for install. |
2025-01-10 00:04:34 +00:00
| software.packages | array | body | An array of objects with values below. |
2025-05-22 21:20:56 +00:00
| software.packages.hash_sha256 | string | body | SHA256 hash of the package. If provided, must be 64 lower-case hex characters. One or both of sha256 or url must be provided. |
| software.packages.url | string | body | URL to the software package (PKG, MSI, EXE or DEB). If sha256 is also provided and the installer isn't already uploaded with the same hash for that URL, call will fail if the downloaded installer doesn't match the hash. |
| software.packages.categories | string[] | body | An array of categories, as they are displayed in the UI, to assign to the package. |
2025-01-10 00:04:34 +00:00
| software.packages.install_script | string | body | Command that Fleet runs to install software. |
| software.packages.pre_install_query | string | body | Condition query that determines if the install will proceed. |
| software.packages.post_install_script | string | body | Script that runs after software install. |
| software.packages.uninstall_script | string | body | Command that Fleet runs to uninstall software. |
| software.packages.self_service | boolean | body | Specifies whether or not end users can install self-service. |
| software.packages.labels_include_any | array | body | Target hosts that have any label in the array. Only one of `labels_include_any` or `labels_exclude_any` can be included. If neither are included, all hosts are targeted. |
| software.packages.labels_exclude_any | array | body | Target hosts that don't have any labels in the array. Only one of `labels_include_any` or `labels_exclude_any` can be included. If neither are included, all hosts are targeted. |
2024-10-01 21:09:33 +00:00
2025-05-22 21:20:56 +00:00
`hash_sha256` can be provided alongside or as a replacement for `url` . If provided alongside `url` , adding software only succeeds if the software downloaded matches the specified hash. If provided without a URL, software with that hash must exist (either on that team or globally, depending on what level of access the API client is authorized at) prior to the GitOps run, whether from a previous GitOps run or an upload at the [Add package ](https://fleetdm.com/docs/rest-api/rest-api#add-package ) endpoint, at which point Fleet will ensure the software package exists on the selected team with the specified configuration without needing to retrieve it again.
2024-10-01 21:09:33 +00:00
#### Example
`POST /api/v1/fleet/software/batch`
##### Default response
2024-11-13 23:22:51 +00:00
`Status: 202`
2024-10-01 21:09:33 +00:00
```json
{
"request_uuid": "ec23c7b6-c336-4109-b89d-6afd859659b4",
}
```
### Get status of software batch-apply request
_Available in Fleet Premium._
`GET /api/v1/fleet/software/batch/:request_uuid`
This endpoint allows querying the status of a batch-apply software request (`POST /api/v1/fleet/software/batch`).
Returns `"status"` field that can be one of `"processing"` , `"complete"` or `"failed"` .
If `"status"` is `"completed"` then the `"packages"` field contains the applied packages.
If `"status"` is `"processing"` then the operation is ongoing and the request should be retried.
If `"status"` is `"failed"` then the `"message"` field contains the error message.
#### Parameters
| Name | Type | In | Description |
| ------------ | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| request_uuid | string | query | The request_uuid returned by the `POST /api/v1/fleet/software/batch` endpoint. |
| team_name | string | query | The name of the team to add the software package to. Ommitting these parameters will add software to 'No Team'. |
| dry_run | bool | query | If `true` , will validate the provided software packages and return any validation errors, but will not apply the changes. |
##### Default responses
`Status: 200`
```json
{
"status": "processing",
"message": "",
"packages": null
}
```
`Status: 200`
```json
{
"status": "completed",
"message": "",
"packages": [
{
"team_id": 1,
"title_id": 2751,
2025-05-22 21:20:56 +00:00
"url": "https://ftp.mozilla.org/pub/firefox/releases/129.0.2/win64/en-US/Firefox%20Setup%20129.0.2.msi",
"hash_sha256": "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef"
2024-10-01 21:09:33 +00:00
}
]
}
```
`Status: 200`
```json
{
"status": "failed",
"message": "validation failed: software.url Couldn't edit software. URL (\"https://foobar.does.not.exist.com\") returned \"Not Found\". Please make sure that URLs are reachable from your Fleet server.",
"packages": null
}
```
### Batch-apply App Store apps
_Available in Fleet Premium._
`POST /api/latest/fleet/software/app_store_apps/batch`
#### Parameters
| Name | Type | In | Description |
| --------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| team_name | string | query | The name of the team to add the software package to. Ommitting this parameter will add software to "No team". |
| dry_run | bool | query | If `true` , will validate the provided VPP apps and return any validation errors, but will not apply the changes. |
| app_store_apps | list | body | An array of objects. Each object contains `app_store_id` and `self_service` . |
2025-02-04 18:38:54 +00:00
| app_store_apps | list | body | An array of objects with . Each object contains `app_store_id` and `self_service` . |
2025-05-22 21:20:56 +00:00
| app_store_apps.categories | string[] | body | An array of categories, as they are displayed in the UI, to assign to the app. |
2024-10-01 21:09:33 +00:00
| app_store_apps.app_store_id | string | body | ID of the App Store app. |
| app_store_apps.self_service | boolean | body | Whether the VPP app is "Self-service" or not. |
2025-02-04 18:38:54 +00:00
| app_store_apps.labels_include_any | array | body | App will only be available for install on hosts that **have any** of these labels. Only one of either `labels_include_any` or `labels_exclude_any` can be included in the request. |
| app_store_apps.labels_exclude_any | array | body | App will only be available for install on hosts that **don't have any** of these labels. Only one of either `labels_include_any` or `labels_exclude_any` can be included in the request. |
2024-10-01 21:09:33 +00:00
#### Example
`POST /api/latest/fleet/software/app_store_apps/batch`
```json
{
"team_name": "Foobar",
2025-01-10 17:51:35 +00:00
"app_store_apps": [
2024-10-01 21:09:33 +00:00
{
"app_store_id": "597799333",
2025-05-22 21:20:56 +00:00
"categories": ["Browsers"],
2025-02-04 18:38:54 +00:00
"self_service": false,
"labels_include_any": [
"Engineering",
"Customer Support"
]
2024-10-01 21:09:33 +00:00
},
{
"app_store_id": "497799835",
2025-01-20 17:23:29 +00:00
"self_service": true
2024-10-01 21:09:33 +00:00
}
2025-01-10 17:51:35 +00:00
]
2024-10-01 21:09:33 +00:00
}
```
##### Default response
2025-01-20 17:23:29 +00:00
`Status: 200`
```json
{
"app_store_apps": [
{
"team_id": 1,
"title_id": 123,
"app_store_id": "597799333",
"platform": "darwin"
},
{
"team_id": 1,
"title_id": 124,
"app_store_id": "597799333",
"platform": "ios"
},
{
"team_id": 1,
"title_id": 125,
"app_store_id": "597799333",
"platform": "ipados"
}
]
}
```
2024-10-01 21:09:33 +00:00
### Get token to download package
_Available in Fleet Premium._
`POST /api/v1/fleet/software/titles/:software_title_id/package/token?alt=media`
The returned token is a one-time use token that expires after 10 minutes.
#### Parameters
| Name | Type | In | Description |
|-------------------|---------|-------|------------------------------------------------------------------|
| software_title_id | integer | path | **Required** . The ID of the software title for software package. |
| team_id | integer | query | **Required** . The team ID containing the software package. |
| alt | integer | query | **Required** . Must be specified and set to "media". |
#### Example
`POST /api/v1/fleet/software/titles/123/package/token?alt=media&team_id=2`
##### Default response
`Status: 200`
```json
{
"token": "e905e33e-07fe-4f82-889c-4848ed7eecb7"
}
```
### Download package using a token
_Available in Fleet Premium._
`GET /api/v1/fleet/software/titles/:software_title_id/package/token/:token?alt=media`
#### Parameters
| Name | Type | In | Description |
|-------------------|---------|------|--------------------------------------------------------------------------|
| software_title_id | integer | path | **Required** . The ID of the software title to download software package. |
| token | string | path | **Required** . The token to download the software package. |
#### Example
`GET /api/v1/fleet/software/titles/123/package/token/e905e33e-07fe-4f82-889c-4848ed7eecb7`
##### Default response
`Status: 200`
```http
Status: 200
Content-Type: application/octet-stream
Content-Disposition: attachment
Content-Length: < length >
Body: < blob >
```
2025-04-24 21:10:41 +00:00
---
2025-03-10 17:17:57 +00:00
## Users
### Update user-specific UI settings
`PATCH /api/v1/fleet/users/:id`
#### Parameters
| Name | Type | In | Description |
| --------- | ------ | ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| settings | object | body | The updated user settings. |
#### Example
`PATCH /api/v1/fleet/users/1`
```json
{
"hidden_host_columns": ["hostname"]
}
```
##### Default response
* Note that user settings are *not* included in this response. See below `GET` s for how to get user settings.
`Status: 200`
```json
{
"user": {
"created_at": "2025-01-08T01:04:23Z",
"updated_at": "2025-01-09T00:08:19Z",
"id": 1,
"name": "Sum Bahdee",
"email": "sum@org.com",
"force_password_reset": false,
"gravatar_url": "",
"sso_enabled": false,
"mfa_enabled": false,
"global_role": "admin",
"api_only": false,
"teams": []
}
}
```
### Include settings when getting a user
`GET /api/v1/fleet/users/:id?include_ui_settings=true`
Use of `include_ui_settings=true` is considered the contributor API functionality – without that
param, this endpoint is considered a documented REST API endpoint
#### Parameters
| Name | Type | In | Description |
| ------------------- | ------ | ----- | --------------------------------------------------------------------------------------------------|
| include_ui_settings | bool | query | If `true` , will include the user's settings in the response. For now, this is a single ui setting.
#### Example
`GET /api/v1/fleet/users/2/?include_ui_settings=true`
##### Default response
`Status: 200`
```json
{
"user": {...},
"available_teams": {...}
"settings": {"hidden_host_columns": ["hostname"]},
}
```
### Include settings when getting current user
`GET /api/v1/fleet/me/?include_ui_settings=true`
Use of `include_ui_settings=true` is considered the contributor API functionality – without that
param, this endpoint is considered a documented REST API endpoint
#### Parameters
| Name | Type | In | Description |
| ------------------- | ------ | ----- | --------------------------------------------------------------------------------------------------|
| include_ui_settings | bool | query | If `true` , will include the user's settings in the response. For now, this is a single ui setting.
#### Example
`GET /api/v1/fleet/me?include_ui_settings=true`
##### Default response
`Status: 200`
```json
{
"user": {...},
"available_teams": {...}
"settings": {"hidden_host_columns": ["hostname"]},
}
```
2025-05-22 21:20:56 +00:00
---
## Conditional access
### Initiate Microsoft Entra conditional access
Kick off authentication with Microsoft Entra to configure conditional access.
`POST /api/v1/conditional-access/microsoft`
#### Parameters
| Name | Type | In | Description |
| ---------- | ------ | ---- | --------------------------------------------------------------------------------------------------------------------------- |
| microsoft_tenant_id | string | body | **Required.** The Microsoft Entra tenant ID. |
##### Request body
```json
{
"fleet_license_key": "< LICENSE KEY > ",
"microsoft_tenant_id": "< TENANT ID > "
}
```
##### Default response
`Status: 200`
```json
{
"microsoft_authentication_url": "https://login.microsoftonline.com/< TENANT ID > /adminconsent?client_id=< CLIENT ID > "
}
```
### Confirm Microsoft Entra conditional access configuration
`POST /api/v1/conditional-access/confirm`
Confirm whether admin consent has been granted in Microsoft Entra.
#### Parameters
None.
##### Default response
`Status: 200`
```json
{
2025-06-11 17:22:46 +00:00
"configuration_completed": false
2025-05-22 21:20:56 +00:00
}
```
### Delete Microsoft Entra conditional access
`DELETE /api/v1/conditional-access/microsoft`
#### Parameters
None.
##### Default response
`Status: 200`