forked from open-telemetry/opentelemetry-java-contrib
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathGcpAuthExtensionEndToEndTest.java
More file actions
212 lines (183 loc) · 8.4 KB
/
GcpAuthExtensionEndToEndTest.java
File metadata and controls
212 lines (183 loc) · 8.4 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/
package io.opentelemetry.contrib.gcp.auth;
import static io.opentelemetry.contrib.gcp.auth.GcpAuthAutoConfigurationCustomizerProvider.GCP_USER_PROJECT_ID_KEY;
import static io.opentelemetry.contrib.gcp.auth.GcpAuthAutoConfigurationCustomizerProvider.QUOTA_USER_PROJECT_HEADER;
import static org.awaitility.Awaitility.await;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.mockserver.model.HttpRequest.request;
import static org.mockserver.model.HttpResponse.response;
import static org.mockserver.stop.Stop.stopQuietly;
import com.google.protobuf.InvalidProtocolBufferException;
import io.opentelemetry.contrib.gcp.auth.springapp.Application;
import io.opentelemetry.proto.collector.trace.v1.ExportTraceServiceRequest;
import io.opentelemetry.proto.common.v1.AnyValue;
import io.opentelemetry.proto.common.v1.KeyValue;
import io.opentelemetry.proto.trace.v1.ResourceSpans;
import java.net.URI;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.X509Certificate;
import java.time.Duration;
import java.util.Arrays;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.mockserver.client.MockServerClient;
import org.mockserver.integration.ClientAndServer;
import org.mockserver.model.Body;
import org.mockserver.model.Headers;
import org.mockserver.model.HttpRequest;
import org.mockserver.model.JsonBody;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.boot.test.web.client.TestRestTemplate;
import org.springframework.boot.test.web.server.LocalServerPort;
@SpringBootTest(
classes = {Application.class},
webEnvironment = WebEnvironment.RANDOM_PORT)
public class GcpAuthExtensionEndToEndTest {
@LocalServerPort private int testApplicationPort; // port at which the spring app is running
@Autowired private TestRestTemplate template;
// The port at which the backend server will receive telemetry
private static final int EXPORTER_ENDPOINT_PORT = 4318;
// The port at which the mock GCP OAuth 2.0 server will run
private static final int MOCK_GCP_OAUTH2_PORT = 8090;
// Backend server to which the application under test will export traces
// the export config is specified in the build.gradle file.
private static ClientAndServer backendServer;
// Mock server to intercept calls to the GCP OAuth 2.0 server and provide fake credentials
private static ClientAndServer mockGcpOAuth2Server;
private static final String DUMMY_GCP_QUOTA_PROJECT = System.getenv("GOOGLE_CLOUD_QUOTA_PROJECT");
private static final String DUMMY_GCP_PROJECT = System.getProperty("google.cloud.project");
@BeforeAll
public static void setup() throws NoSuchAlgorithmException, KeyManagementException {
// Setup proxy host(s)
System.setProperty("http.proxyHost", "localhost");
System.setProperty("http.proxyPort", MOCK_GCP_OAUTH2_PORT + "");
System.setProperty("https.proxyHost", "localhost");
System.setProperty("https.proxyPort", MOCK_GCP_OAUTH2_PORT + "");
System.setProperty("http.nonProxyHost", "localhost");
System.setProperty("https.nonProxyHost", "localhost");
// Disable SSL validation for integration test
// The OAuth2 token validation requires SSL validation
disableSSLValidation();
// Set up mock OTLP backend server to which traces will be exported
backendServer = ClientAndServer.startClientAndServer(EXPORTER_ENDPOINT_PORT);
backendServer.when(request()).respond(response().withStatusCode(200));
String accessTokenResponse =
"{\"access_token\": \"fake.access_token\",\"expires_in\": 3600, \"token_type\": \"Bearer\"}";
mockGcpOAuth2Server = ClientAndServer.startClientAndServer(MOCK_GCP_OAUTH2_PORT);
MockServerClient mockServerClient =
new MockServerClient("localhost", MOCK_GCP_OAUTH2_PORT).withSecure(true);
// mock the token refresh - always respond with 200
mockServerClient
.when(request().withMethod("POST").withPath("/token"))
.respond(
response()
.withStatusCode(200)
.withHeader("Content-Type", "application/json")
.withBody(new JsonBody(accessTokenResponse)));
}
@AfterAll
public static void teardown() {
// Stop the backend server
stopQuietly(backendServer);
stopQuietly(mockGcpOAuth2Server);
}
@Test
public void authExtensionSmokeTest() {
template.getForEntity(
URI.create("http://localhost:" + testApplicationPort + "/ping"), String.class);
await()
.atMost(Duration.ofSeconds(10))
.untilAsserted(
() -> {
HttpRequest[] requests = backendServer.retrieveRecordedRequests(request());
List<Headers> extractedHeaders = extractHeadersFromRequests(requests);
verifyRequestHeaders(extractedHeaders);
List<ResourceSpans> extractedResourceSpans =
extractResourceSpansFromRequests(requests);
verifyResourceAttributes(extractedResourceSpans);
});
}
// Helper methods
private static void disableSSLValidation()
throws NoSuchAlgorithmException, KeyManagementException {
TrustManager[] trustAllCerts =
new TrustManager[] {
new X509TrustManager() {
@Override
public void checkClientTrusted(X509Certificate[] chain, String authType) {}
@Override
public void checkServerTrusted(X509Certificate[] chain, String authType) {}
@Override
public X509Certificate[] getAcceptedIssuers() {
return null;
}
}
};
SSLContext sc = SSLContext.getInstance("SSL");
sc.init(null, trustAllCerts, new java.security.SecureRandom());
HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
}
private static void verifyResourceAttributes(List<ResourceSpans> extractedResourceSpans) {
extractedResourceSpans.forEach(
resourceSpan ->
assertTrue(
resourceSpan
.getResource()
.getAttributesList()
.contains(
KeyValue.newBuilder()
.setKey(GCP_USER_PROJECT_ID_KEY)
.setValue(AnyValue.newBuilder().setStringValue(DUMMY_GCP_PROJECT))
.build())));
}
private static void verifyRequestHeaders(List<Headers> extractedHeaders) {
assertFalse(extractedHeaders.isEmpty());
// verify if extension added the required headers
extractedHeaders.forEach(
headers -> {
assertTrue(headers.containsEntry(QUOTA_USER_PROJECT_HEADER, DUMMY_GCP_QUOTA_PROJECT));
assertTrue(headers.containsEntry("Authorization", "Bearer fake.access_token"));
});
}
private static List<Headers> extractHeadersFromRequests(HttpRequest[] requests) {
return Arrays.stream(requests).map(HttpRequest::getHeaders).collect(Collectors.toList());
}
/**
* Extract resource spans from http requests received by a telemetry collector.
*
* @param requests Request received by a http server trace collector
* @return spans extracted from the request body
*/
private static List<ResourceSpans> extractResourceSpansFromRequests(HttpRequest[] requests) {
return Arrays.stream(requests)
.map(HttpRequest::getBody)
.map(GcpAuthExtensionEndToEndTest::getExportTraceServiceRequest)
.filter(Optional::isPresent)
.map(Optional::get)
.flatMap(
exportTraceServiceRequest -> exportTraceServiceRequest.getResourceSpansList().stream())
.collect(Collectors.toList());
}
private static Optional<ExportTraceServiceRequest> getExportTraceServiceRequest(Body<?> body) {
try {
return Optional.ofNullable(ExportTraceServiceRequest.parseFrom(body.getRawBytes()));
} catch (InvalidProtocolBufferException e) {
return Optional.empty();
}
}
}