diff --git a/google-cloud-bigquerystorage/src/main/java/com/google/cloud/bigquery/storage/v1/RequestProfiler.java b/google-cloud-bigquerystorage/src/main/java/com/google/cloud/bigquery/storage/v1/RequestProfiler.java index 9a0ff29b6a..dd055ff6fd 100644 --- a/google-cloud-bigquerystorage/src/main/java/com/google/cloud/bigquery/storage/v1/RequestProfiler.java +++ b/google-cloud-bigquerystorage/src/main/java/com/google/cloud/bigquery/storage/v1/RequestProfiler.java @@ -1,3 +1,18 @@ +/* + * Copyright 2024 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ package com.google.cloud.bigquery.storage.v1; import java.time.Duration; @@ -19,16 +34,20 @@ * A profiler that would periodically generate a report for the past period with the latency report * for the slowest requests. This is used for debugging only. * - *
The report will contain the execution details of the TOP_K slowest requests, one example: ``` - * INFO: At system time 1720566109971, in total 2 finished during the last 60000 milliseconds, the - * top 10 long latency requests details report: ----------------------------- Request uuid: - * request_1 with total time 1000 milliseconds Operation name json_to_proto_conversion starts at: - * 1720566109971, ends at: 1720566109971, total time: 200 milliseconds Operation name - * backend_latency starts at: 1720566109971, ends at: 1720566109971, total time: 800 milliseconds - * ----------------------------- Request uuid: request_2 with total time 500 milliseconds Operation - * name json_to_proto_conversion starts at: 1720566109971, ends at: 1720566109971, total time: 250 - * milliseconds Operation name backend_latency starts at: 1720566109971, ends at: 1720566109971, - * total time: 250 milliseconds ``` + *
+ * The report will contain the execution details of the TOP_K slowest requests, one example: + * + * INFO: At system time 1720566109971, in total 2 finished during the last 60000 milliseconds, the top 10 long latency requests details report: + * ----------------------------- + * Request uuid: request_1 with total time 1000 milliseconds + * Operation name json_to_proto_conversion starts at: 1720566109971, ends at: 1720566109971, total time: 200 milliseconds + * Operation name backend_latency starts at: 1720566109971, ends at: 1720566109971, total time: 800 milliseconds + * ----------------------------- + * Request uuid: request_2 with total time 500 milliseconds + * Operation name json_to_proto_conversion starts at: 1720566109971, ends at: 1720566109971, total time: 250 milliseconds + * Operation name backend_latency starts at: 1720566109971, ends at: 1720566109971, total time: 250 milliseconds + * ... + **/ public class RequestProfiler { enum OperationName { @@ -37,7 +56,7 @@ enum OperationName { // Json to proto conversion time. JSON_TO_PROTO_CONVERSION("json_to_proto_conversion"), // Time spent to fetch the table schema when user didn't provide it. - SCHEMA_FECTCHING("schema_fetching"), + SCHEMA_FETCHING("schema_fetching"), // Time spent within wait queue before it get picked up. WAIT_QUEUE("wait_queue"), // Time spent within backend to process the request. @@ -51,6 +70,9 @@ enum OperationName { private static final Logger log = Logger.getLogger(RequestProfiler.class.getName()); + // Discard the requests if we are caching too many requests. + private static final int MAX_CACHED_REQUEST = 100000; + // Singleton for easier access. public static final RequestProfiler REQUEST_PROFILER_SINGLETON = new RequestProfiler(); @@ -66,15 +88,39 @@ enum OperationName { private Thread flushThread; + // Count the total number of dropped operations. + long droppedOperationCount = 0; + // Mark an operation for a given request id to be start. void startOperation(OperationName operationName, String requestUniqueId) { - idToIndividualOperation.putIfAbsent( - requestUniqueId, new IndividualRequestProfiler(requestUniqueId)); + if (!idToIndividualOperation.containsKey(requestUniqueId)) { + if (idToIndividualOperation.size() > MAX_CACHED_REQUEST) { + log.warning( + String.format( + "startOperation is triggered for request_id: %s that's hasn't " + + "seen before, this is possible when " + + "we are recording too much ongoing requests. So far we has dropped %s operations.", + requestUniqueId, droppedOperationCount)); + droppedOperationCount++; + return; + } + idToIndividualOperation.put(requestUniqueId, new IndividualRequestProfiler(requestUniqueId)); + } idToIndividualOperation.get(requestUniqueId).startOperation(operationName); } // Mark an operation for a given request id to be end. void endOperation(OperationName operationName, String requestUniqueId) { + if (!idToIndividualOperation.containsKey(requestUniqueId)) { + log.warning( + String.format( + "endOperation is triggered for request_id: %s that's hasn't " + + "seen before, this is possible when " + + "we are recording too much ongoing requests. So far we has dropped %s operations.", + requestUniqueId, droppedOperationCount)); + droppedOperationCount++; + return; + } idToIndividualOperation.get(requestUniqueId).endOperation(operationName); } @@ -82,6 +128,7 @@ void flushReport() { log.info(flushAndGenerateReportText()); } + // Periodically trigger the report generation. void startPeriodicalReportFlushing() { this.flushThread = new Thread( @@ -111,6 +158,9 @@ String flushAndGenerateReportText() { Iterator