An Agent Check is a Python class that inherits from AgentCheck
and implements
the check
method:
from datadog_checks.checks import AgentCheck
class MyCheck(AgentCheck):
def check(self, instance):
# Do the check. Collect metrics, emit events, submit service checks,
# ...
The Agent creates an object of type MyCheck
for each element contained in the
instances
sequence within the corresponding config file:
instances:
- host: localhost
port: 6379
- host: example.com
port: 6379
Any mapping contained in instances
is passed to the check
method through the
named parameter instance
. The check
method will be invoked at every run of the
collector.
The AgentCheck
base class provides some useful attributes and methods:
self.name
: the name of the checkself.init_config
: theinit_config
section from the check configuration.self.log
: a logger.get_instance_proxy()
: a function returning a dictionary containing information on the Proxy being used
Warning: when loading a Python check, the Agent will inspect the available
modules and packages searching for a subclass of AgentCheck
. If such a class
exists but has been derived in turn, it'll be ignored - you should never derive from an existing Check.
In the event of a wrong configuration, a runtime error or in any case when it can't work properly, a Check should raise a meaningful exception. Exceptions are logged and being shown in the Agent status page to help diagnose problems.
The warning
method will log a warning message and display it in the Agent's
status page.
self.warning("This will be visible in the status page")
The self.log
field is a Logger
instance that prints to the Agent's main log file. You can set the log level in
the Agent config file datadog.yaml
.
You can call the following methods from anywhere in your check:
self.gauge(name, value, tags, hostname): # Sample a gauge metric
self.count(name, value, tags, hostname): # Sample a raw count metric
self.rate(name, value, tags, hostname): # Sample a point, with the rate calculated at the end of the check
self.increment(name, value, tags, hostname): # Increment a counter metric
self.decrement(name, value, tags, hostname): # Decrement a counter metric
self.histogram(name, value, tags, hostname): # Sample a histogram metric
self.historate(name, value, tags, hostname): # Sample a histogram based on rate metrics
self.monotonic_count(name, value, tags, hostname): # Sample an increasing counter metric
Each method takes the following arguments:
metric
: The name of the metricvalue
: The value for the metric (defaults to 1 on increment, -1 on decrement)tags
: (optional) A list of tags to associate with this metric.hostname
: (optional) A hostname to associate with this metric. Defaults to the current host.
The device_name
argument has been deprecated from Agent 5; you may add a
device:<device_id>
tag in the tags
list.
At the end of your check
function, all metrics that were submitted will be
collected and flushed out with the other Agent metrics.
To learn more about metric types, read the metric page.
You can call self.event(event_dict)
from anywhere in your check. The
event_dict
parameter is a dictionary with the following keys and data types:
{
"timestamp": int, # the epoch timestamp for the event
"event_type": string, # the event name
"api_key": string, # the api key for your account
"msg_title": string, # the title of the event
"msg_text": string, # the text body of the event
"aggregation_key": string, # a key to use for aggregating events
"alert_type": string, # (optional) one of ('error', 'warning', 'success', 'info'), defaults to 'info'
"source_type_name": string, # (optional) the source type name
"host": string, # (optional) the name of the host
"tags": list, # (optional) a list of tags to associate with this event
"priority": string, # (optional) specifies the priority of the event ("normal" or "low")
}
At the end of your check
function, all events will be collected and flushed with the
rest of the Agent payload.
Your check can also report the status of a service by calling the service_check
method:
self.service_check(name, status, tags=None, message="")
The method will accept the following arguments:
name
: The name of the service check.status
: An constant describing the service status defined in theAgentCheck
class:AgentCheck.OK
for successAgentCheck.WARNING
for warningAgentCheck.CRITICAL
for failureAgentCheck.UNKNOWN
for indeterminate status
tags
: (optional) A list of tags to associate with this check.message
: (optional) Additional information or a description of why this status occurred.
In general, you don't need and you should not override anything from the base
class except the check
method but sometimes it might be useful for a Check to
have its own constructor.
When overriding __init__
you have to remember that depending on the configuration,
the Agent might create several different Check instances and the method would be
called as many times.
To ease the porting of existing Check to the new Agent, the __init__
method in
AgentCheck
was implemented with the following signature:
def __init__(self, *args, **kwargs):
When overriding, the following convention must be followed:
from datadog_checks.checks import AgentCheck
class MyCheck(AgentCheck):
def __init__(self, name, init_config, instances):
super(MyCheck, self).__init__(name, init_config, instances)
The arguments that needs to be received and then passed to super
are the
following:
name
: the name of the check._init_config
: the init_config section of the configuration.instances
: a one-element list containing the instance options from the configuration file (to be backwards compatible with agent5 checks we have to pass a list here).
Due to the Python interpreter being embedded in an inherently multi-threaded environment (the go runtime) there are some limitations to the ways in which Python Checks can run subprocesses.
To run a subprocess from your Check, please use the get_subprocess_output
function
provided in datadog_checks.utils.subprocess_output
:
from datadog_checks.utils.subprocess_output import get_subprocess_output
class MyCheck(AgentCheck):
def check(self, instance):
# [...]
out, err, retcode = get_subprocess_output(cmd, self.log, raise_on_empty_output=True)
Using the subprocess
and multiprocessing
modules provided by the python standard library is not
supported, and may result in your Agent crashing and/or creating processes that remain in a stuck or zombie
state.