Metrics
Introduction
Mendix supports reporting metrics through Micrometer.
The Metrics can be configured in the following ways:
- Metrics registries configuration – to configure the metrics supported by Mendix
- Application tags – to include tags in all the metrics in application level
- Microflow activities – to handle metrics in microflow activities
- Java API – to handle metrics using a Java API in a Java action
- Logging – to log metrics to a log node
Metrics Registries Configuration
Micrometer can send metrics to multiple registries. To configure micrometer for a specific registry, use the following syntax in runtime settings
with the custom runtime setting name Metrics.Registries
. See Runtime Customization for more information. The setting is in JSON format.
- Custom Runtime Setting – Name:
Metrics.Registries
[
{
"type": "<registry1>",
"settings": { <settings> },
"filters": [ <list-of-filters> ]
},
{
"type": "<registry2>",
"settings": { <settings> },
"filters": [ <list-of-filters> ]
},
…
]
The details of each settings are listed below.
type
(mandatory) – the type of registry to use. Currently supported types areprometheus
,jmx
,influx
,statsd
, andopentelemetry
. Depending on the type of the registry thesettings
may vary.settings
(conditional mandatory) – settings for the registry. Each registry has different settings depending upon the type specified. Follow the links below to see the settings for each type:filters
(optional) – instructions on which metrics to accept or deny. See the Filters section, below, for more information.
Settings
The following settings can be used, depending on the type of metrics being generated:
Setting | DataType | Manda-tory | Type | Description | Default Value | Examples |
---|---|---|---|---|---|---|
db |
String | No | influx | the database to which to send the metrics | mydb | customDb, metricDb |
password |
String | Yes | influx | Authenticate requests with this password | - | - |
uri |
String | No | influx | The URI for the back end | http://localhost:8086 (for Influx) | - |
url |
String | No | opentelemetry | The URL for the back end | http://localhost:4318/v1/metrics | - |
userName |
String | Yes | influx | Authenticate requests with this user | - | - |
protocol |
String | No | influx | Protocol of the statsd connection | UDP | TCP, UDP |
domain |
String | No | jmx | Jmx domain to publish the metrics to | metrics | “Mendix”, “Employee” |
enabled |
Boolean | No | influx / statsd | Enables / Disables the meter | true | true, false |
flavor |
StatsdFlavor | No | statsd | The variant of the StatsD protocol | DATADOG | ETSY, TELEGRAF, SYSDIG |
host |
String | No | statsd | The host name of the StatsD agent | localhost | - |
port |
Int | No | statsd | The port of the StatsD agent | 8125 | - |
step |
Duration | No | all | The step size (reporting frequency) to use | 1m | 1ms , 2s , 3m , 4h , 5d or ISO8601 -> P3Y6M4DT12H30M5S |
filters |
Json | No | all | Custom setting from Mendix to filter metrics | - | See below |
aggregationTemporality |
String | No | opentelemetry | Determines how the additive quantities are expressed, in relation to time | cumulative |
cumulative or delta |
headers |
String | No | opentelemetry | Additional headers to send with exported metrics, this can be used for authorization headers | - | “header1=value1,header2=value2” |
resourceAttributes |
String | No | opentelemetry | List of attributes which can be used for including information about the environment | - | “attribute1=value1,attribute2=value2” |
Prometheus
We do not support multiple Prometheus. When the Prometheus registry is set, it can be accessed through the /prometheus
context path over the admin endpoint.
step
– the step size or reporting frequency to use.
Example 1
[
{
"type": "prometheus",
"settings": {
"step": "3m"
}
}
]
Example 2
[
{
"type": "prometheus",
"settings": {
"step": "P3Y6M4DT12H30M5S"
}
}
]
Jmx
step
– The step size or reporting frequency to use.domain
– The Jmx domain to which to publish the metrics.
Example 1
[
{
"type": "jmx",
"settings": {
"step": "3m",
"domain": "Mendix"
}
}
]
Example 2
[
{
"type": "jmx",
"settings": {
"step": "P3Y6M4DT12H30M5S"
}
}
]
Influx
uri
– the URI for the Influx back end.db
– the database name to which to send the metrics.userName
– the userName for authentication.password
– the password for authentication.step
– the step size or reporting frequency to use.enabled
– set totrue
to enable the registry. This means you can switch the meter on and off while keeping the settings in the configuration.
Example 1
[
{
"type": "influx",
"settings": {
"userName": "mendix",
"password": "MayBeThis**"
}
}
]
Example 2
[
{
"type": "influx",
"settings": {
"uri": "http://mendix.influx.com",
"db": "mx.metrics",
"userName": "mendix",
"password": "MayBeThis**",
"step": "P2Y6M4DT12H21M5S"
}
}
]
StatsD
flavor
– specifies the variant of the StatsD protocol to use.host
– the host name of the StatsD agent.port
– the port used by the StatsD agent.step
– the step size or reporting frequency to use.protocol
– the protocol of the connection.enabled
– set totrue
to enable the registry. This means you can switch the meter on and off while keeping the settings in the configuration.
Example 1
[
{
"type": "statsd",
"settings": {
"host": "mx.datadog.com",
"port": "8181",
"protocol": "TCP",
"flavor": "TELEGRAF"
}
}
]
Example 2
[
{
"type": "statsd",
"settings": {
"flavor": "SYSDIG",
"step": "3m"
}
}
]
OpenTelemetry
url
– the OpenTelemetry metrics endpoint url to which data is reported.aggregationTemporality
– determines how the additive quantities are expressed, in relation to time. The supported values arecumulative
ordelta
. Defaults tocumulative
.headers
– additional headers to send with exported metrics, this can be used for authorization headers.step
– the step size or reporting frequency to use.resourceAttributes
– additional attributes used for all metrics published. You can use this, for example, to include information about the environment.enabled
– set totrue
to enable the registry. This means you can switch the meter on and off while keeping the settings in the configuration.
Example 1
[
{
"type": "opentelemetry",
"settings": {
"url": "http://localhost:4318/v1/metrics",
"step": "30s"
}
}
]
Example 2
[
{
"type": "opentelemetry",
"settings": {
"url": "https://otlp.example.com:4318/v1/metrics",
"aggregationTemporality": "delta",
"step": "20s",
"headers": "header1=value1,header2=value2",
"resourceAttributes": "service.name=shoppingcart,env=test"
}
}
]
Filters
Filters are optional, but can help in filtering metrics based on given criteria. Below is the syntax:
"filters" : [
{"type" : "<filter-type>", "result" : "filter-result", "values": [ "<list-of-filter-values>"] }
]
filter-type
– the type of filter to apply. Currently we support the following:nameStartsWith
– matches the metric if its name begins with the given list-of-filter-values.
filter-result
– the desired result of applying the filter-type to the list-of-filter-values. Supported values are:accept
– metrics matching the criteria are passed to the registrydeny
– metrics matching the criteria are not passed to the registry
list-of-filter-values
– a list of values used in the given filter type
Example 1
[
{
"type": "statsd",
"settings": { "protocol": "TCP" },
"filters": [
{ "type" : "nameStartsWith", "result" : "accept", "values" : ["app.", "employee.", "myapp."] }
]
}
]
The above filter accepts metrics which starts with app.
, employee.
, or myapp.
.
Example 2
[
{
"type": "influx",
"settings": {
"userName": "mendix",
"password": "MayBeThis**"
},
"filters": [
{ "type" : "nameStartsWith", "result" : "deny", "values" : ["Unnamed.", "Invalid.", "Internal."] }
]
}
]
The above filter discards metrics which start with Unnamed.
, Invalid.
, or Internal.
.
Notes
-
Filters also affect internal metrics used by Mendix. For example, metrics emitted by Mendix which start with
mx.
. -
If you have two metrics with the same name but one has additional tags, these are considered as different metrics. For example, metrics
app.counter1
with tag"version" -> "1"
andapp.counter1
with no tag are different. -
Filters are executed on a first come, first served basis. In other words, the first matching filter gets the priority. For example, take the filters defined below:
[ { "type": "statsd", "settings": { "protocol": "TCP" }, "filters": [ { "type" : "nameStartsWith", "result" : "accept", "values" : ["app."] }, { "type" : "nameStartsWith", "result" : "deny", "values" : ["app.others."] } ] } ]
This example allows metrics like
app.others.counter
as they pass the firstaccept
filter,nameStartsWith "app."
. However, if you reverse the filters,app.others.counter
would be denied, whileapp.somethingelse.
would still be accepted.Ensure that more specific filters are defined before less specific ones so that they are applied correctly. -
To accept only specific filters and deny all others, add a deny filter with the value
""
as the last filter. For example, to accept only metrics starting withapp.
[ { "type": "statsd", "settings": { "protocol": "TCP" }, "filters": [ { "type" : "nameStartsWith", "result" : "accept", "values" : ["app."] }, { "type" : "nameStartsWith", "result" : "deny", "values" : [""] } ] } ]
Application Tags
Common tags which should be reported by every metric can be specified using the Metrics.ApplicationTags
setting. This setting should be in JSON format.
- Custom Runtime Setting – Name:
Metrics.ApplicationTags
{
"hostId": "7a3c0356-4dab-42ab-a139-87efb497f3e9",
"environment": "prod"
}
Microflow Activities
You can use activities to provide custom metrics from your app. See Metrics Activities for information about these activities .
Java API
Micrometer metrics can be accessed through Mendix Runtime Java APIs as well inside Mendix. This can be achieved by using the custom runtime setting com.mendix.metrics.Type
. This setting defaults to micrometer
.
- Custom Runtime Setting – Name:
com.mendix.metrics.Type
- Value:
micrometer
Current Metrics and Usage
Currently supported metrics are counter
, gauge
, and timer
. We support Bloch’s builder pattern to create the metrics and support Tag
and Description
which can be added to each metric.
To create the metrics,
import com.mendix.metrics.Counter;
import com.mendix.core.Core;
…
// Create a counter
Counter counter1 = Core.metrics().createCounter("app.strikes")
.withTag("app", "myapp")
.withDescription("Metrics to count the strike outs")
.build();
Counter counter2 = Core.metrics().createCounter("app.count").build();
In the same way, we can also create a gauge
and a timer
using the createGauge
and createTimer
methods respectively.
The name must adhere to the following rules:
- The name can only contain alpha-numeric characters, dots or underscores.
- The name must start with a letter.
- The name cannot start with
mx
, because this prefix is reserved for Mendix internal metrics. - The name is case-insensitive.
⚠ Deprecated Usages
The following deprecated usages will be removed in the future releases,
- The
Core.metrics()
Mendix Runtime Java API methodscounter()
,timer()
,gauges()
, andsets()
, and the correspondingCounters
,Timers
,Gauges
andSets
interfaces are deprecated.
Logging
Metering-related log messages are sent to the Metering
log node. If a registry is enabled, they will be reported with severity debug
.
List of Metrics
The Runtime Server produces various metrics. Some of these metrics are controlled by Mendix: these are prefixed with mx
.
Other metrics are produced by Micrometer, the library that is used for metrics. This library outputs metrics for other libraries that it recognizes, such as the Jetty server that is embedded in the Runtime Server. These additional Micrometer metrics are not under our control and might change.
Runtime Server Metrics
The Runtime Server produces the following metrics out-of-the-box:
Name | Type | Tags | Description |
---|---|---|---|
mx. |
counter | XASId , name |
The total number of requests on a request handler (name ) that were received by a node (XASId ) since it was started. |
mx. |
gauge | The current number of active, named users in the database. | |
mx. |
gauge | The current number of sessions in the database for named users. | |
mx. |
gauge | The current number of sessions in the database for anonymous users. | |
mx. |
counter | XASId |
The total number of transactions on the database that were created by a node (XASId ) since it was started. |
mx. |
counter | XASId |
The total number of SELECT statements that were executed on the database by a node (XASId ) since it was started. |
mx. |
counter | XASId |
The total number of INSERT statements that were executed on the database by a node (XASId ) since it was started. |
mx. |
counter | XASId |
The total number of UPDATE statements that were executed on the database by a node (XASId ) since it was started. |
mx. |
counter | XASId |
The total number of DELETE statements that were executed on the database by a node (XASId ) since it was started. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were created using the Send External Object activity. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were updated using the Send External Object activity. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were created using the Delete External Object activity. |
mx. |
counter | entity |
The total number of objects that were served for a particular type of object (entity ) by a published OData/GraphQL service. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were created due to client requests to a published OData service. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were updated due to client requests to a published OData service. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were deleted due to client requests to a published OData service. |
mx. |
counter | entity |
The total number of objects of a certain entity type (entity ) that were retrieved from an OData service. |
_bytes
).
Additional Metrics
System and Process Metrics
Name | Type | Description |
---|---|---|
system.cpu.count |
gauge | The number of processors available to the Java Virtual Machine. |
system.cpu.usage |
gauge | The recent CPU usage of the system the Runtime Server is running on (in the range [0…1]). |
system.load.average.1m |
gauge | The average number of runnable (queued and running) threads on the available processors. |
process.cpu.usage |
gauge | The recent CPU usage for the Java Virtual Machine process (in the range [0…1]). |
Java Virtual Machine Metrics
Name | Type | Tags | Description |
---|---|---|---|
jvm_info |
gauge | vendor , runtime , version |
Details about the Java Virtual Machine (represented in the tags, the value is always 1). |
jvm.classes.loaded |
gauge | The number of classes that are currently loaded in the Java Virtual Machine. | |
jvm.classes.unloaded |
counter | The total number of classes unloaded since the Java Virtual Machine started execution. | |
jvm.buffer.count |
gauge | id |
An estimate of the number of buffers in a Java Virtual Machine memory pool (id ). |
jvm.buffer.memory_used |
gauge | id |
An estimate of the memory (in bytes) that the Java Virtual Machine is using for a buffer pool (id ). |
jvm.buffer.total_capacity |
gauge | id |
An estimate of the total capacity (in bytes) of the buffers in a buffer pool (id ). |
jvm.memory.used |
gauge | id , area |
The amount of memory (in bytes) used for a certain area (area ): heap or nonheap . |
jvm.memory.committed |
gauge | id , area |
The amount of memory (in bytes) that is committed for the Java Virtual Machine to use. |
jvm.memory.max |
gauge | id , area |
The maximum amount of memory (in bytes) that can be used for memory management. |
jvm.memory.usage.after.gc |
gauge | area , pool |
The percentage of long-lived heap pool used after the last Garbage Collection (GC) event (in the range [0…1]). |
jvm.threads.live_threads |
gauge | The current number of live threads (both daemon and non-daemon threads). | |
jvm.threads.daemon_threads |
gauge | The current number of live daemon threads. | |
jvm.threads.peak_threads |
gauge | The peak live thread count since the Java Virtual Machine started or peak was reset. | |
jvm.threads.states_threads |
gauge | state |
The current number of threads having in a certain state (state ). |
jvm.gc.live.data.size |
gauge | The current size (in bytes) of the long-lived heap memory pool. | |
jvm.gc.max.data.size |
gauge | The maximum size (in bytes) of the long-lived heap memory pool. | |
jvm.gc.memory.allocated |
counter | Incremented for an increase in the size of the (young) heap memory pool after one GC to before the next. | |
jvm.gc.memory.promoted |
counter | Count of positive increases in the size of the old generation memory pool before GC or to after GC. | |
jvm.gc.pause |
timer | action , cause |
The time spent in GC pause. |
jvm.gc.concurrent.phase.time |
timer | action , cause |
The time spent in the concurrent GC phase. |
jvm.gc.overhead |
gauge | An approximation of the percent of CPU time used by GC activities over the last lookback period or since monitoring began, whichever is shorter (in the range [0…1]). |
Refer to the Java Virtual Machine documentation for more details.
Jetty HTTP Server Metrics
Name | Type | Description |
---|---|---|
jetty.connections.messages.in |
counter | The number of messages received by tracked connections. |
jetty.connections.messages.out |
counter | The number of messages sent by tracked connections. |
jetty.connections.bytes.in |
gauge | The number of bytes received by tracked connections. |
jetty.connections.bytes.out |
gauge | The number of bytes sent by tracked connections. |
jetty.connections.current |
gauge | The current number of open Jetty connections. |
jetty.connections.max |
gauge | The maximum number of observed connections over a rolling 2-minute interval. |
jetty.threads.current |
gauge | The total number of threads in the Jetty pool. |
jetty.threads.idle |
gauge | The number of idle threads in the Jetty pool. |
jetty.threads.busy |
gauge | The number of busy threads in the Jetty pool. |
jetty.threads.jobs |
gauge | The number of jobs queued waiting for a thread. |
jetty.threads.config.min |
gauge | The minimum number of threads in the Jetty pool. |
jetty.threads.config.max |
gauge | The maximum number of threads in the Jetty pool. |
Database Connection Pool Metrics
Name | Type | Description |
---|---|---|
commons.pool2.num.idle |
gauge | The number of connections currently idle in the pool. |
commons.pool2.num.active |
gauge | The number of connections currently active in the pool. |
commons.pool2.num.waiters |
gauge | An estimate of the number of threads currently blocked waiting for a connection from the pool. |
commons.pool2.mean.active |
gauge | The mean time for which connections are active. |
commons.pool2.mean.idle |
gauge | The mean time for which connections are idle. |
commons.pool2.mean.borrow.wait |
gauge | The mean time threads wait to get a connection. |
commons.pool2.max.borrow.wait |
gauge | The maximum time a thread has waited to get a connection from the pool. |
commons.pool2.created |
counter | The total number of connections created for the pool over its lifetime. |
commons.pool2.borrowed |
counter | The total number of connections borrowed from the pool over its lifetime. |
commons.pool2.returned |
counter | The total number of connections returned to the pool over its lifetime. |
commons.pool2.destroyed |
counter | The total number of connections destroyed by the pool over its lifetime. |
commons.pool2.destroyed.by.evictor |
counter | The total number of connections destroyed by the evictor associated with the pool over its lifetime. |
commons.pool2.destroyed.by.borrow.validation |
counter | The total number of connections destroyed by the pool over its lifetime as a result of failing validation during borrowing. |