-
Notifications
You must be signed in to change notification settings - Fork 376
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #3734 from DataDog/anmarchenko/metrics_management
[SDTEST-409] Telemetry metrics data model
- Loading branch information
Showing
6 changed files
with
610 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,149 @@ | ||
# frozen_string_literal: true | ||
|
||
module Datadog | ||
module Core | ||
module Telemetry | ||
# Telemetry metrics data model (internal Datadog metrics for client libraries) | ||
module Metric | ||
def self.metric_id(type, name, tags = []) | ||
"#{type}::#{name}::#{tags.join(',')}" | ||
end | ||
|
||
# Base class for all metric types | ||
class Base | ||
attr_reader :name, :tags, :values, :common, :interval | ||
|
||
# @param name [String] metric name | ||
# @param tags [Array<String>|Hash{String=>String}] metric tags as hash of array of "tag:val" strings | ||
# @param common [Boolean] true if the metric is common for all languages, false for Ruby-specific metric | ||
# @param interval [Integer] metrics aggregation interval in seconds | ||
def initialize(name, tags: {}, common: true, interval: nil) | ||
@name = name | ||
@values = [] | ||
@tags = tags_to_array(tags) | ||
@common = common | ||
@interval = interval | ||
end | ||
|
||
def track(value); end | ||
|
||
def type; end | ||
|
||
def to_h | ||
# @type var res: Hash[Symbol, untyped] | ||
res = { | ||
metric: name, | ||
points: values, | ||
type: type, | ||
tags: tags, | ||
common: common | ||
} | ||
res[:interval] = interval if interval | ||
res | ||
end | ||
|
||
private | ||
|
||
def tags_to_array(tags) | ||
return tags if tags.is_a?(Array) | ||
|
||
tags.map { |k, v| "#{k}:#{v}" } | ||
end | ||
end | ||
|
||
# Count metric adds up all the submitted values in a time interval. This would be suitable for a | ||
# metric tracking the number of website hits, for instance. | ||
class Count < Base | ||
TYPE = 'count' | ||
|
||
def type | ||
TYPE | ||
end | ||
|
||
def inc(value = 1) | ||
track(value) | ||
end | ||
|
||
def dec(value = 1) | ||
track(-value) | ||
end | ||
|
||
def track(value) | ||
if values.empty? | ||
values << [Time.now.to_i, value] | ||
else | ||
values[0][0] = Time.now.to_i | ||
values[0][1] += value | ||
end | ||
end | ||
end | ||
|
||
# A gauge type takes the last value reported during the interval. This type would make sense for tracking RAM or | ||
# CPU usage, where taking the last value provides a representative picture of the host’s behavior during the time | ||
# interval. | ||
class Gauge < Base | ||
TYPE = 'gauge' | ||
|
||
def type | ||
TYPE | ||
end | ||
|
||
def track(value) | ||
if values.empty? | ||
values << [Time.now.to_i, value] | ||
else | ||
values[0][0] = Time.now.to_i | ||
values[0][1] = value | ||
end | ||
end | ||
end | ||
|
||
# The rate type takes the count and divides it by the length of the time interval. This is useful if you’re | ||
# interested in the number of hits per second. | ||
class Rate < Base | ||
TYPE = 'rate' | ||
|
||
def initialize(name, tags: {}, common: true, interval: nil) | ||
super | ||
|
||
@value = 0.0 | ||
end | ||
|
||
def type | ||
TYPE | ||
end | ||
|
||
def track(value = 1.0) | ||
@value += value | ||
|
||
rate = interval ? @value / interval : 0.0 | ||
@values = [[Time.now.to_i, rate]] | ||
end | ||
end | ||
|
||
# Distribution metric represents the global statistical distribution of a set of values. | ||
class Distribution < Base | ||
TYPE = 'distributions' | ||
|
||
def type | ||
TYPE | ||
end | ||
|
||
def track(value) | ||
values << value | ||
end | ||
|
||
# distribution metric data does not have type field | ||
def to_h | ||
{ | ||
metric: name, | ||
points: values, | ||
tags: tags, | ||
common: common | ||
} | ||
end | ||
end | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,102 @@ | ||
module Datadog | ||
module Core | ||
module Telemetry | ||
module Metric | ||
type metric_type = "count" | "gauge" | "rate" | "distributions" | nil | ||
|
||
type input_value = Integer | Float | ||
|
||
type metric_value = Array[input_value] | ||
type distribution_value = input_value | ||
|
||
type tags_input = ::Hash[String, String] | Array[String] | ||
|
||
def self.metric_id: (metric_type type, String name, ?Array[String] tags) -> ::String | ||
|
||
class Base | ||
@name: String | ||
|
||
@values: Array[untyped] | ||
|
||
@tags: Array[String] | ||
|
||
@common: bool | ||
|
||
@interval: Integer? | ||
|
||
attr_reader name: String | ||
|
||
attr_reader tags: Array[String] | ||
|
||
attr_reader values: Array[untyped] | ||
|
||
attr_reader common: bool | ||
|
||
attr_reader interval: Integer? | ||
|
||
def initialize: (String name, ?tags: tags_input, ?common: bool, ?interval: Integer?) -> void | ||
|
||
def track: (Numeric value) -> void | ||
|
||
def type: () -> metric_type | ||
|
||
def to_h: () -> Hash[Symbol, untyped] | ||
|
||
private | ||
|
||
def tags_to_array: (tags_input tags) -> Array[String] | ||
end | ||
|
||
class Count < Base | ||
TYPE: "count" | ||
|
||
@values: Array[metric_value] | ||
attr_reader values: Array[metric_value] | ||
|
||
def type: () -> "count" | ||
|
||
def inc: (?::Integer value) -> void | ||
|
||
def dec: (?::Integer value) -> void | ||
|
||
def track: (Integer value) -> void | ||
end | ||
|
||
class Gauge < Base | ||
TYPE: "gauge" | ||
|
||
def type: () -> "gauge" | ||
|
||
def track: (input_value value) -> void | ||
end | ||
|
||
class Rate < Base | ||
@value: Float | ||
|
||
@values: Array[metric_value] | ||
attr_reader values: Array[metric_value] | ||
|
||
TYPE: "rate" | ||
|
||
def initialize: (String name, ?tags: tags_input, ?common: bool, ?interval: Integer?) -> void | ||
|
||
def type: () -> "rate" | ||
|
||
def track: (?::Float value) -> void | ||
end | ||
|
||
class Distribution < Base | ||
TYPE: "distributions" | ||
|
||
@values: Array[distribution_value] | ||
attr_reader values: Array[distribution_value] | ||
|
||
def type: () -> "distributions" | ||
|
||
def track: (input_value value) -> void | ||
def to_h: () -> { metric: String, points: Array[distribution_value], tags: Array[String], common: bool } | ||
end | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.