Skip to content

Commit

Permalink
Eagerly verify Sample values are float-convertible
Browse files Browse the repository at this point in the history
Addresses #527.

There's a test setting GaugeHistogramMetricFamily.gsum_value = None,
so I've preserved that behaviour, by not appending and crashing if
gsum_value is None.

I was a bit unsure about this bit:

assert isinstance(exception.args[-1], core.Metric)

I'm not sure what exception.args[-1] is, the python docs for
TypeError and ValueErrordon't explain. I've removed the assertion.

Signed-off-by: Mark Hansen <mark@markhansen.co.nz>
  • Loading branch information
mhansen committed Mar 20, 2020
1 parent 0497442 commit 2c5b9db
Show file tree
Hide file tree
Showing 3 changed files with 32 additions and 21 deletions.
9 changes: 6 additions & 3 deletions prometheus_client/metrics_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -257,10 +257,13 @@ def add_metric(self, labels, buckets, gsum_value, timestamp=None):
dict(list(zip(self._labelnames, labels)) + [('le', bucket)]),
value, timestamp))
# +Inf is last and provides the count value.
self.samples.extend([
self.samples.append(
Sample(self.name + '_gcount', dict(zip(self._labelnames, labels)), buckets[-1][1], timestamp),
Sample(self.name + '_gsum', dict(zip(self._labelnames, labels)), gsum_value, timestamp),
])
)
if gsum_value is not None:
self.samples.append(
Sample(self.name + '_gsum', dict(zip(self._labelnames, labels)), gsum_value, timestamp),
)


class InfoMetricFamily(Metric):
Expand Down
11 changes: 8 additions & 3 deletions prometheus_client/samples.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,17 @@ def __gt__(self, other):


# Timestamp and exemplar are optional.
# Value can be an int or a float.
# Value must be a float.
# Timestamp can be a float containing a unixtime in seconds,
# a Timestamp object, or None.
# Exemplar can be an Exemplar object, or None.
Sample = namedtuple('Sample', ['name', 'labels', 'value', 'timestamp', 'exemplar'])
Sample.__new__.__defaults__ = (None, None)
sample = namedtuple('Sample', ['name', 'labels', 'value', 'timestamp', 'exemplar'])


# Wrap the namedtuple to provide eager type-checking that value is a float.
def Sample(name, labels, value, timestamp=None, exemplar=None):
return sample(name, labels, float(value), timestamp, exemplar)


Exemplar = namedtuple('Exemplar', ['labels', 'value', 'timestamp'])
Exemplar.__new__.__defaults__ = (None,)
33 changes: 18 additions & 15 deletions tests/test_exposition.py
Original file line number Diff line number Diff line change
Expand Up @@ -348,17 +348,6 @@ def collect(self):
return [self.metric_family]


def _expect_metric_exception(registry, expected_error):
try:
generate_latest(registry)
except expected_error as exception:
assert isinstance(exception.args[-1], core.Metric)
# Got a valid error as expected, return quietly
return

raise RuntimeError('Expected exception not raised')


@pytest.mark.parametrize('MetricFamily', [
core.CounterMetricFamily,
core.GaugeMetricFamily,
Expand All @@ -373,7 +362,12 @@ def _expect_metric_exception(registry, expected_error):
def test_basic_metric_families(registry, MetricFamily, value, error):
metric_family = MetricFamily(MetricFamily.__name__, 'help')
registry.register(Collector(metric_family, value))
_expect_metric_exception(registry, error)
try:
generate_latest(registry)
except error as exception:
# Got a valid error as expected, return quietly
return
raise RuntimeError('Expected exception not raised')


@pytest.mark.parametrize('count_value,sum_value,error', [
Expand All @@ -389,14 +383,18 @@ def test_basic_metric_families(registry, MetricFamily, value, error):
def test_summary_metric_family(registry, count_value, sum_value, error):
metric_family = core.SummaryMetricFamily('summary', 'help')
registry.register(Collector(metric_family, count_value, sum_value))
_expect_metric_exception(registry, error)
try:
generate_latest(registry)
except error as exception:
# Got a valid error as expected, return quietly
return
raise RuntimeError('Expected exception not raised')


@pytest.mark.parametrize('MetricFamily', [
core.GaugeHistogramMetricFamily,
])
@pytest.mark.parametrize('buckets,sum_value,error', [
([('spam', 0), ('eggs', 0)], None, TypeError),
([('spam', 0), ('eggs', None)], 0, TypeError),
([('spam', 0), (None, 0)], 0, AttributeError),
([('spam', None), ('eggs', 0)], 0, TypeError),
Expand All @@ -408,7 +406,12 @@ def test_summary_metric_family(registry, count_value, sum_value, error):
def test_histogram_metric_families(MetricFamily, registry, buckets, sum_value, error):
metric_family = MetricFamily(MetricFamily.__name__, 'help')
registry.register(Collector(metric_family, buckets, sum_value))
_expect_metric_exception(registry, error)
try:
generate_latest(registry)
except error as exception:
# Got a valid error as expected, return quietly
return
raise RuntimeError('Expected exception not raised')


if __name__ == '__main__':
Expand Down

0 comments on commit 2c5b9db

Please sign in to comment.