mirror of
https://github.com/datahub-project/datahub.git
synced 2025-10-24 15:34:57 +00:00
503 lines
24 KiB
Markdown
503 lines
24 KiB
Markdown
# Getting Started
|
|
|
|
## Introduction to OpenAPI - What And Why
|
|
|
|
The OpenAPI standard is a widely used documentation and design approach for APIs. Historically, we have published our RESTful APIs
|
|
under the Rest.li standard that is used internally by LinkedIn that we inherited from the beginning of the project.
|
|
|
|
Rest.li is a very opinionated framework that has not seen wide adoption in the Open Source community, so users are often unfamiliar
|
|
with the best ways to interact with these endpoints. To make it easier to integrate with DataHub, we are publishing an OpenAPI based set of endpoints.
|
|
|
|
## Using DataHub's OpenAPI - Where and How
|
|
|
|
Currently, the OpenAPI endpoints are isolated to a servlet on GMS and are automatically deployed with a GMS server.
|
|
The servlet includes auto-generation of an OpenAPI UI, also known as Swagger, which is available at `<GMS-server-host>:<GMS-port>/openapi/swagger-ui/index.html`.
|
|
This is also exposed through DataHub frontend as a proxy with the same endpoint, but GMS host and port replaced with DataHub frontend's url
|
|
and is available in the top right dropdown under the user profile picture as a link. Note that it is possible to get
|
|
the raw json or yaml formats of the OpenAPI spec by navigating to `<baseUrl>/openapi/v3/api-docs` or `<baseUrl>/openapi/v3/api-docs.yaml`.
|
|
The raw forms can be fed into codegen systems to generate client side code in the language of your choice that support OpenAPI format. We have noticed varying
|
|
degrees of maturity with different languages in these codegen systems so some may require customizations to be fully compatible.
|
|
|
|
The OpenAPI UI includes explorable schemas for request and response objects that are fully documented. The models used
|
|
in the OpenAPI UI are all autogenerated at build time from the PDL models to JSON Schema compatible Java Models.
|
|
|
|
Programmatic usage of the models can be done through the Java Rest Emitter which includes the generated models. A minimal
|
|
Java project for emitting to the OpenAPI endpoints would need the following dependencies (gradle format):
|
|
|
|
```groovy
|
|
dependencies {
|
|
implementation 'io.acryl:datahub-client:<DATAHUB_CLIENT_VERSION>'
|
|
implementation 'org.apache.httpcomponents:httpclient:<APACHE_HTTP_CLIENT_VERSION>'
|
|
implementation 'org.apache.httpcomponents:httpasyncclient:<APACHE_ASYNC_CLIENT_VERSION>'
|
|
}
|
|
```
|
|
|
|
and would construct a list of `UpsertAspectRequest`s to emit:
|
|
|
|
```java
|
|
import io.datahubproject.openapi.generated.DatasetProperties;
|
|
import datahub.client.rest.RestEmitter;
|
|
import datahub.event.UpsertAspectRequest;
|
|
import java.io.IOException;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.concurrent.ExecutionException;
|
|
|
|
|
|
public class Main {
|
|
public static void main(String[] args) throws IOException, ExecutionException, InterruptedException {
|
|
RestEmitter emitter = RestEmitter.createWithDefaults();
|
|
|
|
List<UpsertAspectRequest> requests = new ArrayList<>();
|
|
UpsertAspectRequest upsertAspectRequest = UpsertAspectRequest.builder()
|
|
.entityType("dataset")
|
|
.entityUrn("urn:li:dataset:(urn:li:dataPlatform:bigquery,my-project.my-other-dataset.user-table,PROD)")
|
|
.aspect(new DatasetProperties().description("This is the canonical User profile dataset"))
|
|
.build();
|
|
UpsertAspectRequest upsertAspectRequest2 = UpsertAspectRequest.builder()
|
|
.entityType("dataset")
|
|
.entityUrn("urn:li:dataset:(urn:li:dataPlatform:bigquery,my-project.another-dataset.user-table,PROD)")
|
|
.aspect(new DatasetProperties().description("This is the canonical User profile dataset 2"))
|
|
.build();
|
|
requests.add(upsertAspectRequest);
|
|
requests.add(upsertAspectRequest2);
|
|
System.out.println(emitter.emit(requests, null).get());
|
|
System.exit(0);
|
|
}
|
|
}
|
|
```
|
|
|
|
### Example requests
|
|
|
|
#### Curl
|
|
|
|
##### POST
|
|
|
|
```shell
|
|
curl --location --request POST 'localhost:8080/openapi/entities/v1/' \
|
|
--header 'Content-Type: application/json' \
|
|
--header 'Accept: application/json' \
|
|
--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJhY3RvclR5cGUiOiJVU0VSIiwiYWN0b3JJZCI6ImRhdGFodWIiLCJ0eXBlIjoiUEVSU09OQUwiLCJ2ZXJzaW9uIjoiMSIsImV4cCI6MTY1MDY2MDY1NSwianRpIjoiM2E4ZDY3ZTItOTM5Yi00NTY3LWE0MjYtZDdlMDA1ZGU3NjJjIiwic3ViIjoiZGF0YWh1YiIsImlzcyI6ImRhdGFodWItbWV0YWRhdGEtc2VydmljZSJ9.pp_vW2u1tiiTT7U0nDF2EQdcayOMB8jatiOA8Je4JJA' \
|
|
--data-raw '[
|
|
{
|
|
"aspect": {
|
|
"__type": "SchemaMetadata",
|
|
"schemaName": "SampleHdfsSchema",
|
|
"platform": "urn:li:dataPlatform:platform",
|
|
"platformSchema": {
|
|
"__type": "MySqlDDL",
|
|
"tableSchema": "schema"
|
|
},
|
|
"version": 0,
|
|
"created": {
|
|
"time": 1621882982738,
|
|
"actor": "urn:li:corpuser:etl",
|
|
"impersonator": "urn:li:corpuser:jdoe"
|
|
},
|
|
"lastModified": {
|
|
"time": 1621882982738,
|
|
"actor": "urn:li:corpuser:etl",
|
|
"impersonator": "urn:li:corpuser:jdoe"
|
|
},
|
|
"hash": "",
|
|
"fields": [
|
|
{
|
|
"fieldPath": "county_fips_codefg",
|
|
"jsonPath": "null",
|
|
"nullable": true,
|
|
"description": "null",
|
|
"type": {
|
|
"type": {
|
|
"__type": "StringType"
|
|
}
|
|
},
|
|
"nativeDataType": "String()",
|
|
"recursive": false
|
|
},
|
|
{
|
|
"fieldPath": "county_name",
|
|
"jsonPath": "null",
|
|
"nullable": true,
|
|
"description": "null",
|
|
"type": {
|
|
"type": {
|
|
"__type": "StringType"
|
|
}
|
|
},
|
|
"nativeDataType": "String()",
|
|
"recursive": false
|
|
}
|
|
]
|
|
},
|
|
"entityType": "dataset",
|
|
"entityUrn": "urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)"
|
|
}
|
|
]'
|
|
```
|
|
|
|
##### GET
|
|
|
|
```shell
|
|
curl --location --request GET 'localhost:8080/openapi/entities/v1/latest?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&aspectNames=schemaMetadata' \
|
|
--header 'Accept: application/json' \
|
|
--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJhY3RvclR5cGUiOiJVU0VSIiwiYWN0b3JJZCI6ImRhdGFodWIiLCJ0eXBlIjoiUEVSU09OQUwiLCJ2ZXJzaW9uIjoiMSIsImV4cCI6MTY1MDY2MDY1NSwianRpIjoiM2E4ZDY3ZTItOTM5Yi00NTY3LWE0MjYtZDdlMDA1ZGU3NjJjIiwic3ViIjoiZGF0YWh1YiIsImlzcyI6ImRhdGFodWItbWV0YWRhdGEtc2VydmljZSJ9.pp_vW2u1tiiTT7U0nDF2EQdcayOMB8jatiOA8Je4JJA'
|
|
```
|
|
|
|
##### DELETE
|
|
|
|
```shell
|
|
curl --location --request DELETE 'localhost:8080/openapi/entities/v1/?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&soft=true' \
|
|
--header 'Accept: application/json' \
|
|
--header 'Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJhY3RvclR5cGUiOiJVU0VSIiwiYWN0b3JJZCI6ImRhdGFodWIiLCJ0eXBlIjoiUEVSU09OQUwiLCJ2ZXJzaW9uIjoiMSIsImV4cCI6MTY1MDY2MDY1NSwianRpIjoiM2E4ZDY3ZTItOTM5Yi00NTY3LWE0MjYtZDdlMDA1ZGU3NjJjIiwic3ViIjoiZGF0YWh1YiIsImlzcyI6ImRhdGFodWItbWV0YWRhdGEtc2VydmljZSJ9.pp_vW2u1tiiTT7U0nDF2EQdcayOMB8jatiOA8Je4JJA'
|
|
```
|
|
|
|
#### Postman Collection
|
|
|
|
Collection includes a POST, GET, and DELETE for a single entity with a SchemaMetadata aspect
|
|
|
|
```json
|
|
{
|
|
"info": {
|
|
"_postman_id": "87b7401c-a5dc-47e4-90b4-90fe876d6c28",
|
|
"name": "DataHub OpenAPI",
|
|
"description": "A description",
|
|
"schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json"
|
|
},
|
|
"item": [
|
|
{
|
|
"name": "entities/v1",
|
|
"item": [
|
|
{
|
|
"name": "post Entities 1",
|
|
"request": {
|
|
"method": "POST",
|
|
"header": [
|
|
{
|
|
"key": "Content-Type",
|
|
"value": "application/json"
|
|
},
|
|
{
|
|
"key": "Accept",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"body": {
|
|
"mode": "raw",
|
|
"raw": "[\n {\n \"aspect\": {\n \"__type\": \"SchemaMetadata\",\n \"schemaName\": \"SampleHdfsSchema\",\n \"platform\": \"urn:li:dataPlatform:platform\",\n \"platformSchema\": {\n \"__type\": \"MySqlDDL\",\n \"tableSchema\": \"schema\"\n },\n \"version\": 0,\n \"created\": {\n \"time\": 1621882982738,\n \"actor\": \"urn:li:corpuser:etl\",\n \"impersonator\": \"urn:li:corpuser:jdoe\"\n },\n \"lastModified\": {\n \"time\": 1621882982738,\n \"actor\": \"urn:li:corpuser:etl\",\n \"impersonator\": \"urn:li:corpuser:jdoe\"\n },\n \"hash\": \"\",\n \"fields\": [\n {\n \"fieldPath\": \"county_fips_codefg\",\n \"jsonPath\": \"null\",\n \"nullable\": true,\n \"description\": \"null\",\n \"type\": {\n \"type\": {\n \"__type\": \"StringType\"\n }\n },\n \"nativeDataType\": \"String()\",\n \"recursive\": false\n },\n {\n \"fieldPath\": \"county_name\",\n \"jsonPath\": \"null\",\n \"nullable\": true,\n \"description\": \"null\",\n \"type\": {\n \"type\": {\n \"__type\": \"StringType\"\n }\n },\n \"nativeDataType\": \"String()\",\n \"recursive\": false\n }\n ]\n },\n \"aspectName\": \"schemaMetadata\",\n \"entityType\": \"dataset\",\n \"entityUrn\": \"urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)\"\n }\n]",
|
|
"options": {
|
|
"raw": {
|
|
"language": "json"
|
|
}
|
|
}
|
|
},
|
|
"url": {
|
|
"raw": "{{baseUrl}}/openapi/entities/v1/",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"openapi",
|
|
"entities",
|
|
"v1",
|
|
""
|
|
]
|
|
}
|
|
},
|
|
"response": [
|
|
{
|
|
"name": "OK",
|
|
"originalRequest": {
|
|
"method": "POST",
|
|
"header": [],
|
|
"body": {
|
|
"mode": "raw",
|
|
"raw": "[\n {\n \"aspect\": {\n \"value\": \"<Error: Too many levels of nesting to fake this schema>\"\n },\n \"aspectName\": \"aliquip ipsum tempor\",\n \"entityType\": \"ut est\",\n \"entityUrn\": \"enim in nulla\",\n \"entityKeyAspect\": {\n \"value\": \"<Error: Too many levels of nesting to fake this schema>\"\n }\n },\n {\n \"aspect\": {\n \"value\": \"<Error: Too many levels of nesting to fake this schema>\"\n },\n \"aspectName\": \"ipsum id\",\n \"entityType\": \"deser\",\n \"entityUrn\": \"aliqua sit\",\n \"entityKeyAspect\": {\n \"value\": \"<Error: Too many levels of nesting to fake this schema>\"\n }\n }\n]",
|
|
"options": {
|
|
"raw": {
|
|
"language": "json"
|
|
}
|
|
}
|
|
},
|
|
"url": {
|
|
"raw": "{{baseUrl}}/entities/v1/",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"entities",
|
|
"v1",
|
|
""
|
|
]
|
|
}
|
|
},
|
|
"status": "OK",
|
|
"code": 200,
|
|
"_postman_previewlanguage": "json",
|
|
"header": [
|
|
{
|
|
"key": "Content-Type",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"cookie": [],
|
|
"body": "[\n \"c\",\n \"labore dolor exercitation in\"\n]"
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"name": "delete Entities",
|
|
"request": {
|
|
"method": "DELETE",
|
|
"header": [
|
|
{
|
|
"key": "Accept",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"url": {
|
|
"raw": "{{baseUrl}}/openapi/entities/v1/?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&soft=true",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"openapi",
|
|
"entities",
|
|
"v1",
|
|
""
|
|
],
|
|
"query": [
|
|
{
|
|
"key": "urns",
|
|
"value": "urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)",
|
|
"description": "(Required) A list of raw urn strings, only supports a single entity type per request."
|
|
},
|
|
{
|
|
"key": "urns",
|
|
"value": "labore dolor exercitation in",
|
|
"description": "(Required) A list of raw urn strings, only supports a single entity type per request.",
|
|
"disabled": true
|
|
},
|
|
{
|
|
"key": "soft",
|
|
"value": "true",
|
|
"description": "Determines whether the delete will be soft or hard, defaults to true for soft delete"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"response": [
|
|
{
|
|
"name": "OK",
|
|
"originalRequest": {
|
|
"method": "DELETE",
|
|
"header": [],
|
|
"url": {
|
|
"raw": "{{baseUrl}}/entities/v1/?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&soft=true",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"entities",
|
|
"v1",
|
|
""
|
|
],
|
|
"query": [
|
|
{
|
|
"key": "urns",
|
|
"value": "urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)"
|
|
},
|
|
{
|
|
"key": "urns",
|
|
"value": "officia occaecat elit dolor",
|
|
"disabled": true
|
|
},
|
|
{
|
|
"key": "soft",
|
|
"value": "true"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"status": "OK",
|
|
"code": 200,
|
|
"_postman_previewlanguage": "json",
|
|
"header": [
|
|
{
|
|
"key": "Content-Type",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"cookie": [],
|
|
"body": "[\n {\n \"rowsRolledBack\": [\n {\n \"urn\": \"urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)\"\n }\n ],\n \"rowsDeletedFromEntityDeletion\": 1\n }\n]"
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"name": "get Entities",
|
|
"protocolProfileBehavior": {
|
|
"disableUrlEncoding": false
|
|
},
|
|
"request": {
|
|
"method": "GET",
|
|
"header": [
|
|
{
|
|
"key": "Accept",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"url": {
|
|
"raw": "{{baseUrl}}/openapi/entities/v1/latest?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&aspectNames=schemaMetadata",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"openapi",
|
|
"entities",
|
|
"v1",
|
|
"latest"
|
|
],
|
|
"query": [
|
|
{
|
|
"key": "urns",
|
|
"value": "urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)",
|
|
"description": "(Required) A list of raw urn strings, only supports a single entity type per request."
|
|
},
|
|
{
|
|
"key": "urns",
|
|
"value": "labore dolor exercitation in",
|
|
"description": "(Required) A list of raw urn strings, only supports a single entity type per request.",
|
|
"disabled": true
|
|
},
|
|
{
|
|
"key": "aspectNames",
|
|
"value": "schemaMetadata",
|
|
"description": "The list of aspect names to retrieve"
|
|
},
|
|
{
|
|
"key": "aspectNames",
|
|
"value": "labore dolor exercitation in",
|
|
"description": "The list of aspect names to retrieve",
|
|
"disabled": true
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"response": [
|
|
{
|
|
"name": "OK",
|
|
"originalRequest": {
|
|
"method": "GET",
|
|
"header": [],
|
|
"url": {
|
|
"raw": "{{baseUrl}}/entities/v1/latest?urns=urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)&aspectNames=schemaMetadata",
|
|
"host": [
|
|
"{{baseUrl}}"
|
|
],
|
|
"path": [
|
|
"entities",
|
|
"v1",
|
|
"latest"
|
|
],
|
|
"query": [
|
|
{
|
|
"key": "urns",
|
|
"value": "non exercitation occaecat",
|
|
"disabled": true
|
|
},
|
|
{
|
|
"key": "urns",
|
|
"value": "urn:li:dataset:(urn:li:dataPlatform:platform,testSchemaIngest,PROD)"
|
|
},
|
|
{
|
|
"key": "aspectNames",
|
|
"value": "non exercitation occaecat",
|
|
"disabled": true
|
|
},
|
|
{
|
|
"key": "aspectNames",
|
|
"value": "schemaMetadata"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"status": "OK",
|
|
"code": 200,
|
|
"_postman_previewlanguage": "json",
|
|
"header": [
|
|
{
|
|
"key": "Content-Type",
|
|
"value": "application/json"
|
|
}
|
|
],
|
|
"cookie": [],
|
|
"body": "{\n \"responses\": {\n \"urn:li:dataset:(urn:li:dataPlatform:hive,SampleHiveDataset,PROD)\": {\n \"entityName\": \"dataset\",\n \"urn\": \"urn:li:dataset:(urn:li:dataPlatform:hive,SampleHiveDataset,PROD)\",\n \"aspects\": {\n \"datasetKey\": {\n \"name\": \"datasetKey\",\n \"type\": \"VERSIONED\",\n \"version\": 0,\n \"value\": {\n \"__type\": \"DatasetKey\",\n \"platform\": \"urn:li:dataPlatform:hive\",\n \"name\": \"SampleHiveDataset\",\n \"origin\": \"PROD\"\n },\n \"created\": {\n \"time\": 1650657843351,\n \"actor\": \"urn:li:corpuser:__datahub_system\"\n }\n },\n \"schemaMetadata\": {\n \"name\": \"schemaMetadata\",\n \"type\": \"VERSIONED\",\n \"version\": 0,\n \"value\": {\n \"__type\": \"SchemaMetadata\",\n \"schemaName\": \"SampleHiveSchema\",\n \"platform\": \"urn:li:dataPlatform:hive\",\n \"version\": 0,\n \"created\": {\n \"time\": 1581407189000,\n \"actor\": \"urn:li:corpuser:jdoe\"\n },\n \"lastModified\": {\n \"time\": 1581407189000,\n \"actor\": \"urn:li:corpuser:jdoe\"\n },\n \"hash\": \"\",\n \"platformSchema\": {\n \"__type\": \"KafkaSchema\",\n \"documentSchema\": \"{\\\"type\\\":\\\"record\\\",\\\"name\\\":\\\"SampleHiveSchema\\\",\\\"namespace\\\":\\\"com.linkedin.dataset\\\",\\\"doc\\\":\\\"Sample Hive dataset\\\",\\\"fields\\\":[{\\\"name\\\":\\\"field_foo\\\",\\\"type\\\":[\\\"string\\\"]},{\\\"name\\\":\\\"field_bar\\\",\\\"type\\\":[\\\"boolean\\\"]}]}\"\n },\n \"fields\": [\n {\n \"fieldPath\": \"field_foo\",\n \"nullable\": false,\n \"description\": \"Foo field description\",\n \"type\": {\n \"type\": {\n \"__type\": \"BooleanType\"\n }\n },\n \"nativeDataType\": \"varchar(100)\",\n \"recursive\": false,\n \"isPartOfKey\": true\n },\n {\n \"fieldPath\": \"field_bar\",\n \"nullable\": false,\n \"description\": \"Bar field description\",\n \"type\": {\n \"type\": {\n \"__type\": \"BooleanType\"\n }\n },\n \"nativeDataType\": \"boolean\",\n \"recursive\": false,\n \"isPartOfKey\": false\n }\n ]\n },\n \"created\": {\n \"time\": 1650610810000,\n \"actor\": \"urn:li:corpuser:UNKNOWN\"\n }\n }\n }\n }\n }\n}"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"auth": {
|
|
"type": "bearer",
|
|
"bearer": [
|
|
{
|
|
"key": "token",
|
|
"value": "{{token}}",
|
|
"type": "string"
|
|
}
|
|
]
|
|
},
|
|
"event": [
|
|
{
|
|
"listen": "prerequest",
|
|
"script": {
|
|
"type": "text/javascript",
|
|
"exec": [
|
|
""
|
|
]
|
|
}
|
|
},
|
|
{
|
|
"listen": "test",
|
|
"script": {
|
|
"type": "text/javascript",
|
|
"exec": [
|
|
""
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"event": [
|
|
{
|
|
"listen": "prerequest",
|
|
"script": {
|
|
"type": "text/javascript",
|
|
"exec": [
|
|
""
|
|
]
|
|
}
|
|
},
|
|
{
|
|
"listen": "test",
|
|
"script": {
|
|
"type": "text/javascript",
|
|
"exec": [
|
|
""
|
|
]
|
|
}
|
|
}
|
|
],
|
|
"variable": [
|
|
{
|
|
"key": "baseUrl",
|
|
"value": "localhost:8080",
|
|
"type": "string"
|
|
},
|
|
{
|
|
"key": "token",
|
|
"value": "eyJhbGciOiJIUzI1NiJ9.eyJhY3RvclR5cGUiOiJVU0VSIiwiYWN0b3JJZCI6ImRhdGFodWIiLCJ0eXBlIjoiUEVSU09OQUwiLCJ2ZXJzaW9uIjoiMSIsImV4cCI6MTY1MDY2MDY1NSwianRpIjoiM2E4ZDY3ZTItOTM5Yi00NTY3LWE0MjYtZDdlMDA1ZGU3NjJjIiwic3ViIjoiZGF0YWh1YiIsImlzcyI6ImRhdGFodWItbWV0YWRhdGEtc2VydmljZSJ9.pp_vW2u1tiiTT7U0nDF2EQdcayOMB8jatiOA8Je4JJA",
|
|
"type": "default"
|
|
}
|
|
]
|
|
}
|
|
``` |