Querying Prometheus

    Prometheus provides a functional query language called PromQL (Prometheus Query Language) that lets the user select and aggregate time series data in real time. The result of an expression can either be shown as a graph, viewed as tabular data in Prometheus’s expression browser, or consumed by external systems via the .

    This document is meant as a reference. For learning, it might be easier to start with a couple of examples.

    Expression language data types

    In Prometheus’s expression language, an expression or sub-expression can evaluate to one of four types:

    • Instant vector - a set of time series containing a single sample for each time series, all sharing the same timestamp
    • Range vector - a set of time series containing a range of data points over time for each time series
    • String - a simple string value; currently unused

    Depending on the use-case (e.g. when graphing vs. displaying the output of an expression), only some of these types are legal as the result from a user-specified expression. For example, an expression that returns an instant vector is the only type that can be directly graphed.

    Literals

    Strings may be specified as literals in single quotes, double quotes or backticks.

    PromQL follows the same . In single or double quotes a backslash begins an escape sequence, which may be followed by , b, f, n, r, t, v or \. Specific characters can be provided using octal (\nnn) or hexadecimal (\xnn, \unnnn and \Unnnnnnnn).

    No escaping is processed inside backticks. Unlike Go, Prometheus does not discard newlines inside backticks.

    Example:

    Float literals

    Scalar float values can be written as literal integer or floating-point numbers in the format (whitespace only included for better readability):

    1. [-+]?(
    2. [0-9]*\.?[0-9]+([eE][-+]?[0-9]+)?
    3. | 0[xX][0-9a-fA-F]+
    4. | [nN][aA][nN]
    5. | [iI][nN][fF]
    6. )

    Examples:

    1. 23
    2. -2.43
    3. 3.4e-9
    4. 0x8f
    5. -Inf
    6. NaN

    Instant vector selectors allow the selection of a set of time series and a single sample value for each at a given timestamp (instant): in the simplest form, only a metric name is specified. This results in an instant vector containing elements for all time series that have this metric name.

    This example selects all time series that have the http_requests_total metric name:

    1. http_requests_total

    It is possible to filter these time series further by appending a comma separated list of label matchers in curly braces ({}).

    This example selects only those time series with the http_requests_total metric name that also have the job label set to and their group label set to canary:

    1. http_requests_total{job="prometheus",group="canary"}
    • =: Select labels that are exactly equal to the provided string.
    • !=: Select labels that are not equal to the provided string.
    • =~: Select labels that regex-match the provided string.
    • !~: Select labels that do not regex-match the provided string.

    For example, this selects all http_requests_total time series for staging, testing, and development environments and HTTP methods other than GET.

      Label matchers that match empty label values also select all time series that do not have the specific label set at all. Regex-matches are fully anchored. It is possible to have multiple matchers for the same label name.

      Vector selectors must either specify a name or at least one label matcher that does not match the empty string. The following expression is illegal:

      In contrast, these expressions are valid as they both have a selector that does not match empty label values.

      1. {job=~".+"} # Good!
      2. {job=~".*",method="get"} # Good!

      Label matchers can also be applied to metric names by matching against the internal __name__ label. For example, the expression http_requests_total is equivalent to {__name__="http_requests_total"}. Matchers other than = (!=, =~, !~) may also be used. The following expression selects all metrics that have a name starting with job::

      1. {__name__=~"job:.*"}

      The metric name must not be one of the keywords bool, on, ignoring, group_left and . The following expression is illegal:

      1. on{} # Bad!

      A workaround for this restriction is to use the __name__ label:

      1. {__name__="on"} # Good!

      All regular expressions in Prometheus use RE2 syntax.

      Range Vector Selectors

      Range vector literals work like instant vector literals, except that they select a range of samples back from the current instant. Syntactically, a is appended in square brackets ([]) at the end of a vector selector to specify how far back in time values should be fetched for each resulting range vector element.

      In this example, we select all the values we have recorded within the last 5 minutes for all time series that have the metric name http_requests_total and a job label set to prometheus:

      1. http_requests_total{job="prometheus"}[5m]

      Time durations are specified as a number, followed immediately by one of the following units:

      • ms - milliseconds
      • s - seconds
      • m - minutes
      • h - hours
      • d - days - assuming a day has always 24h
      • w - weeks - assuming a week has always 7d
      • y - years - assuming a year has always 365d

      Time durations can be combined, by concatenation. Units must be ordered from the longest to the shortest. A given unit must only appear once in a time duration.

      Here are some examples of valid time durations:

      Offset modifier

      The offset modifier allows changing the time offset for individual instant and range vectors in a query.

      1. http_requests_total offset 5m

      Note that the offset modifier always needs to follow the selector immediately, i.e. the following would be correct:

      1. sum(http_requests_total{method="GET"} offset 5m) // GOOD.

      While the following would be incorrect:

      1. sum(http_requests_total{method="GET"}) offset 5m // INVALID.

      The same works for range vectors. This returns the 5-minute rate that http_requests_total had a week ago:

      1. rate(http_requests_total[5m] offset 1w)

      Subquery

      Subquery allows you to run an instant query for a given range and resolution. The result of a subquery is a range vector.

      Syntax: <instant_query> '[' <range> ':' [<resolution>] ']' [ offset <duration> ]

      • <resolution> is optional. Default is the global evaluation interval.

      Operators

      Prometheus supports many binary and aggregation operators. These are described in detail in the expression language operators page.

      Prometheus supports several functions to operate on data. These are described in detail in the page.

      Comments

      PromQL supports line comments that start with #. Example:

      1. # This is a comment

      Gotchas

      When queries are run, timestamps at which to sample data are selected independently of the actual present time series data. This is mainly to support cases like aggregation (sum, , and so on), where multiple aggregated time series do not exactly align in time. Because of their independence, Prometheus needs to assign a value at those timestamps for each relevant time series. It does so by simply taking the newest sample before this timestamp.

      If a target scrape or rule evaluation no longer returns a sample for a time series that was previously present, that time series will be marked as stale. If a target is removed, its previously returned time series will be marked as stale soon afterwards.

      If a query is evaluated at a sampling timestamp after a time series is marked stale, then no value is returned for that time series. If new samples are subsequently ingested for that time series, they will be returned as normal.

      If no sample is found (by default) 5 minutes before a sampling timestamp, no value is returned for that time series at this point in time. This effectively means that time series “disappear” from graphs at times where their latest collected sample is older than 5 minutes or after they are marked stale.

      Staleness will not be marked for time series that have timestamps included in their scrapes. Only the 5 minute threshold will be applied in that case.

      Avoiding slow queries and overloads

      If a query needs to operate on a very large amount of data, graphing it might time out or overload the server or browser. Thus, when constructing queries over unknown data, always start building the query in the tabular view of Prometheus’s expression browser until the result set seems reasonable (hundreds, not thousands, of time series at most). Only when you have filtered or aggregated your data sufficiently, switch to graph mode. If the expression still takes too long to graph ad-hoc, pre-record it via a recording rule.