From b550bc87376ff020b6605f536923f80604f74ced Mon Sep 17 00:00:00 2001 From: Nikita Tkachenko <121111529+nikita-tkachenko-datadog@users.noreply.github.com> Date: Fri, 16 Feb 2024 11:05:36 +0100 Subject: [PATCH] Implement telemetry for CI Visibility APIs (#6667) --- .../agent-ci-visibility/build.gradle | 1 + .../CiVisibilityRepoServices.java | 17 ++- .../communication/BackendApi.java | 8 +- .../communication/EvpProxyApi.java | 10 +- .../civisibility/communication/IntakeApi.java | 10 +- .../communication/TelemetryListener.java | 132 ++++++++++++++++++ .../config/CiVisibilitySettings.java | 9 +- .../config/ConfigurationApiImpl.java | 76 ++++++++-- .../civisibility/git/tree/GitDataApi.java | 31 +++- .../git/tree/GitDataUploaderImpl.java | 36 +++-- .../config/ConfigurationApiImplTest.groovy | 7 +- .../git/tree/GitDataApiTest.groovy | 7 +- .../git/tree/GitDataUploaderImplTest.groovy | 2 +- .../CiVisibilityDistributionMetric.java | 2 +- 14 files changed, 308 insertions(+), 40 deletions(-) create mode 100644 dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/TelemetryListener.java diff --git a/dd-java-agent/agent-ci-visibility/build.gradle b/dd-java-agent/agent-ci-visibility/build.gradle index 1f3c4866ff3..36da0b46a90 100644 --- a/dd-java-agent/agent-ci-visibility/build.gradle +++ b/dd-java-agent/agent-ci-visibility/build.gradle @@ -38,6 +38,7 @@ excludedClassesCoverage += [ "datadog.trace.civisibility.config.CachingModuleExecutionSettingsFactory", "datadog.trace.civisibility.config.CachingModuleExecutionSettingsFactory.Key", "datadog.trace.civisibility.config.CiVisibilitySettings", + "datadog.trace.civisibility.config.ConfigurationApiImpl", "datadog.trace.civisibility.config.ConfigurationApiImpl.MultiEnvelopeDto", "datadog.trace.civisibility.config.ConfigurationApi.1", "datadog.trace.civisibility.config.ModuleExecutionSettingsFactoryImpl", diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/CiVisibilityRepoServices.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/CiVisibilityRepoServices.java index 4daa6e638cd..f767726c8c4 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/CiVisibilityRepoServices.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/CiVisibilityRepoServices.java @@ -2,6 +2,7 @@ import datadog.trace.api.Config; import datadog.trace.api.civisibility.config.ModuleExecutionSettings; +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector; import datadog.trace.api.git.GitInfoProvider; import datadog.trace.civisibility.ci.CIInfo; import datadog.trace.civisibility.ci.CIProviderInfo; @@ -61,6 +62,7 @@ public class CiVisibilityRepoServices { gitDataUploader = buildGitDataUploader( services.config, + services.metricCollector, services.gitInfoProvider, services.gitClientFactory, services.backendApi, @@ -75,7 +77,12 @@ public class CiVisibilityRepoServices { } else { moduleExecutionSettingsFactory = buildModuleExecutionSettingsFactory( - services.config, services.backendApi, gitDataUploader, repoIndexProvider, repoRoot); + services.config, + services.metricCollector, + services.backendApi, + gitDataUploader, + repoIndexProvider, + repoRoot); } } @@ -112,6 +119,7 @@ private static ModuleExecutionSettingsFactory buildModuleExecutionSettingsFetche private static ModuleExecutionSettingsFactory buildModuleExecutionSettingsFactory( Config config, + CiVisibilityMetricCollector metricCollector, BackendApi backendApi, GitDataUploader gitDataUploader, RepoIndexProvider repoIndexProvider, @@ -122,7 +130,7 @@ private static ModuleExecutionSettingsFactory buildModuleExecutionSettingsFactor "Remote config and skippable tests requests will be skipped since backend API client could not be created"); configurationApi = ConfigurationApi.NO_OP; } else { - configurationApi = new ConfigurationApiImpl(backendApi); + configurationApi = new ConfigurationApiImpl(backendApi, metricCollector); } return new CachingModuleExecutionSettingsFactory( config, @@ -132,6 +140,7 @@ private static ModuleExecutionSettingsFactory buildModuleExecutionSettingsFactor private static GitDataUploader buildGitDataUploader( Config config, + CiVisibilityMetricCollector metricCollector, GitInfoProvider gitInfoProvider, GitClient.Factory gitClientFactory, BackendApi backendApi, @@ -153,10 +162,10 @@ private static GitDataUploader buildGitDataUploader( } String remoteName = config.getCiVisibilityGitRemoteName(); - GitDataApi gitDataApi = new GitDataApi(backendApi); + GitDataApi gitDataApi = new GitDataApi(backendApi, metricCollector); GitClient gitClient = gitClientFactory.create(repoRoot); return new GitDataUploaderImpl( - config, gitDataApi, gitClient, gitInfoProvider, repoRoot, remoteName); + config, metricCollector, gitDataApi, gitClient, gitInfoProvider, repoRoot, remoteName); } private static SourcePathResolver buildSourcePathResolver( diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/BackendApi.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/BackendApi.java index 02f4712950c..52aa8f70641 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/BackendApi.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/BackendApi.java @@ -1,13 +1,19 @@ package datadog.trace.civisibility.communication; +import datadog.communication.http.OkHttpUtils; import datadog.trace.civisibility.utils.IOThrowingFunction; import java.io.IOException; import java.io.InputStream; +import javax.annotation.Nullable; import okhttp3.RequestBody; /** API for posting HTTP requests to backend */ public interface BackendApi { - T post(String uri, RequestBody requestBody, IOThrowingFunction responseParser) + T post( + String uri, + RequestBody requestBody, + IOThrowingFunction responseParser, + @Nullable OkHttpUtils.CustomListener requestListener) throws IOException; } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/EvpProxyApi.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/EvpProxyApi.java index 50b8a3bf811..eea481345d6 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/EvpProxyApi.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/EvpProxyApi.java @@ -6,6 +6,7 @@ import java.io.IOException; import java.io.InputStream; import java.util.zip.GZIPInputStream; +import javax.annotation.Nullable; import okhttp3.HttpUrl; import okhttp3.OkHttpClient; import okhttp3.Request; @@ -56,7 +57,10 @@ public EvpProxyApi( @Override public T post( - String uri, RequestBody requestBody, IOThrowingFunction responseParser) + String uri, + RequestBody requestBody, + IOThrowingFunction responseParser, + @Nullable OkHttpUtils.CustomListener requestListener) throws IOException { final HttpUrl url = evpProxyUrl.resolve(uri); @@ -67,6 +71,10 @@ public T post( .addHeader(X_DATADOG_TRACE_ID_HEADER, traceId) .addHeader(X_DATADOG_PARENT_ID_HEADER, traceId); + if (requestListener != null) { + requestBuilder.tag(OkHttpUtils.CustomListener.class, requestListener); + } + if (gzipEnabled) { requestBuilder.addHeader(ACCEPT_ENCODING_HEADER, GZIP_ENCODING); } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/IntakeApi.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/IntakeApi.java index bb0769990f1..a92f1bc5ddb 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/IntakeApi.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/IntakeApi.java @@ -6,6 +6,7 @@ import java.io.IOException; import java.io.InputStream; import java.util.zip.GZIPInputStream; +import javax.annotation.Nullable; import okhttp3.HttpUrl; import okhttp3.OkHttpClient; import okhttp3.Request; @@ -61,7 +62,10 @@ public IntakeApi( @Override public T post( - String uri, RequestBody requestBody, IOThrowingFunction responseParser) + String uri, + RequestBody requestBody, + IOThrowingFunction responseParser, + @Nullable OkHttpUtils.CustomListener requestListener) throws IOException { HttpUrl url = hostUrl.resolve(uri); Request.Builder requestBuilder = @@ -72,6 +76,10 @@ public T post( .addHeader(X_DATADOG_TRACE_ID_HEADER, traceId) .addHeader(X_DATADOG_PARENT_ID_HEADER, traceId); + if (requestListener != null) { + requestBuilder.tag(OkHttpUtils.CustomListener.class, requestListener); + } + if (gzipEnabled) { requestBuilder.addHeader(ACCEPT_ENCODING_HEADER, GZIP_ENCODING); } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/TelemetryListener.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/TelemetryListener.java new file mode 100644 index 00000000000..4fb25df98a3 --- /dev/null +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/communication/TelemetryListener.java @@ -0,0 +1,132 @@ +package datadog.trace.civisibility.communication; + +import datadog.communication.http.OkHttpUtils; +import datadog.trace.api.civisibility.telemetry.CiVisibilityCountMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityDistributionMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector; +import datadog.trace.api.civisibility.telemetry.tag.ErrorType; +import java.io.IOException; +import javax.annotation.Nullable; +import okhttp3.Call; +import okhttp3.Response; + +public class TelemetryListener extends OkHttpUtils.CustomListener { + + private final CiVisibilityMetricCollector metricCollector; + private final @Nullable CiVisibilityCountMetric requestCountMetric; + private final @Nullable CiVisibilityCountMetric requestErrorsMetric; + private final @Nullable CiVisibilityDistributionMetric requestBytesMetric; + private final @Nullable CiVisibilityDistributionMetric requestDurationMetric; + private final @Nullable CiVisibilityDistributionMetric responseBytesMetric; + private long callStartTimestamp; + + private TelemetryListener( + CiVisibilityMetricCollector metricCollector, + @Nullable CiVisibilityCountMetric requestCountMetric, + @Nullable CiVisibilityCountMetric requestErrorsMetric, + @Nullable CiVisibilityDistributionMetric requestBytesMetric, + @Nullable CiVisibilityDistributionMetric requestDurationMetric, + @Nullable CiVisibilityDistributionMetric responseBytesMetric) { + this.metricCollector = metricCollector; + this.requestCountMetric = requestCountMetric; + this.requestErrorsMetric = requestErrorsMetric; + this.requestBytesMetric = requestBytesMetric; + this.requestDurationMetric = requestDurationMetric; + this.responseBytesMetric = responseBytesMetric; + } + + public void callStart(Call call) { + callStartTimestamp = System.currentTimeMillis(); + if (requestCountMetric != null) { + metricCollector.add(requestCountMetric, 1); + } + } + + public void requestBodyEnd(Call call, long byteCount) { + if (requestBytesMetric != null) { + metricCollector.add(requestBytesMetric, (int) byteCount); + } + } + + public void responseHeadersEnd(Call call, Response response) { + if (requestErrorsMetric != null) { + if (!response.isSuccessful()) { + int responseCode = response.code(); + metricCollector.add(requestErrorsMetric, 1, ErrorType.from(responseCode)); + } + } + } + + @Override + public void responseBodyEnd(Call call, long byteCount) { + if (responseBytesMetric != null) { + metricCollector.add(responseBytesMetric, (int) byteCount); + } + } + + public void callEnd(Call call) { + if (requestDurationMetric != null) { + int durationMillis = (int) (System.currentTimeMillis() - callStartTimestamp); + metricCollector.add(requestDurationMetric, durationMillis); + } + } + + public void callFailed(Call call, IOException ioe) { + if (requestDurationMetric != null) { + int durationMillis = (int) (System.currentTimeMillis() - callStartTimestamp); + metricCollector.add(requestDurationMetric, durationMillis); + } + + if (requestErrorsMetric != null) { + metricCollector.add(requestErrorsMetric, 1, ErrorType.NETWORK); + } + } + + public static final class Builder { + private final CiVisibilityMetricCollector metricCollector; + private @Nullable CiVisibilityCountMetric requestCountMetric; + private @Nullable CiVisibilityCountMetric requestErrorsMetric; + private @Nullable CiVisibilityDistributionMetric requestBytesMetric; + private @Nullable CiVisibilityDistributionMetric requestDurationMetric; + private @Nullable CiVisibilityDistributionMetric responseBytesMetric; + + public Builder(CiVisibilityMetricCollector metricCollector) { + this.metricCollector = metricCollector; + } + + public Builder requestCount(@Nullable CiVisibilityCountMetric requestCountMetric) { + this.requestCountMetric = requestCountMetric; + return this; + } + + public Builder requestErrors(@Nullable CiVisibilityCountMetric requestErrorsMetric) { + this.requestErrorsMetric = requestErrorsMetric; + return this; + } + + public Builder requestBytes(@Nullable CiVisibilityDistributionMetric requestBytesMetric) { + this.requestBytesMetric = requestBytesMetric; + return this; + } + + public Builder requestDuration(@Nullable CiVisibilityDistributionMetric requestDurationMetric) { + this.requestDurationMetric = requestDurationMetric; + return this; + } + + public Builder responseBytes(@Nullable CiVisibilityDistributionMetric responseBytesMetric) { + this.responseBytesMetric = responseBytesMetric; + return this; + } + + public OkHttpUtils.CustomListener build() { + return new TelemetryListener( + metricCollector, + requestCountMetric, + requestErrorsMetric, + requestBytesMetric, + requestDurationMetric, + responseBytesMetric); + } + } +} diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/CiVisibilitySettings.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/CiVisibilitySettings.java index 523cab55226..b98ec9e9103 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/CiVisibilitySettings.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/CiVisibilitySettings.java @@ -5,24 +5,31 @@ public class CiVisibilitySettings { public static final CiVisibilitySettings DEFAULT = - new CiVisibilitySettings(false, false, false, false); + new CiVisibilitySettings(false, false, false, false, false); + private final boolean itr_enabled; private final boolean code_coverage; private final boolean tests_skipping; private final boolean require_git; private final boolean flaky_test_retries_enabled; public CiVisibilitySettings( + boolean itr_enabled, boolean code_coverage, boolean tests_skipping, boolean require_git, boolean flaky_test_retries_enabled) { + this.itr_enabled = itr_enabled; this.code_coverage = code_coverage; this.tests_skipping = tests_skipping; this.require_git = require_git; this.flaky_test_retries_enabled = flaky_test_retries_enabled; } + public boolean isItrEnabled() { + return itr_enabled; + } + public boolean isCodeCoverageEnabled() { return code_coverage; } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/ConfigurationApiImpl.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/ConfigurationApiImpl.java index 5870d67be66..6e71c33135f 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/ConfigurationApiImpl.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/config/ConfigurationApiImpl.java @@ -3,8 +3,17 @@ import com.squareup.moshi.JsonAdapter; import com.squareup.moshi.Moshi; import com.squareup.moshi.Types; +import datadog.communication.http.OkHttpUtils; import datadog.trace.api.civisibility.config.TestIdentifier; +import datadog.trace.api.civisibility.telemetry.CiVisibilityCountMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityDistributionMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector; +import datadog.trace.api.civisibility.telemetry.tag.CoverageEnabled; +import datadog.trace.api.civisibility.telemetry.tag.ItrEnabled; +import datadog.trace.api.civisibility.telemetry.tag.ItrSkipEnabled; +import datadog.trace.api.civisibility.telemetry.tag.RequireGit; import datadog.trace.civisibility.communication.BackendApi; +import datadog.trace.civisibility.communication.TelemetryListener; import java.io.IOException; import java.lang.reflect.ParameterizedType; import java.util.Collection; @@ -24,18 +33,23 @@ public class ConfigurationApiImpl implements ConfigurationApi { private static final String FLAKY_TESTS_URI = "ci/libraries/tests/flaky"; private final BackendApi backendApi; + private final CiVisibilityMetricCollector metricCollector; private final Supplier uuidGenerator; private final JsonAdapter> requestAdapter; private final JsonAdapter> settingsResponseAdapter; private final JsonAdapter> testIdentifiersResponseAdapter; - public ConfigurationApiImpl(BackendApi backendApi) { - this(backendApi, () -> UUID.randomUUID().toString()); + public ConfigurationApiImpl(BackendApi backendApi, CiVisibilityMetricCollector metricCollector) { + this(backendApi, metricCollector, () -> UUID.randomUUID().toString()); } - ConfigurationApiImpl(BackendApi backendApi, Supplier uuidGenerator) { + ConfigurationApiImpl( + BackendApi backendApi, + CiVisibilityMetricCollector metricCollector, + Supplier uuidGenerator) { this.backendApi = backendApi; + this.metricCollector = metricCollector; this.uuidGenerator = uuidGenerator; Moshi moshi = @@ -65,26 +79,65 @@ public CiVisibilitySettings getSettings(TracerEnvironment tracerEnvironment) thr new DataDto<>(uuid, "ci_app_test_service_libraries_settings", tracerEnvironment)); String json = requestAdapter.toJson(settingsRequest); RequestBody requestBody = RequestBody.create(JSON, json); - return backendApi.post( - SETTINGS_URI, - requestBody, - is -> settingsResponseAdapter.fromJson(Okio.buffer(Okio.source(is))).data.attributes); + + OkHttpUtils.CustomListener telemetryListener = + new TelemetryListener.Builder(metricCollector) + .requestCount(CiVisibilityCountMetric.GIT_REQUESTS_SETTINGS) + .requestErrors(CiVisibilityCountMetric.GIT_REQUESTS_SETTINGS_ERRORS) + .requestDuration(CiVisibilityDistributionMetric.GIT_REQUESTS_SETTINGS_MS) + .build(); + + CiVisibilitySettings settings = + backendApi.post( + SETTINGS_URI, + requestBody, + is -> settingsResponseAdapter.fromJson(Okio.buffer(Okio.source(is))).data.attributes, + telemetryListener); + + metricCollector.add( + CiVisibilityCountMetric.GIT_REQUESTS_SETTINGS_RESPONSE, + 1, + settings.isItrEnabled() ? ItrEnabled.TRUE : null, + settings.isTestsSkippingEnabled() ? ItrSkipEnabled.TRUE : null, + settings.isCodeCoverageEnabled() ? CoverageEnabled.TRUE : null, + settings.isGitUploadRequired() ? RequireGit.TRUE : null); + + return settings; } @Override public Collection getSkippableTests(TracerEnvironment tracerEnvironment) throws IOException { - return getTestsData(SKIPPABLE_TESTS_URI, "test_params", tracerEnvironment); + OkHttpUtils.CustomListener telemetryListener = + new TelemetryListener.Builder(metricCollector) + .requestCount(CiVisibilityCountMetric.ITR_SKIPPABLE_TESTS_REQUEST) + .requestErrors(CiVisibilityCountMetric.ITR_SKIPPABLE_TESTS_REQUEST_ERRORS) + .requestDuration(CiVisibilityDistributionMetric.ITR_SKIPPABLE_TESTS_REQUEST_MS) + .responseBytes(CiVisibilityDistributionMetric.ITR_SKIPPABLE_TESTS_RESPONSE_BYTES) + .build(); + + Collection skippableTests = + getTestsData(SKIPPABLE_TESTS_URI, "test_params", tracerEnvironment, telemetryListener); + + metricCollector.add( + CiVisibilityCountMetric.ITR_SKIPPABLE_TESTS_RESPONSE_TESTS, skippableTests.size()); + + return skippableTests; } @Override public Collection getFlakyTests(TracerEnvironment tracerEnvironment) throws IOException { - return getTestsData(FLAKY_TESTS_URI, "flaky_test_from_libraries_params", tracerEnvironment); + return getTestsData( + FLAKY_TESTS_URI, "flaky_test_from_libraries_params", tracerEnvironment, null); } private Collection getTestsData( - String endpoint, String dataType, TracerEnvironment tracerEnvironment) throws IOException { + String endpoint, + String dataType, + TracerEnvironment tracerEnvironment, + OkHttpUtils.CustomListener requestListener) + throws IOException { String uuid = uuidGenerator.get(); EnvelopeDto request = new EnvelopeDto<>(new DataDto<>(uuid, dataType, tracerEnvironment)); @@ -94,7 +147,8 @@ private Collection getTestsData( backendApi.post( endpoint, requestBody, - is -> testIdentifiersResponseAdapter.fromJson(Okio.buffer(Okio.source(is))).data); + is -> testIdentifiersResponseAdapter.fromJson(Okio.buffer(Okio.source(is))).data, + requestListener); return response.stream().map(DataDto::getAttributes).collect(Collectors.toList()); } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataApi.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataApi.java index 9b6fad20684..11448c6f3d5 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataApi.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataApi.java @@ -3,7 +3,12 @@ import com.squareup.moshi.Json; import com.squareup.moshi.JsonAdapter; import com.squareup.moshi.Moshi; +import datadog.communication.http.OkHttpUtils; +import datadog.trace.api.civisibility.telemetry.CiVisibilityCountMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityDistributionMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector; import datadog.trace.civisibility.communication.BackendApi; +import datadog.trace.civisibility.communication.TelemetryListener; import datadog.trace.civisibility.utils.IOUtils; import java.io.IOException; import java.nio.file.Files; @@ -29,12 +34,14 @@ public class GitDataApi { private static final String UPLOAD_PACKFILES_URI = "git/repository/packfile"; private final BackendApi backendApi; + private final CiVisibilityMetricCollector metricCollector; private final JsonAdapter searchCommitsRequestAdapter; private final JsonAdapter searchCommitsResponseAdapter; private final JsonAdapter pushedShaAdapter; - public GitDataApi(BackendApi backendApi) { + public GitDataApi(BackendApi backendApi, CiVisibilityMetricCollector metricCollector) { this.backendApi = backendApi; + this.metricCollector = metricCollector; Moshi moshi = new Moshi.Builder().build(); searchCommitsRequestAdapter = moshi.adapter(SearchCommitsRequest.class); @@ -58,11 +65,20 @@ public Collection searchCommits(String gitRemoteUrl, List commit String json = searchCommitsRequestAdapter.toJson(searchCommitsRequest); RequestBody requestBody = RequestBody.create(JSON, json); + + OkHttpUtils.CustomListener telemetryListener = + new TelemetryListener.Builder(metricCollector) + .requestCount(CiVisibilityCountMetric.GIT_REQUESTS_SEARCH_COMMITS) + .requestErrors(CiVisibilityCountMetric.GIT_REQUESTS_SEARCH_COMMITS_ERRORS) + .requestDuration(CiVisibilityDistributionMetric.GIT_REQUESTS_SEARCH_COMMITS_MS) + .build(); + SearchCommitsResponse response = backendApi.post( SEARCH_COMMITS_URI, requestBody, - is -> searchCommitsResponseAdapter.fromJson(Okio.buffer(Okio.source(is)))); + is -> searchCommitsResponseAdapter.fromJson(Okio.buffer(Okio.source(is))), + telemetryListener); return response.data.stream().map(Commit::getId).collect(Collectors.toSet()); } @@ -94,7 +110,16 @@ public void uploadPackFile(String gitRemoteUrl, String currentCommitHash, Path p .addFormDataPart("packfile", packFileNameWithoutRandomPrefix, packFileBody) .build(); - String response = backendApi.post(UPLOAD_PACKFILES_URI, requestBody, IOUtils::readFully); + OkHttpUtils.CustomListener telemetryListener = + new TelemetryListener.Builder(metricCollector) + .requestCount(CiVisibilityCountMetric.GIT_REQUESTS_OBJECTS_PACK) + .requestErrors(CiVisibilityCountMetric.GIT_REQUESTS_OBJECTS_PACK_ERRORS) + .requestDuration(CiVisibilityDistributionMetric.GIT_REQUESTS_OBJECTS_PACK_MS) + .requestBytes(CiVisibilityDistributionMetric.GIT_REQUESTS_OBJECTS_PACK_BYTES) + .build(); + + String response = + backendApi.post(UPLOAD_PACKFILES_URI, requestBody, IOUtils::readFully, telemetryListener); LOGGER.debug("Uploading pack file {} returned response {}", packFile, response); } diff --git a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataUploaderImpl.java b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataUploaderImpl.java index e8da31861f5..03a7bee890a 100644 --- a/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataUploaderImpl.java +++ b/dd-java-agent/agent-ci-visibility/src/main/java/datadog/trace/civisibility/git/tree/GitDataUploaderImpl.java @@ -1,6 +1,8 @@ package datadog.trace.civisibility.git.tree; import datadog.trace.api.Config; +import datadog.trace.api.civisibility.telemetry.CiVisibilityDistributionMetric; +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector; import datadog.trace.api.git.GitInfo; import datadog.trace.api.git.GitInfoProvider; import datadog.trace.civisibility.utils.FileUtils; @@ -16,7 +18,7 @@ import java.util.concurrent.Future; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; -import java.util.stream.Stream; +import java.util.stream.Collectors; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -25,6 +27,7 @@ public class GitDataUploaderImpl implements GitDataUploader { private static final Logger LOGGER = LoggerFactory.getLogger(GitDataUploaderImpl.class); private final Config config; + private final CiVisibilityMetricCollector metricCollector; private final GitDataApi gitDataApi; private final GitClient gitClient; private final GitInfoProvider gitInfoProvider; @@ -35,12 +38,14 @@ public class GitDataUploaderImpl implements GitDataUploader { public GitDataUploaderImpl( Config config, + CiVisibilityMetricCollector metricCollector, GitDataApi gitDataApi, GitClient gitClient, GitInfoProvider gitInfoProvider, String repoRoot, String remoteName) { this.config = config; + this.metricCollector = metricCollector; this.gitDataApi = gitDataApi; this.gitClient = gitClient; this.gitInfoProvider = gitInfoProvider; @@ -133,17 +138,24 @@ private void uploadGitData() { String currentCommit = latestCommits.get(0); Path packFilesDirectory = gitClient.createPackFiles(objectHashes); - try (Stream packFiles = Files.list(packFilesDirectory)) { - packFiles - .filter(pf -> pf.getFileName().toString().endsWith(".pack")) // skipping ".idx" files - .forEach( - pf -> { - try { - gitDataApi.uploadPackFile(remoteUrl, currentCommit, pf); - } catch (Exception e) { - throw new RuntimeException("Could not upload pack file " + pf, e); - } - }); + try { + List packFiles = + Files.list(packFilesDirectory) + .filter( + pf -> pf.getFileName().toString().endsWith(".pack")) // skipping ".idx" files + .collect(Collectors.toList()); + + metricCollector.add( + CiVisibilityDistributionMetric.GIT_REQUESTS_OBJECTS_PACK_FILES, packFiles.size()); + + for (Path packFile : packFiles) { + try { + gitDataApi.uploadPackFile(remoteUrl, currentCommit, packFile); + } catch (Exception e) { + throw new RuntimeException("Could not upload pack file " + packFile, e); + } + } + } finally { FileUtils.delete(packFilesDirectory); } diff --git a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/config/ConfigurationApiImplTest.groovy b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/config/ConfigurationApiImplTest.groovy index d7cf3381cfd..1f66eca1066 100644 --- a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/config/ConfigurationApiImplTest.groovy +++ b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/config/ConfigurationApiImplTest.groovy @@ -6,6 +6,7 @@ import datadog.communication.http.OkHttpUtils import datadog.trace.agent.test.server.http.TestHttpServer import datadog.trace.api.civisibility.config.Configurations import datadog.trace.api.civisibility.config.TestIdentifier +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector import datadog.trace.civisibility.communication.BackendApi import datadog.trace.civisibility.communication.EvpProxyApi import datadog.trace.civisibility.communication.IntakeApi @@ -143,9 +144,10 @@ class ConfigurationApiImplTest extends Specification { def "test settings request: #displayName"() { given: def tracerEnvironment = givenTracerEnvironment() + def metricCollector = Stub(CiVisibilityMetricCollector) when: - def configurationApi = new ConfigurationApiImpl(api, () -> "1234") + def configurationApi = new ConfigurationApiImpl(api, metricCollector, () -> "1234") def settings = configurationApi.getSettings(tracerEnvironment) then: @@ -165,9 +167,10 @@ class ConfigurationApiImplTest extends Specification { def "test skippable tests request: #displayName"() { given: def tracerEnvironment = givenTracerEnvironment() + def metricCollector = Stub(CiVisibilityMetricCollector) when: - def configurationApi = new ConfigurationApiImpl(api, () -> "1234") + def configurationApi = new ConfigurationApiImpl(api, metricCollector, () -> "1234") def skippableTests = configurationApi.getSkippableTests(tracerEnvironment) then: diff --git a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataApiTest.groovy b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataApiTest.groovy index c280fe51f76..9b15c9d7b15 100644 --- a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataApiTest.groovy +++ b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataApiTest.groovy @@ -4,6 +4,7 @@ import com.squareup.moshi.Moshi import datadog.communication.http.HttpRetryPolicy import datadog.communication.http.OkHttpUtils import datadog.trace.agent.test.server.http.TestHttpServer +import datadog.trace.api.civisibility.telemetry.CiVisibilityMetricCollector import datadog.trace.civisibility.communication.BackendApi import datadog.trace.civisibility.communication.EvpProxyApi import datadog.trace.test.util.MultipartRequestParser @@ -90,10 +91,11 @@ class GitDataApiTest extends Specification { def "test commits search"() { given: + def metricCollector = Stub(CiVisibilityMetricCollector) def evpProxy = givenEvpProxy() when: - def gitDataApi = new GitDataApi(evpProxy) + def gitDataApi = new GitDataApi(evpProxy, metricCollector) def commits = new ArrayList<>(gitDataApi.searchCommits("gitRemoteUrl", ["sha1", "sha2"])) then: @@ -102,11 +104,12 @@ class GitDataApiTest extends Specification { def "test pack file upload"() { given: + def metricCollector = Stub(CiVisibilityMetricCollector) def evpProxy = givenEvpProxy() def packFile = givenPackFile() when: - def gitDataApi = new GitDataApi(evpProxy) + def gitDataApi = new GitDataApi(evpProxy, metricCollector) gitDataApi.uploadPackFile("gitRemoteUrl", "sha1", packFile) then: diff --git a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataUploaderImplTest.groovy b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataUploaderImplTest.groovy index b632296217e..8d18ffab8a1 100644 --- a/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataUploaderImplTest.groovy +++ b/dd-java-agent/agent-ci-visibility/src/test/groovy/datadog/trace/civisibility/git/tree/GitDataUploaderImplTest.groovy @@ -40,7 +40,7 @@ class GitDataUploaderImplTest extends Specification { gitInfoProvider.getGitInfo(repoRoot) >> new GitInfo(repoUrl, null, null, null) def gitClient = new GitClient(metricCollector, repoRoot, "25 years ago", 3, TIMEOUT_MILLIS) - def uploader = new GitDataUploaderImpl(config, api, gitClient, gitInfoProvider, repoRoot, "origin") + def uploader = new GitDataUploaderImpl(config, metricCollector, api, gitClient, gitInfoProvider, repoRoot, "origin") when: def future = uploader.startOrObserveGitDataUpload() diff --git a/internal-api/src/main/java/datadog/trace/api/civisibility/telemetry/CiVisibilityDistributionMetric.java b/internal-api/src/main/java/datadog/trace/api/civisibility/telemetry/CiVisibilityDistributionMetric.java index 518f9696966..258ddd6d0b8 100644 --- a/internal-api/src/main/java/datadog/trace/api/civisibility/telemetry/CiVisibilityDistributionMetric.java +++ b/internal-api/src/main/java/datadog/trace/api/civisibility/telemetry/CiVisibilityDistributionMetric.java @@ -27,7 +27,7 @@ public enum CiVisibilityDistributionMetric { GIT_REQUESTS_OBJECTS_PACK_MS("git_requests.objects_pack_ms"), /** The sum of the sizes of the object pack files inside a single payload */ GIT_REQUESTS_OBJECTS_PACK_BYTES("git_requests.objects_pack_bytes"), - /** The number of files sent in the object pack payload */ + /** The number of pack files created in a test session */ GIT_REQUESTS_OBJECTS_PACK_FILES("git_requests.objects_pack_files"), /** The time it takes to get the response of the settings endpoint request in ms */ GIT_REQUESTS_SETTINGS_MS("git_requests.settings_ms"),