# PromQL Service PromQL([Prometheus Query Language](https://prometheus.io/docs/prometheus/latest/querying/basics/)) Service exposes Prometheus Querying HTTP APIs including the bundled PromQL expression system. Third-party systems or visualization platforms that already support PromQL (such as Grafana), could obtain metrics through PromeQL Service. As SkyWalking and Prometheus have fundamental differences in metrics classification, format, storage, etc. The PromQL Service supported will be a subset of the complete PromQL ## Details Of Supported Protocol The following doc describes the details of the supported protocol and compared it to the PromQL official documentation. If not mentioned, it will not be supported by default. ### Time series Selectors #### [Instant Vector Selectors](https://prometheus.io/docs/prometheus/latest/querying/basics/#instant-vector-selectors) For example: select metric `service_cpm` which the service is `$service` and the layer is `$layer`. ```text service_cpm{service='$service', layer='$layer'} ``` **Note: The label matching operators only support `=` instead of regular expressions.** #### [Range Vector Selectors](https://prometheus.io/docs/prometheus/latest/querying/basics/#range-vector-selectors) For example: select metric `service_cpm` which the service is `$service` and the layer is `$layer` within the last 5 minutes. ```text service_cpm{service='$service', layer='$layer'}[5m] ``` #### [Time Durations](https://prometheus.io/docs/prometheus/latest/querying/basics/#time-durations) | Unit | Definition | Support | |------|--------------|---------| | ms | milliseconds | yes | | s | seconds | yes | | m | minutes | yes | | h | hours | yes | | d | days | yes | | w | weeks | yes | | y | years | **no** | ### Binary operators #### [Arithmetic binary operators](https://prometheus.io/docs/prometheus/latest/querying/operators/#arithmetic-binary-operators) | Operator | Definition | Support | |----------|----------------------|---------| | + | addition | yes | | - | subtraction | yes | | * | multiplication | yes | | / | division | yes | | % | modulo | yes | | ^ | power/exponentiation | **no** | ##### Between two scalars For example: ```text 1 + 2 ``` ##### Between an instant vector and a scalar For example: ```text service_cpm{service='$service', layer='$layer'} / 100 ``` ##### Between two instant vectors For example: ```text service_cpm{service='$service', layer='$layer'} + service_cpm{service='$service', layer='$layer'} ``` **Note: The operations between vectors require the same metric and labels, and don't support [Vector matching](https://prometheus.io/docs/prometheus/latest/querying/operators/#vector-matching).** #### [Comparison binary operators](https://prometheus.io/docs/prometheus/latest/querying/operators/#comparison-binary-operators) | Operator | Definition | Support | |----------|------------------|---------| | == | equal | yes | | != | not-equal | yes | | \> | greater-than | yes | | < | less-than | yes | | \>= | greater-or-equal | yes | | <= | less-or-equal) | yes | ##### Between two scalars For example: ```text 1 > bool 2 ``` ##### Between an instant vector and a scalar For example: ```text service_cpm{service='$service', layer='$layer'} > 1 ``` ##### Between two instant vectors For example: ```text service_cpm{service='service_A', layer='$layer'} > service_cpm{service='service_B', layer='$layer'} ``` ### HTTP API #### Expression queries ##### [Instant queries](https://prometheus.io/docs/prometheus/latest/querying/api/#instant-queries) ```text GET|POST /api/v1/query ``` | Parameter | Definition | Support | Optional | |-----------|-------------------------------------------------------------------------------------------------------------------------------------|---------|------------| | query | prometheus expression | yes | no | | time | **The latest metrics value from current time to this time is returned. If time is empty, the default look-back time is 2 minutes.** | yes | yes | | timeout | evaluation timeout | **no** | **ignore** | For example: ```text /api/v1/query?query=service_cpm{service='agent::songs', layer='GENERAL'} ``` Result: ```json { "status": "success", "data": { "resultType": "vector", "result": [ { "metric": { "__name__": "service_cpm", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "value": [ 1677548400, "6" ] } ] } } ``` ##### [Range queries](https://prometheus.io/docs/prometheus/latest/querying/api/#range-queries) ```text GET|POST /api/v1/query_range ``` | Parameter | Definition | Support | Optional | |-----------|--------------------------------------------------------------------------------------|---------|------------| | query | prometheus expression | yes | no | | start | start timestamp, **seconds** | yes | no | | end | end timestamp, **seconds** | yes | no | | step | **SkyWalking will automatically fit Step(DAY, HOUR, MINUTE) through start and end.** | **no** | **ignore** | | timeout | evaluation timeout | **no** | **ignore** | For example: ```text /api/v1/query_range?query=service_cpm{service='agent::songs', layer='GENERAL'}&start=1677479336&end=1677479636 ``` Result: ```json { "status": "success", "data": { "resultType": "matrix", "result": [ { "metric": { "__name__": "service_cpm", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "values": [ [ 1677479280, "18" ], [ 1677479340, "18" ], [ 1677479400, "18" ], [ 1677479460, "18" ], [ 1677479520, "18" ], [ 1677479580, "18" ] ] } ] } } ``` #### Querying metadata ##### [Finding series by label matchers](https://prometheus.io/docs/prometheus/latest/querying/api/#finding-series-by-label-matchers) ```text GET|POST /api/v1/series ``` | Parameter | Definition | Support | Optional | |-----------|------------------------------|---------|----------| | match[] | series selector | yes | no | | start | start timestamp, **seconds** | yes | no | | end | end timestamp, **seconds** | yes | no | For example: ```text /api/v1/series?match[]=service_traffic{layer='GENERAL'}&start=1677479336&end=1677479636 ``` Result: ```json { "status": "success", "data": [ { "__name__": "service_traffic", "service": "agent::songs", "scope": "Service", "layer": "GENERAL" }, { "__name__": "service_traffic", "service": "agent::recommendation", "scope": "Service", "layer": "GENERAL" }, { "__name__": "service_traffic", "service": "agent::app", "scope": "Service", "layer": "GENERAL" }, { "__name__": "service_traffic", "service": "agent::gateway", "scope": "Service", "layer": "GENERAL" }, { "__name__": "service_traffic", "service": "agent::frontend", "scope": "Service", "layer": "GENERAL" } ] } ``` **Note: SkyWalking's metadata exists in the following metrics(traffics):** - service_traffic - instance_traffic - endpoint_traffic #### [Getting label names](https://prometheus.io/docs/prometheus/latest/querying/api/#getting-label-names) ```text GET|POST /api/v1/labels ``` | Parameter | Definition | Support | Optional | |-----------|-----------------|---------|----------| | match[] | series selector | yes | yes | | start | start timestamp | **no** | yes | | end | end timestamp | **no** | yes | For example: ```text /api/v1/labels?match[]=instance_jvm_cpu' ``` Result: ```json { "status": "success", "data": [ "layer", "scope", "top_n", "order", "service_instance", "parent_service" ] } ``` #### [Querying label values](https://prometheus.io/docs/prometheus/latest/querying/api/#querying-label-values) ```text GET /api/v1/label//values ``` | Parameter | Definition | Support | Optional | |-----------|-----------------|---------|----------| | match[] | series selector | yes | no | | start | start timestamp | **no** | yes | | end | end timestamp | **no** | yes | For example: ```text /api/v1/label/__name__/values ``` Result: ```json { "status": "success", "data": [ "meter_mysql_instance_qps", "service_cpm", "envoy_cluster_up_rq_active", "instance_jvm_class_loaded_class_count", "k8s_cluster_memory_requests", "meter_vm_memory_used", "meter_apisix_sv_bandwidth_unmatched", "meter_vm_memory_total", "instance_jvm_thread_live_count", "instance_jvm_thread_timed_waiting_state_thread_count", "browser_app_page_first_pack_percentile", "instance_clr_max_worker_threads", ... ] } ``` #### [Querying metric metadata](https://prometheus.io/docs/prometheus/latest/querying/api/#querying-metric-metadata) ```text GET /api/v1/metadata ``` | Parameter | Definition | Support | Optional | |-----------|---------------------------------------------|---------|----------| | limit | maximum number of metrics to return | yes | **yes** | | metric | **metric name, support regular expression** | yes | **yes** | For example: ```text /api/v1/metadata?limit=10 ``` Result: ```json { "status": "success", "data": { "meter_mysql_instance_qps": [ { "type": "gauge", "help": "", "unit": "" } ], "meter_apisix_sv_bandwidth_unmatched": [ { "type": "gauge", "help": "", "unit": "" } ], "service_cpm": [ { "type": "gauge", "help": "", "unit": "" } ], ... } } ``` ## Metrics Type For Query ### Supported Metrics [Scope](../../../oap-server/server-core/src/main/java/org/apache/skywalking/oap/server/core/query/enumeration/Scope.java)(Catalog) All scopes are not supported completely, please check the following table: | Scope | Support | |-------------------------|---------| | Service | yes | | ServiceInstance | yes | | Endpoint | yes | | ServiceRelation | no | | ServiceInstanceRelation | no | | Process | no | | ProcessRelation | no | ### General labels Each metric contains general labels: [layer](../../../oap-server/server-core/src/main/java/org/apache/skywalking/oap/server/core/analysis/Layer.java). Different metrics will have different labels depending on their Scope and metric value type. | Query Labels | Scope | Expression Example | |----------------------------------|-----------------|------------------------------------------------------------------------------------------------| | layer, service | Service | service_cpm{service='$service', layer='$layer'} | | layer, service, service_instance | ServiceInstance | service_instance_cpm{service='$service', service_instance='$service_instance', layer='$layer'} | | layer, service, endpoint | Endpoint | endpoint_cpm{service='$service', endpoint='$endpoint', layer='$layer'} | ### Common Value Metrics - Query Labels: ```text {General labels} ``` - Expression Example: ```text service_cpm{service='agent::songs', layer='GENERAL'} ``` - Result (Instant Query): ```json { "status": "success", "data": { "resultType": "vector", "result": [ { "metric": { "__name__": "service_cpm", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "value": [ 1677490740, "3" ] } ] } } ``` ### Labeled Value Metrics - Query Labels: ```text --{General labels} --labels: Used to filter the value labels to be returned --relabels: Used to rename the returned value labels note: The number and order of labels must match the number and order of relabels. ``` - Expression Example: ```text service_percentile{service='agent::songs', layer='GENERAL', labels='0,1,2', relabels='P50,P75,P90'} ``` - Result (Instant Query): ```json { "status": "success", "data": { "resultType": "vector", "result": [ { "metric": { "__name__": "service_percentile", "label": "P50", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "value": [ 1677493380, "0" ] }, { "metric": { "__name__": "service_percentile", "label": "P75", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "value": [ 1677493380, "0" ] }, { "metric": { "__name__": "service_percentile", "label": "P90", "layer": "GENERAL", "scope": "Service", "service": "agent::songs" }, "value": [ 1677493380, "0" ] } ] } } ``` ### Sort Metrics - Query Labels: ```text --parent_service: Name of the parent service. --top_n: The max number of the selected metric value --order: ASC/DES ``` - Expression Example: ```text service_instance_cpm{parent_service='agent::songs', layer='GENERAL', top_n='10', order='DES'} ``` - Result (Instant Query): ```json { "status": "success", "data": { "resultType": "vector", "result": [ { "metric": { "__name__": "service_instance_cpm", "layer": "GENERAL", "scope": "ServiceInstance", "service_instance": "651db53c0e3843d8b9c4c53a90b4992a@10.4.0.28" }, "value": [ 1677494280, "14" ] }, { "metric": { "__name__": "service_instance_cpm", "layer": "GENERAL", "scope": "ServiceInstance", "service_instance": "4c04cf44d6bd408880556aa3c2cfb620@10.4.0.232" }, "value": [ 1677494280, "6" ] }, { "metric": { "__name__": "service_instance_cpm", "layer": "GENERAL", "scope": "ServiceInstance", "service_instance": "f5ac8ead31af4e6795cae761729a2742@10.4.0.236" }, "value": [ 1677494280, "5" ] } ] } } ``` ### Sampled Records - Query Labels: ```text --parent_service: Name of the parent service --top_n: The max number of the selected records value --order: ASC/DES ``` - Expression Example: ```text top_n_database_statement{parent_service='localhost:-1', layer='VIRTUAL_DATABASE', top_n='10', order='DES'} ``` - Result (Instant Query): ```json { "status": "success", "data": { "resultType": "vector", "result": [ { "metric": { "__name__": "top_n_database_statement", "layer": "VIRTUAL_DATABASE", "scope": "Service", "record": "select song0_.id as id1_0_, song0_.artist as artist2_0_, song0_.genre as genre3_0_, song0_.liked as liked4_0_, song0_.name as name5_0_ from song song0_ where song0_.liked>?" }, "value": [ 1677501360, "1" ] }, { "metric": { "__name__": "top_n_database_statement", "layer": "VIRTUAL_DATABASE", "scope": "Service", "record": "select song0_.id as id1_0_, song0_.artist as artist2_0_, song0_.genre as genre3_0_, song0_.liked as liked4_0_, song0_.name as name5_0_ from song song0_ where song0_.liked>?" }, "value": [ 1677501360, "1" ] }, { "metric": { "__name__": "top_n_database_statement", "layer": "VIRTUAL_DATABASE", "scope": "Service", "record": "select song0_.id as id1_0_, song0_.artist as artist2_0_, song0_.genre as genre3_0_, song0_.liked as liked4_0_, song0_.name as name5_0_ from song song0_ where song0_.liked>?" }, "value": [ 1677501360, "1" ] } ] } } ```