A Python wrapper for the Librato Metrics API.
In your shell:
$ easy_install librato-metrics
or
$ pip install librato-metrics
From your application or script:
import librato
We first use our credentials to connect to the API. I am assuming you have a librato account for Metrics. Go to your account settings page and save your username (email address) and token (long hexadecimal string).
api = librato.connect(user, token)
To iterate over your metrics:
for m in api.list_metrics():
print m.name
Let's now create a Metric:
api.submit("temperature", 10, description="temperature at home")
By default submit()
will create a gauge metric. The metric will be
created automatically by the server if it does not exist. We can remove it:
api.delete("temperature")
For creating a counter metric, we can:
api.submit("connections", 20, type='counter', description="server connections")
And again to remove:
api.delete("connections")
To iterate over your metrics:
for m in api.list_metrics():
print "%s: %s" % (m.name, m.description)
To retrieve a specific metric:
gauge = api.get("temperature")
counter = api.get("connections")
For sending more measurements:
for temp in [20, 21, 22]:
api.submit('temperature', temp)
for num_con in [100, 200, 300]:
api.submit('connections', num_con, type='counter')
Let's now iterate over the measurements of our Metrics:
metric = api.get("temperature", count=100, resolution=1)
for m in metric.measurements['unassigned']:
print "%s: %s" % (m['value'], m['measure_time'])
Notice a couple of things here. First, we are using the key unassigned
since
we have not associated our measurements to any source. Read more about it in
the API documentation. In addition, notice how
we are passing the count and resolution parameters to make sure the API
returns measurements in its answer and not only the metric properties.
Read more about them here.
Sending a measurement in a single HTTP request is inefficient. The overhead both at protocol and backend level is very high. That's why we provide an alternative method to submit your measurements. The idea is to send measurements in batch mode. We push measurements that are stored and when we are ready, they will be submitted in an efficient matter. Here is an example:
api = librato.connect(user, token)
q = api.new_queue()
q.add('temperature', 22.1, source='upstairs')
q.add('temperature', 23.1, source='dowstairs')
q.add('num_requests', 100, type='counter', source='server1')
q.add('num_requests', 102, type='counter', source='server2')
q.submit()
Queues can also be used as context managers. Once the context block is complete the queue
is submitted automatically. This is true even if an exception interrupts flow. In the
example below if potentially_dangerous_operation
causes an exception the queue will
submit the first measurement as it was the only one successfully added.
If the operation succeeds both measurements will be submitted.
api = librato.connect(user, token)
with api.new_queue() as q:
q.add('temperature', 22.1, source='upstairs')
potentially_dangerous_operation()
q.add('num_requests', 100, type='counter', source='server1')
Queues by default will collect metrics until they are told to submit. You may create a queue that autosubmits based on metric volume.
api = librato.connect(user, token)
# Submit when the 400th metric is queued
q = api.new_queue(auto_submit=400)
You can update the information for a metric by using the update
method,
for example:
api = librato.connect(user, token)
for metric in api.list_metrics():
gauge = api.get(m.name)
attrs = gauge.attributes
attrs['display_units_long'] = 'ms'
api.update(metric.name, attributes=attrs)
List Annotation all annotation streams:
for stream in api.list_annotation_streams
print "%s:%s" % (stream.name,stream.display_name)
View the metadata on a named annotation stream:
stream=api.get_annotation_stream("api.pushes")
print stream
Retrieve all of the events inside a named annotation stream, by adding a start_time parameter to the get_annotation_stream() call:
stream=api.get_annotation_stream("api.pushes",start_time="1386050400")
for source in stream.events:
print source
events=stream.events[source]
for event in events:
print event['id']
print event['title']
print event['description']
Submit a new annotation to a named annotation stream (creates the stream if it doesn't exist). Title is a required parameter, and all other parameters are optional
api.post_annotation("testing",title="foobarbiz")
api.post_annotation("TravisCI",title="build %s"%travisBuildID, source=SystemSource, description="Application %s, Travis build %s"%(appName,travisBuildID))
Delete a named annotation stream:
api.delete_annotation_stream("testing")
List Alerts
for alert in api.list_alerts(version=2, name="some.name"):
print "%s: %s" % (alert.id, alert.name)
Get a particular Alert and Delete it
alert = api.get_alert(alert_id, version=2)
status = api.delete_alert(alert)
print "delete status is %s" % status
Create an Alert
conditions = [
{'condition_type': 'below',
'metric_name': 'some.metric.name',
'source': 'some.source',
'threshold': 1,
'duration': 300,
'summary_function': 'average'
}
]
services = [service_1_id, service_2_id]
new_alert = api.create_alert("new.alert.name", version=2, conditions=conditions, services=services)
This is a modified version of Librato's python-librato