Skip to content

Commit 8bd366d

Browse files
committed
chore(x-goog-spanner-request-id): commit foundation
This change adds in the bases to bring in the functionality for the "x-goog-spanner-request-id" that'll be used for debugging without the limits of trace continuity and sampling. Updates googleapis#3537 Address code review comments
1 parent c74369b commit 8bd366d

File tree

2 files changed

+138
-0
lines changed

2 files changed

+138
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
/*
2+
* Copyright 2025 Google LLC
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package com.google.cloud.spanner;
18+
19+
import com.google.api.core.InternalApi;
20+
import com.google.common.annotations.VisibleForTesting;
21+
import java.security.SecureRandom;
22+
import java.util.Objects;
23+
24+
@InternalApi
25+
public class XGoogSpannerRequestId {
26+
// 1. Generate the random process Id singleton.
27+
@VisibleForTesting static String RAND_PROCESS_ID = XGoogSpannerRequestId.generateRandProcessId();
28+
29+
@VisibleForTesting
30+
static final long VERSION = 1; // The version of the specification being implemented.
31+
32+
private final long nthClientId;
33+
private final long nthChannelId;
34+
private final long nthRequest;
35+
private long attempt;
36+
37+
XGoogSpannerRequestId(long nthClientId, long nthChannelId, long nthRequest, long attempt) {
38+
this.nthClientId = nthClientId;
39+
this.nthChannelId = nthChannelId;
40+
this.nthRequest = nthRequest;
41+
this.attempt = attempt;
42+
}
43+
44+
public static XGoogSpannerRequestId of(
45+
long nthClientId, long nthChannelId, long nthRequest, long attempt) {
46+
return new XGoogSpannerRequestId(nthClientId, nthChannelId, nthRequest, attempt);
47+
}
48+
49+
private static String generateRandProcessId() {
50+
return String.format("%08x", new SecureRandom().nextInt());
51+
}
52+
53+
@Override
54+
public String toString() {
55+
return String.format(
56+
"%d.%s.%d.%d.%d.%d",
57+
XGoogSpannerRequestId.VERSION,
58+
XGoogSpannerRequestId.RAND_PROCESS_ID,
59+
this.nthClientId,
60+
this.nthChannelId,
61+
this.nthRequest,
62+
this.attempt);
63+
}
64+
65+
@Override
66+
public boolean equals(Object other) {
67+
// instanceof for a null object returns false.
68+
if (!(other instanceof XGoogSpannerRequestId)) {
69+
return false;
70+
}
71+
72+
XGoogSpannerRequestId otherReqId = (XGoogSpannerRequestId) (other);
73+
74+
return Objects.equals(this.nthClientId, otherReqId.nthClientId)
75+
&& Objects.equals(this.nthChannelId, otherReqId.nthChannelId)
76+
&& Objects.equals(this.nthRequest, otherReqId.nthRequest)
77+
&& Objects.equals(this.attempt, otherReqId.attempt);
78+
}
79+
80+
@Override
81+
public int hashCode() {
82+
return Objects.hash(this.nthClientId, this.nthChannelId, this.nthRequest, this.attempt);
83+
}
84+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
/*
2+
* Copyright 2025 Google LLC
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package com.google.cloud.spanner;
18+
19+
import static org.junit.Assert.assertEquals;
20+
import static org.junit.Assert.assertNotEquals;
21+
import static org.junit.Assert.assertTrue;
22+
23+
import java.util.regex.Matcher;
24+
import java.util.regex.Pattern;
25+
import org.junit.Test;
26+
import org.junit.runner.RunWith;
27+
import org.junit.runners.JUnit4;
28+
29+
@RunWith(JUnit4.class)
30+
public class XGoogSpannerRequestIdTest {
31+
private static final Pattern REGEX_RAND_PROCESS_ID =
32+
Pattern.compile("1.([0-9a-z]{8})(\\.\\d+){3}\\.(\\d+)$");
33+
34+
@Test
35+
public void testEquals() {
36+
XGoogSpannerRequestId reqID1 = XGoogSpannerRequestId.of(1, 1, 1, 1);
37+
XGoogSpannerRequestId reqID2 = XGoogSpannerRequestId.of(1, 1, 1, 1);
38+
assertEquals(reqID1, reqID2);
39+
assertEquals(reqID1, reqID1);
40+
assertEquals(reqID2, reqID2);
41+
42+
XGoogSpannerRequestId reqID3 = XGoogSpannerRequestId.of(1, 1, 1, 2);
43+
assertNotEquals(reqID1, reqID3);
44+
assertNotEquals(reqID3, reqID1);
45+
assertEquals(reqID3, reqID3);
46+
}
47+
48+
@Test
49+
public void testEnsureHexadecimalFormatForRandProcessID() {
50+
String str = XGoogSpannerRequestId.of(1, 2, 3, 4).toString();
51+
Matcher m = XGoogSpannerRequestIdTest.REGEX_RAND_PROCESS_ID.matcher(str);
52+
assertTrue(m.matches());
53+
}
54+
}

0 commit comments

Comments
 (0)