PGAdapter supports OpenTelemetry tracing and metrics. You can enable the tracing or the metrics
by adding the -enable_otel
or -enable_otel_metrics
command line argument when starting
PGAdapter. Tracing and metrics can be enabled at the same time.
Optionally, you can also set a trace sample ratio to limit the number of traces that will be
collected and set with the -otel_trace_ratio=<ratio>
command line argument. If you omit this
argument, PGAdapter will use a 0.05
trace ratio. The ratio must be in the range [0.0, 1.0]
.
Example:
docker run \
-d -p 5432:5432 \
-v /path/to/credentials.json:/credentials.json:ro \
gcr.io/cloud-spanner-pg-adapter/pgadapter \
-p my-project -i my-instance \
-c /credentials.json -x \
-enable_otel -otel_trace_ratio=0.1 \
-enable_otel_metrics
PGAdapter will by default export traces to Google Cloud Trace. PGAdapter uses the OpenTelemetry SDK Autoconfigure module for configuring OpenTelemetry. You can use the options of this module to configure another exporter.
PGAdapter uses a different set of default values for the following OpenTelemetry Autoconfigure properties:
otel.traces.exporter
:none
otel.metrics.exporter
:none
otel.logs.exporter
:none
In addition, PGAdapter always adds an exporter for Google Cloud Trace.
The traces are by default exported to Google Cloud Trace. The traces follow the structure of the PostgreSQL wire-protocol. Traces typically consist of the following spans:
query_protocol_handler
: The PostgreSQL wire-protocol consists of multiple messages to parse, describe, bind, and execute SQL statements. Clients and drivers will send multiple of these to PGAdapter, and PGAdapter will translate these into a set of RPC invocations on CLoud Spanner. PGAdapter minimizes the number of RPC invocations that is needed to respond to these messages. One span ofquery_protocol_handler
consists of all the round-trips that PGAdapter needed to respond to the messages that it received during one query session.execute
/analyze
/execute_batch
: PGAdapter will translate the incoming messages intoexecute
,analyze
, andexecute_batch
actions on Cloud Spanner. One span ofquery_protocol_handler
normally consists of one or more of these actions.execute_on_spanner
: This span contains the time when PGAdapter hands off the execution of a statement to the Cloud Spanner Java client, and is roughly identical to the end-to-end execution of the statement.execute_batch
: This span indicates a batch of DML or DDL statements that are being received and buffered by PGAdapter. The statements will be sent as one batch to Cloud Spanner for execution. The actual execution of the batch is recorded in theexecute_batch_on_spanner
span.send_result_set
: This span is only recorded for queries that send rows back to the client. It shows the time that it took for PGAdapter to receive all rows from Cloud Spanner and send these to the client. This span will be longer for queries that return a large number of rows. The time it takes to send the results to the client also depends on how quickly the client application can consume the rows that are being sent, and on the network speed between PGAdapter and Cloud Spanner.
The following sections show samples of commonly executed statements. All spans that are exported by Cloud Spanner contain the following attributes:
db.statement
: The SQL statement that is being executed.pgadapter.connection_id
: Each connection in PGAdapter is assigned a random ID. You can use this ID to filter spans for a specific connection.pgadapter.transaction_id
: Each transaction in PGAdapter is assigned a random ID. YOu can use this ID to filter spans for a specific transaction. This field is not present for statements that are executed outside an (explicit) transaction.
A typical execution of a query consists of the following spans:
query_protocol_handler
: The front-end query protocol handler of PGAdapter that receives messages from the client and returns results to the client. The query handler normally receives five messages from the client when executing a query:P
(Parse): Parse the query string.B
(Bind): Bind the query parameter values.D
(Describe): Describe the query result (which columns will be returned by the query).E
(Execute): Execute the statement and return the results.S
(Sync): Flush all buffered messages and return all results.
execute
: The front-end query protocol handler requests the backend connection to Cloud Spanner to execute the query that it constructed from the messages it received.execute_on_spanner
: This the actual RPC invocation on Cloud Spanner.send_result_set
: Shows the time it takes to receive all rows from Cloud Spanner and the time it takes to send these to the client application.
The query_protocol_handler
span also includes the time it takes to send the results to the client.
This time will also depend on the time that the client requires to actually receive and process the
results. You can see this time in the example above as the time that the query_protocol_handler
span is active after the statement has been executed.
A typical execution of a DML statement consists of the following spans:
query_protocol_handler
: The front-end query protocol handler of PGAdapter that receives messages from the client and returns results to the client. The query handler normally receives five messages from the client when executing a DML statement:P
(Parse): Parse the DML string.B
(Bind): Bind the DML parameter values.D
(Describe): Describe the result of the statement. For DML statement, this is an empty result.E
(Execute): Execute the statement and return the update count.S
(Sync): Flush all buffered messages and return all results.
execute
: The front-end query protocol handler requests the backend connection to Cloud Spanner to execute the DML statement that it constructed from the messages it received.execute_on_spanner
: This the actual RPC invocation on Cloud Spanner.
The query_protocol_handler
span also includes the time it takes to send the update count to the
client. This time is negligible compared to sending a large result set from PGAdapter to the client.
The PostgreSQL wire-protocol also supports batching multiple statements into one execution. This is
done by sending multiple E
(Execute) messages to the server (PGAdapter) without sending an S
(Sync) message. This will allow the server to buffer the statements and wait with sending the
results of these until it sees an S
(Sync) message. A typical execution of a batch of DML
statements consists of the following spans:
query_protocol_handler
: The front-end query protocol handler of PGAdapter that receives messages from the client and returns results to the client. The query handler normally receives four messages for each DML statement in a batch from the client:P
(Parse): Parse the DML string.B
(Bind): Bind the DML parameter values.D
(Describe): Describe the result of the statement. For DML statement, this is an empty result.E
(Execute): Execute the statement and return the update count.
- It then receives an
S
(Sync) message when the client has sent all the DML statements. execute_batch
: The front-end query protocol handler requests the backend connection to create a DML batch, buffer the DML statements on the client, and then execute the statements on Cloud Spanner. The total time of this span is the time it took to buffer all the statements on the client and to execute the batch on Cloud Spanner.buffer
: For each DML statement, abuffer
span is created within the DML batch. These statements are not yet executed on spanner, but buffered in the Spanner client until all the DML statements have been collected.execute_batch_on_spanner
: Once all DML statements have been collected, PGAdapter creates one batch and sends this as one request to Cloud Spanner. This span shows the time it takes to execute this batch request on Cloud Spanner.
Request tags (statement tags) and transaction tags are added to the traces that are generated by the Spanner client. You can set statement tags and transaction tags in PGAdapter like this:
begin;
-- This transaction tag will be applied to all statements in this transaction.
set spanner.transaction_tag = 'my_transaction_tag';
-- Statement tags can be included as a comment in the SQL string.
/*@statement_tag='my_statement_tag1'*/ insert into my_table (id, value) values (1, 'One');
/*@statement_tag='my_statement_tag2'*/ insert into my_table (id, value) values (2, 'Two');
-- Statement tags can also be set with a SET statement. This tag is only applied to the next
-- statement that is executed and automatically cleared after its execution.
set spanner.statement_tag = 'my_statement_tag3';
insert into my_table (id, value) values (3, 'Three');
commit;
The available metrics in PGAdapter are:
spanner/pgadapter/roundtrip_latencies
: Latency between PGAdapter receiving a statement from the client and PGAdapter returning the last row of the response to the client.spanner/pgadapter/client_lib_latencies
: Latency when the Spanner's client library receives a call and returns a response.
In addition, client library's metrics are avaialble for use when the
-enable_otel_metrics
command line argument is provided.
How can I find all the statements that were executed on the same connection as the trace I'm looking at?
All traces include an attribute pgadapter.connection_id
. This connection ID is randomly chosen,
but remains the same for a connection during its lifetime. You can use this ID to look up all other
traces for the same connection by adding a filter pgadapter.connection_id:<id>
.
All traces for statements that used an explicit transaction include an attribute pgadapter.transaction_id
.
This transaction ID is randomly chosen, but remains the same for a transaction during its lifetime.
You can use this ID to look up all other traces for the same transaction by adding a filter pgadapter.transaction_id:<id>
.
Cloud Spanner can abort any read/write transaction. A transaction can be aborted for various reasons:
- Lock conflicts
- Database schema changes
- Too long inactivity (a read/write transaction is aborted after 10 seconds of inactivity)
But it can also be aborted for internal technical reasons in Cloud Spanner. PGAdapter will attempt
to retry transactions that are aborted by Cloud Spanner. This can cause some statements to seem to
execute much slower than they should, as they have to wait for the internal retry attempt to finish
first. You can find statements that waited for a transaction retry by adding the filter
HasLabel:pgadapter.retry_attempt
.
Note from the above screenshot:
- The retry internal retry is initiated and executed as part of the statement execution. This makes it seem like the statement itself takes long to execute.
- Cloud Spanner also sends the client a back-off value that it should wait before retrying the transaction. This is the reason that the total execution time of the statements seems to be much longer than the transaction retry, as the Java client library will wait for this back-off time before actually starting the retry.