Skip to content

Commit 537dbe8

Browse files
authoredMar 22, 2024··
binder: Helper class to allow in process servers to use peer uids in test (#11014)
1 parent 10cb4a3 commit 537dbe8

File tree

2 files changed

+186
-0
lines changed

2 files changed

+186
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
package io.grpc.binder;
2+
3+
import static com.google.common.truth.Truth.assertThat;
4+
import static java.nio.charset.StandardCharsets.UTF_8;
5+
6+
import com.google.common.collect.ImmutableList;
7+
import io.grpc.CallOptions;
8+
import io.grpc.Channel;
9+
import io.grpc.ClientInterceptors;
10+
import io.grpc.ManagedChannel;
11+
import io.grpc.Metadata;
12+
import io.grpc.MethodDescriptor;
13+
import io.grpc.ServerCallHandler;
14+
import io.grpc.ServerInterceptor;
15+
import io.grpc.ServerInterceptors;
16+
import io.grpc.ServerServiceDefinition;
17+
import io.grpc.inprocess.InProcessChannelBuilder;
18+
import io.grpc.inprocess.InProcessServerBuilder;
19+
import io.grpc.stub.ClientCalls;
20+
import io.grpc.stub.MetadataUtils;
21+
import io.grpc.stub.ServerCalls;
22+
import io.grpc.testing.GrpcCleanupRule;
23+
import java.io.ByteArrayInputStream;
24+
import java.io.IOException;
25+
import java.io.InputStream;
26+
import java.util.concurrent.atomic.AtomicReference;
27+
import org.junit.Rule;
28+
import org.junit.Test;
29+
import org.junit.runner.RunWith;
30+
import org.junit.runners.JUnit4;
31+
32+
@RunWith(JUnit4.class)
33+
public final class PeerUidTestHelperTest {
34+
35+
@Rule
36+
public final GrpcCleanupRule grpcCleanup = new GrpcCleanupRule();
37+
38+
private static final int FAKE_UID = 12345;
39+
40+
private final AtomicReference<PeerUid> clientUidCapture = new AtomicReference<>();
41+
42+
@Test
43+
public void keyPopulatedWithInterceptorAndHeader() throws Exception {
44+
makeServiceCall(/* includeInterceptor= */ true, /* includeUidInHeader= */ true, FAKE_UID);
45+
assertThat(clientUidCapture.get()).isEqualTo(new PeerUid(FAKE_UID));
46+
}
47+
48+
@Test
49+
public void keyNotPopulatedWithInterceptorAndNoHeader() throws Exception {
50+
makeServiceCall(/* includeInterceptor= */ true, /* includeUidInHeader= */ false, /* uid= */ -1);
51+
assertThat(clientUidCapture.get()).isNull();
52+
}
53+
54+
@Test
55+
public void keyNotPopulatedWithoutInterceptorAndWithHeader() throws Exception {
56+
makeServiceCall(
57+
/* includeInterceptor= */ false, /* includeUidInHeader= */ true, /* uid= */ FAKE_UID);
58+
assertThat(clientUidCapture.get()).isNull();
59+
}
60+
61+
private final MethodDescriptor<String, String> method =
62+
MethodDescriptor.newBuilder(StringMarshaller.INSTANCE, StringMarshaller.INSTANCE)
63+
.setFullMethodName("test/method")
64+
.setType(MethodDescriptor.MethodType.UNARY)
65+
.build();
66+
67+
private void makeServiceCall(boolean includeInterceptor, boolean includeUidInHeader, int uid)
68+
throws Exception {
69+
ServerCallHandler<String, String> callHandler =
70+
ServerCalls.asyncUnaryCall(
71+
(req, respObserver) -> {
72+
clientUidCapture.set(PeerUids.REMOTE_PEER.get());
73+
respObserver.onNext(req);
74+
respObserver.onCompleted();
75+
});
76+
ImmutableList<ServerInterceptor> interceptors;
77+
if (includeInterceptor) {
78+
interceptors = ImmutableList.of(PeerUidTestHelper.newTestPeerIdentifyingServerInterceptor());
79+
} else {
80+
interceptors = ImmutableList.of();
81+
}
82+
ServerServiceDefinition serviceDef =
83+
ServerInterceptors.intercept(
84+
ServerServiceDefinition.builder("test").addMethod(method, callHandler).build(),
85+
interceptors);
86+
87+
InProcessServerBuilder server =
88+
InProcessServerBuilder.forName("test").directExecutor().addService(serviceDef);
89+
90+
grpcCleanup.register(server.build().start());
91+
92+
Channel channel = InProcessChannelBuilder.forName("test").directExecutor().build();
93+
grpcCleanup.register((ManagedChannel) channel);
94+
95+
if (includeUidInHeader) {
96+
Metadata header = new Metadata();
97+
header.put(PeerUidTestHelper.UID_KEY, uid);
98+
channel =
99+
ClientInterceptors.intercept(channel, MetadataUtils.newAttachHeadersInterceptor(header));
100+
}
101+
102+
ClientCalls.blockingUnaryCall(channel, method, CallOptions.DEFAULT, "hello");
103+
}
104+
105+
private static class StringMarshaller implements MethodDescriptor.Marshaller<String> {
106+
107+
public static final StringMarshaller INSTANCE = new StringMarshaller();
108+
109+
@Override
110+
public InputStream stream(String value) {
111+
return new ByteArrayInputStream(value.getBytes(UTF_8));
112+
}
113+
114+
@Override
115+
public String parse(InputStream stream) {
116+
try {
117+
return new String(stream.readAllBytes(), UTF_8);
118+
} catch (IOException ex) {
119+
throw new RuntimeException(ex);
120+
}
121+
}
122+
}
123+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
package io.grpc.binder;
2+
3+
import io.grpc.Context;
4+
import io.grpc.Contexts;
5+
import io.grpc.Metadata;
6+
import io.grpc.ServerCall;
7+
import io.grpc.ServerCallHandler;
8+
import io.grpc.ServerInterceptor;
9+
10+
/**
11+
* Class which helps set up {@link PeerUids} to be used in tests.
12+
*/
13+
public final class PeerUidTestHelper {
14+
15+
/**
16+
* The UID of the calling package is set with the value of this key.
17+
*/
18+
public static final Metadata.Key<Integer> UID_KEY =
19+
Metadata.Key.of("binder-remote-uid-for-unit-testing", PeerUidTestMarshaller.INSTANCE);
20+
21+
/**
22+
* Creates an interceptor that associates the {@link PeerUids#REMOTE_PEER} key in the request
23+
* {@link Context} with a UID provided by the client in the {@link #UID_KEY} request header, if
24+
* present.
25+
*
26+
* <p>The returned interceptor works with any gRPC transport but is meant for in-process unit
27+
* testing of gRPC/binder services that depend on {@link PeerUids}.
28+
*/
29+
public static ServerInterceptor newTestPeerIdentifyingServerInterceptor() {
30+
return new ServerInterceptor() {
31+
@Override
32+
public <ReqT, RespT> ServerCall.Listener<ReqT> interceptCall(
33+
ServerCall<ReqT, RespT> call, Metadata headers, ServerCallHandler<ReqT, RespT> next) {
34+
if (headers.containsKey(UID_KEY)) {
35+
Context context =
36+
Context.current().withValue(PeerUids.REMOTE_PEER, new PeerUid(headers.get(UID_KEY)));
37+
return Contexts.interceptCall(context, call, headers, next);
38+
}
39+
return next.startCall(call, headers);
40+
}
41+
};
42+
}
43+
44+
private PeerUidTestHelper() {
45+
}
46+
47+
private static class PeerUidTestMarshaller implements Metadata.AsciiMarshaller<Integer> {
48+
49+
public static final PeerUidTestMarshaller INSTANCE = new PeerUidTestMarshaller();
50+
51+
@Override
52+
public String toAsciiString(Integer value) {
53+
return value.toString();
54+
}
55+
56+
@Override
57+
public Integer parseAsciiString(String serialized) {
58+
return Integer.parseInt(serialized);
59+
}
60+
}
61+
62+
;
63+
}

0 commit comments

Comments
 (0)
Please sign in to comment.