Skip to content

Latest commit

 

History

History
200 lines (157 loc) · 6.89 KB

add-new-api.md

File metadata and controls

200 lines (157 loc) · 6.89 KB

How to add a new API

It might happen that a new API in Elasticsearch is not yet defined in this repository, or we do have an endpoint definition in /specification/_json_spec but we don't have a type definition for it. In this document you will see how to add a new endpopint and how to add a new endpoint definition.

How to add a new endpoint

Add a new endpoint is straightforward, you only need to copy-paste the json rest-api-spec defintion from the Elasticsearch repository inside /specification/_json_spec and you are good to go.

You can find the rest-api-spec definitions here or here.

How to add the definition of an endpoint

Once you have added a new endpoint definition, the next step is to add its type definition. First of all, you should find the most approariate place inside /specification where to put the new definition. The content of /specification tryied to mimic the Elasticsearch online documentation, so you can use it as inspiration. For example, the index document defintion can be found in /specification/_global/index.

Once you have found the best place for the new definition, you should create a new file for it. The filename should be the same of the type definition you are writing, for example:

// IndexRequest.ts
interface Request {}
// IndexResponse.ts
class Response {}

Try to use less files as possible, for example there is no need to create a custom file for an enum, you can define it in the same file where it's used, unless is a commonly used type.

Add the endpoint request definition

Request definitions are slighly different from other definitions. It is required that the request definition is named Request. A request definition is an interface and should contains three top level keys:

  • urls: the URL paths templates and allowed HTTP methods
  • path_parts: the path parameters (eg: indices, id...)
  • query_parameters: the query parameters (eg: timeout, pipeline...)
  • body: the body parameters (eg: query or user defined entities)

Furthermore, every request definition must contain three JS Doc tags:

  • @rest_spec_name: the API name (eg: search, indices.create...).
  • @availability Which flavor of Elasticsearch is this API available for? (eg stack or serverless)
    • stability: the API stability, one of experimental, beta, stable
    • visibility: the API stability, one of public or private.
    • since: the version of Elasticsearch when the API has been introduced (eg: 7.7.0). This field is only available for stack.
    • feature_flag: the feature flag value, only valid if the visibility is set to feature_flag. This field is only available for stack.

Following you can find a template valid for any request definition.

 /*
 * @rest_spec_name endpoint.name
 * @availability stack since=1.2.3 stability=stable|beta|experimental
 */
interface Request extends RequestBase {
  urls: [
    {
      path: "/path/with/{parts}"
      methods: ["POST"]
    }
  ]
  path_parts: {

  }
  query_parameters: {

  }
  body: {

  }
}

In some cases, the request could take one or more generics, in such case the definition will be:

 /*
 * @rest_spec_name endpoint.name
 * @availability stack since=1.2.3 stability=stable|beta|experimental
 */
interface Request<Generic> extends RequestBase {
  urls: [
    {
      path: "/path/with/{parts}"
      methods: ["POST"]
    }
  ]
  path_parts: {

  }
  query_parameters: {

  }
  body: {

  }
}

And the generic will be used somewhere inside the definition. There are cases where the generic might be the entire body, see IndexRequest.

Add the endpoint response definition

Responses definitions should always be defined after a request definition, otherwise the compiler will not pick them up. It is required that the response definition is named Response.

Following you can find a template valid for any response definition.

class Response {
  body: {

  }
}

As you can see, for responses there are no custom top level keys, as the response definition represents the body of a succesful response.

Generics

In some cases, the response could take one or more generics, in such case the definition will be:

class Response<Generic> {
  body: {
    key: Generic
  }
}

And the generic will be used somewhere inside the definition.

Exceptions

Normally, every API returns the exact same structure in case of error, which is defined in ErrorResponseBase. In some edge cases, the response structure may change. To document this situations, you should use the exceptions key.

class Response {
  body: SuccessResponseBodyType
  exceptions: [
    {
      /**
       * A brief description of the exception.
       */
      statusCodes: [404]
      body: NotFoundException
    }
  ]
}

Add endpoint request and response examples

Add an examples folder and request and xxx_response subfolders (where xxx is the relevant response code).

These examples are for use in the API documentation and must adhere to the OpenAPI 3.0 Example object specification. They must have a value field that contains the request or response body. If there are multiple examples for the endpoint, they must each have a brief summary field, which is used as the label for the example. You can also optionaly provide an explanation in a description field.

For example:

summary: Sequence query
# method_request: GET /my-data-stream/_eql/search
# type: request
description: >
  Run `GET /my-data-stream/_eql/search` to search for a sequence of events.
  The sequence starts with an event with an `event.category` of `file`, a `file.name` of `cmd.exe`, and a `process.pid` other than `2013`.
  It is followed by an event with an `event.category` of `process` and a `process.executable` that contains the substring `regsvr32`.
  These events must also share the same `process.pid` value.
value: |-
  {
    "query": """
      sequence by process.pid
        [ file where file.name == "cmd.exe" and process.pid != 2013 ]
        [ process where stringContains(process.executable, "regsvr32") ]
    """
  }

NOTE: A good example covers a very common use case or demonstrates a more complex but critical use case. It involves realistic data sets ( rather than generic "hello world" values). If it requires detailed setup or explanations, however, it is more appropriate for coverage in longer-form narrative documentation.