Skip to content

Commit b257bd5

Browse files
refactor: add shortcut to create a tenant id request context (#8)
* refactor: add shortcut to create a tenant id request context * refactor: add syntax to wrap an existing rx call * refactor: add syntax to wrap existing maybe call
1 parent c864a2e commit b257bd5

File tree

6 files changed

+93
-15
lines changed

6 files changed

+93
-15
lines changed

grpc-client-rx-utils/src/main/java/org/hypertrace/core/grpcutils/client/rx/DefaultGrpcRxExecutionContext.java

+20
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,12 @@
33
import io.grpc.Context;
44
import io.grpc.stub.StreamObserver;
55
import io.reactivex.rxjava3.core.Completable;
6+
import io.reactivex.rxjava3.core.Maybe;
67
import io.reactivex.rxjava3.core.Observable;
78
import io.reactivex.rxjava3.core.Single;
89
import java.util.concurrent.Callable;
910
import java.util.function.Consumer;
11+
import java.util.function.Supplier;
1012
import org.hypertrace.core.grpcutils.context.RequestContext;
1113

1214
class DefaultGrpcRxExecutionContext implements GrpcRxExecutionContext {
@@ -22,6 +24,24 @@ public <TResp> Single<TResp> call(Callable<TResp> callable) {
2224
return Single.fromCallable(buildContext().wrap(callable));
2325
}
2426

27+
@Override
28+
public <TResp> Single<TResp> wrapSingle(Supplier<Single<TResp>> singleSupplier) {
29+
try {
30+
return buildContext().call(singleSupplier::get);
31+
} catch (Exception e) {
32+
return Single.error(e);
33+
}
34+
}
35+
36+
@Override
37+
public <TResp> Maybe<TResp> wrapMaybe(Supplier<Maybe<TResp>> maybeSupplier) {
38+
try {
39+
return buildContext().call(maybeSupplier::get);
40+
} catch (Exception e) {
41+
return Maybe.error(e);
42+
}
43+
}
44+
2545
@Override
2646
public Completable run(Runnable runnable) {
2747
return Completable.fromRunnable(buildContext().wrap(runnable));

grpc-client-rx-utils/src/main/java/org/hypertrace/core/grpcutils/client/rx/GrpcRxExecutionContext.java

+16
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,12 @@
22

33
import io.grpc.stub.StreamObserver;
44
import io.reactivex.rxjava3.core.Completable;
5+
import io.reactivex.rxjava3.core.Maybe;
56
import io.reactivex.rxjava3.core.Observable;
67
import io.reactivex.rxjava3.core.Single;
78
import java.util.concurrent.Callable;
89
import java.util.function.Consumer;
10+
import java.util.function.Supplier;
911
import org.hypertrace.core.grpcutils.context.RequestContext;
1012

1113
/** An execution context that can turn various types of executions into their Rx equivalents. */
@@ -16,6 +18,16 @@ public interface GrpcRxExecutionContext {
1618
*/
1719
<TResp> Single<TResp> call(Callable<TResp> callable);
1820

21+
/**
22+
* Creates the provided single in this execution context, returning the result.
23+
*/
24+
<TResp> Single<TResp> wrapSingle(Supplier<Single<TResp>> singleSupplier);
25+
26+
/**
27+
* Creates the provided maybe in this execution context, returning the result.
28+
*/
29+
<TResp> Maybe<TResp> wrapMaybe(Supplier<Maybe<TResp>> maybeSupplier);
30+
1931
/**
2032
* Executes the given runnable in this execution context, triggering completion or error once the
2133
* call completes.
@@ -34,4 +46,8 @@ static GrpcRxExecutionContext forCurrentContext() {
3446
static GrpcRxExecutionContext forContext(RequestContext requestContext) {
3547
return new DefaultGrpcRxExecutionContext(requestContext);
3648
}
49+
50+
static GrpcRxExecutionContext forTenantContext(String tenantId) {
51+
return forContext(RequestContext.forTenantId(tenantId));
52+
}
3753
}

grpc-client-rx-utils/src/test/java/org/hypertrace/core/grpcutils/client/rx/DefaultGrpcRxExecutionContextTest.java

+19
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
import static org.mockito.Mockito.when;
99

1010
import io.reactivex.rxjava3.core.Completable;
11+
import io.reactivex.rxjava3.core.Maybe;
1112
import io.reactivex.rxjava3.core.Observable;
1213
import io.reactivex.rxjava3.core.Single;
1314
import io.reactivex.rxjava3.observers.TestObserver;
@@ -76,4 +77,22 @@ void canPropagateErrors() {
7677
completable.subscribe(testObserver);
7778
testObserver.assertError(UnsupportedOperationException.class);
7879
}
80+
81+
@Test
82+
void canWrapSingle() {
83+
Single<?> single =
84+
new DefaultGrpcRxExecutionContext(this.mockContext)
85+
.wrapSingle(() -> Single.just(RequestContext.CURRENT.get().getTenantId()));
86+
87+
assertEquals(TEST_TENANT_ID_OPTIONAL, single.blockingGet());
88+
}
89+
90+
@Test
91+
void canWrapMaybe() {
92+
Maybe<?> maybe =
93+
new DefaultGrpcRxExecutionContext(this.mockContext)
94+
.wrapMaybe(() -> Maybe.just(RequestContext.CURRENT.get().getTenantId()));
95+
96+
assertEquals(TEST_TENANT_ID_OPTIONAL, maybe.blockingGet());
97+
}
7998
}

grpc-client-rx-utils/src/test/java/org/hypertrace/core/grpcutils/client/rx/GrpcRxExecutionContextTest.java

+13
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
11
package org.hypertrace.core.grpcutils.client.rx;
22

3+
import static org.junit.jupiter.api.Assertions.assertEquals;
34
import static org.junit.jupiter.api.Assertions.assertSame;
45

56
import io.grpc.Context;
7+
import java.util.Optional;
68
import org.hypertrace.core.grpcutils.context.RequestContext;
79
import org.junit.jupiter.api.Test;
810
import org.junit.jupiter.api.extension.ExtendWith;
@@ -39,4 +41,15 @@ void canCreateExecutionContextWithProvidedContext() throws Exception {
3941
.call(RequestContext.CURRENT::get))
4042
.blockingGet());
4143
}
44+
45+
@Test
46+
void canCreateExecutionContextForProvidedTenant() throws Exception {
47+
final String testTenant = "testTenant";
48+
assertEquals(
49+
Optional.of(testTenant),
50+
GrpcRxExecutionContext.forTenantContext(testTenant)
51+
.call(RequestContext.CURRENT::get)
52+
.map(RequestContext::getTenantId)
53+
.blockingGet());
54+
}
4255
}

grpc-context-utils/src/main/java/org/hypertrace/core/grpcutils/context/RequestContext.java

+10-9
Original file line numberDiff line numberDiff line change
@@ -6,25 +6,26 @@
66
import java.util.Optional;
77

88
/**
9-
* Context of the GRPC request that should be carried and can made available to the services
10-
* so that the service can use them.
11-
* We use this to propagate headers across services.
9+
* Context of the GRPC request that should be carried and can made available to the services so that
10+
* the service can use them. We use this to propagate headers across services.
1211
*/
1312
public class RequestContext {
1413
public static final Context.Key<RequestContext> CURRENT = Context.key("request_context");
1514

15+
public static RequestContext forTenantId(String tenantId) {
16+
RequestContext requestContext = new RequestContext();
17+
requestContext.add(RequestContextConstants.TENANT_ID_HEADER_KEY, tenantId);
18+
return requestContext;
19+
}
20+
1621
private final Map<String, String> headers = new HashMap<>();
1722

18-
/**
19-
* Reads tenant id from this RequestContext based on the tenant id http header and returns it.
20-
*/
23+
/** Reads tenant id from this RequestContext based on the tenant id http header and returns it. */
2124
public Optional<String> getTenantId() {
2225
return get(RequestContextConstants.TENANT_ID_HEADER_KEY);
2326
}
2427

25-
/**
26-
* Method to read all GRPC request headers from this RequestContext.
27-
*/
28+
/** Method to read all GRPC request headers from this RequestContext. */
2829
public Map<String, String> getRequestHeaders() {
2930
return getAll();
3031
}

grpc-context-utils/src/test/java/org/hypertrace/core/grpcutils/context/RequestContextTest.java

+15-6
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,7 @@
55
import org.junit.jupiter.api.Assertions;
66
import org.junit.jupiter.api.Test;
77

8-
/**
9-
* Unit tests for {@link RequestContext} and utility methods in it.
10-
*/
8+
/** Unit tests for {@link RequestContext} and utility methods in it. */
119
public class RequestContextTest {
1210
private static final String TENANT_ID = "example-tenant-id";
1311
private static final String TEST_AUTH_HEADER = "Bearer sample-auth-header";
@@ -34,9 +32,20 @@ public void testGetRequestHeaders() {
3432

3533
Assertions.assertEquals(
3634
Map.of(
37-
RequestContextConstants.AUTHORIZATION_HEADER, TEST_AUTH_HEADER,
38-
"x-some-tenant-header", "v1"
39-
),
35+
RequestContextConstants.AUTHORIZATION_HEADER,
36+
TEST_AUTH_HEADER,
37+
"x-some-tenant-header",
38+
"v1"),
4039
requestHeaders);
4140
}
41+
42+
@Test
43+
public void testCreateForTenantId() {
44+
RequestContext requestContext = RequestContext.forTenantId(TENANT_ID);
45+
Assertions.assertEquals(Optional.of(TENANT_ID), requestContext.getTenantId());
46+
Assertions.assertEquals(
47+
Optional.of(TENANT_ID), requestContext.get(RequestContextConstants.TENANT_ID_HEADER_KEY));
48+
Assertions.assertEquals(
49+
Map.of(RequestContextConstants.TENANT_ID_HEADER_KEY, TENANT_ID), requestContext.getAll());
50+
}
4251
}

0 commit comments

Comments
 (0)