Skip to content

Latest commit

 

History

History
304 lines (242 loc) · 11.4 KB

discovery-api.md

File metadata and controls

304 lines (242 loc) · 11.4 KB

Discovery API

This document explains the discovery API, control flow and the contracts behind it. It starts with a high-level overview and then explains every API in detail.

Overview

Discoveries are implemented with the help of DiscoveryKit and the discovery API through the implementation of a discovery provider. Discovery providers are HTTP servers implementing the discovery API to describe which discoveries are supported and how to execute these. The following diagram illustrates who is issuing calls and in what phases. Besides discoveries, the discovery provider can describe new target types and attributes, which the platform user interface will use to display and select targets.

UML sequence diagram showing in what order the APIs are called

As can be seen above, the discovery provider is called by the Steadybit agent in two phases:

  • Steadybit learns about the supported discoveries and their targets in the discovery registration phase. Once this phase is completed, discoveries will be scheduled within the agent.
  • The agent's scheduler will call the discovery in the execution phase.

The following sections explain the API endpoints, their responsibilities, and structures in more detail.

Index Response

As the name implies, this is the root of a discovery provider and returns a list of supported discoveries, target type, and target attribute descriptions. Or, more specifically, HTTP endpoints that the agent should call to learn more about them.

All paths will be resolved relative to the URL used to register the extension at the agent. For example, if https://extension/some-path was used to register and this endpoint returns /discoveries/cats, the agent will make the request to https://extension/some-path/discoveries/cats. This allows extensions to run behind reverse proxies, rewriting the path.

This endpoint needs to be registered with Steadybit agents.

Example

// Request: GET /

// Response: 200
{
  "discoveries": [
    {
      "path": "/discoveries/cats"
    },
    {
      "path": "/discoveries/dogs"
    }
  ],
  "targetTypes": [
    {
      "path": "/targettypes/cat"
    },
    {
      "path": "/targettypes/dog"
    }
  ],
  "targetAttributes": [
    {
      "path": "/targetattributes/pets"
    }
  ],
  "targetEnrichmentRules": [
    {
      "path": "/enrichmentrules/owner-to-cat"
    }
  ]
}

References

Discovery Description

A discovery description is required for each discovery. The HTTP endpoint serving the description is discovered through the index endpoint.

Discovery descriptions expose information about the endpoint, the call interval and an optional restriction where to run the discovery.

Example

// Request: GET /discoveries/cats

// Response: 200
{
  "id": "cats-discovery",
  "discover": {
    "path": "/discoveries/cats/discover",
    "callInterval": "10s"
  }
}

References

Target Type Description

The targetType specifies how the platform should display targets in the user interface. All attacks are associated with a single target type. Among others, this helps to narrow down the targets for an attack.

A common use-case is to define a custom targetType. Still, it's also possible to let your discovery report additional targets from one of Steadybit's pre-defined target types (for example, hosts, containers or Kubernetes deployments. If you report targets for existing target types, keep in mind that there are attacks that require specific attributes for attack execution.

Versioning

Target types are versioned strictly, and Steadybit will ignore definition changes for the same version. Remember to update the version every time you update the target type description.

Example

// Request: GET /targettypes/cat

// Response: 200
{
  "id": "com.steadybit.extension-pets.cat",
  "version": "1.0.0",
  "label": {
    "one": "cat",
    "other": "cats"
  },
  "category": "animals",
  "icon": "",
  "table": {
    "columns": [
      {
        "attribute": "pet.name"
      },
      {
        "attribute": "pet.age"
      },
      {
        "attribute": "pet.owner"
      }
    ],
    "orderBy": [
      {
        "attribute": "pet.name",
        "direction": "ASC"
      }
    ]
  }
}

References

Target Attribute Description

You can provide a list of attribute definitions. The platform will use these attribute definitions to render human-readable column headers or labels. Attribute definitions are optional. By default, all . are replaced by a space to generate human-readable labels.

Example

// Request: GET /targetattributes/pets

// Response: 200
{
  "attributes": [
    {
      "attribute": "pet.name",
      "label": {
        "one": "pet name",
        "other": "pet names"
      }
    },
    {
      "attribute": "pet.age",
      "label": {
        "one": "pet age",
        "other": "pet ages"
      }
    },
    {
      "attribute": "pet.owner",
      "label": {
        "one": "pet owner",
        "other": "pet owner"
      }
    }
  ]
}

References

Target Enrichment Rules

You can define a set of Target Enrichment Rules. Details about target enrichment can be found here.

Versioning

Target Enrichment Rulesets are versioned strictly, and Steadybit will ignore definition changes for the same version. Remember to update the version every time you update the target type description.

Example

// Request: GET /enrichmentrules/owner-to-cat

// Response: 200
{
  "id": "com.steadybit.extension-pets.owner-to-cat",
  "version": "1.0.0",
  "src": {
    "type": "com.steadybit.extension-pets.owner",
    "selector": {
      "owner.id": "${dest.pet.owner.id}"
    }
  },
  "dest": {
    "type": "com.steadybit.extension-pets.cat",
    "selector": {
      "pet.owner.id": "${src.owner.id}"
    }
  },
  "attributes": [
    {
      "matcher": "EQUALS",
      "name": "owner.name"
    },
    {
      "matcher": "EQUALS",
      "name": "owner.address"
    }
  ]
}

References

Discovery Execution

Discoveries are scheduled by the agent. Steadybit will use the callIntervall provided in the discovery description. The endpoint needs to return a list of all discovered targets or enrichment data.

Example

// Request: GET /discoveries/cats/discover

// Response: 200
{
  "targets": [
    {
      "id": "garfield",
      "label": "Garfield",
      "targetType": "com.steadybit.extension-pets.cat",
      "attributes": {
        "pet.name": [
          "Garfield"
        ],
        "pet.age": [
          "42"
        ],
        "pet.owner": [
          "Daniel"
        ]
      }
    },
    {
      "id": "kitty",
      "label": "Kitty",
      "targetType": "com.steadybit.extension-pets.cat",
      "attributes": {
        "pet.name": [
          "Kitty"
        ],
        "pet.age": [
          "0"
        ],
        "pet.owner": [
          "Ben"
        ]
      }
    }
  ]
}

Apply Deny List to Target Attributes

Use the function ApplyAttributeExcludes to filter out attributes from the targets that should not be exposed to the steadybit platform and should not be used the enhance other targets.

Excludes entries can be full attribute names and / or parts of the attribute key name with a trailing wildcard (*). (e.g.: aws.label.*)

import (
    "github.com/steadybit/discovery-kit/go/discovery_kit_api"
)
excludes := []string{"aws.label.*", "aws-ec2.instance.id"}] // From config or env variable

func getTargets(w http.ResponseWriter, _ *http.Request, _ []byte) {
      exthttp.WriteBody(w, discovery_kit_api.DiscoveryData{Targets: discovery_kit_commons.ApplyAttributeExcludes(targets, excludes)})
}

References