|
18 | 18 |
|
19 | 19 | import static android.os.IBinder.FLAG_ONEWAY; |
20 | 20 | import static android.os.Process.myUid; |
| 21 | +import static com.google.common.truth.Truth.assertAbout; |
21 | 22 | import static com.google.common.truth.Truth.assertThat; |
22 | 23 | import static com.google.common.util.concurrent.MoreExecutors.directExecutor; |
23 | 24 | import static io.grpc.binder.internal.BinderTransport.REMOTE_UID; |
24 | 25 | import static io.grpc.binder.internal.BinderTransport.SETUP_TRANSPORT; |
25 | 26 | import static io.grpc.binder.internal.BinderTransport.SHUTDOWN_TRANSPORT; |
26 | 27 | import static io.grpc.binder.internal.BinderTransport.WIRE_FORMAT_VERSION; |
| 28 | +import static io.grpc.testing.StatusSubject.status; |
27 | 29 | import static java.util.concurrent.TimeUnit.MILLISECONDS; |
28 | 30 | import static org.junit.Assume.assumeTrue; |
29 | 31 | import static org.mockito.ArgumentMatchers.any; |
|
47 | 49 | import com.google.common.collect.ImmutableList; |
48 | 50 | import com.google.common.truth.TruthJUnit; |
49 | 51 | import io.grpc.Attributes; |
| 52 | +import io.grpc.CallOptions; |
50 | 53 | import io.grpc.InternalChannelz.SocketStats; |
| 54 | +import io.grpc.Metadata; |
51 | 55 | import io.grpc.ServerStreamTracer; |
52 | 56 | import io.grpc.Status; |
53 | 57 | import io.grpc.binder.AndroidComponentAddress; |
54 | 58 | import io.grpc.binder.ApiConstants; |
55 | 59 | import io.grpc.binder.AsyncSecurityPolicy; |
56 | 60 | import io.grpc.binder.SecurityPolicies; |
| 61 | +import io.grpc.binder.internal.OneWayBinderProxies.*; |
57 | 62 | import io.grpc.binder.internal.SettableAsyncSecurityPolicy.AuthRequest; |
58 | 63 | import io.grpc.internal.AbstractTransportTest; |
| 64 | +import io.grpc.internal.ClientStream; |
| 65 | +import io.grpc.internal.ClientStreamListenerBase; |
59 | 66 | import io.grpc.internal.ClientTransport; |
60 | 67 | import io.grpc.internal.ClientTransportFactory.ClientTransportOptions; |
61 | 68 | import io.grpc.internal.ConnectionClientTransport; |
|
66 | 73 | import io.grpc.internal.MockServerTransportListener; |
67 | 74 | import io.grpc.internal.ObjectPool; |
68 | 75 | import io.grpc.internal.SharedResourcePool; |
| 76 | +import java.io.InputStream; |
69 | 77 | import java.util.List; |
| 78 | +import java.util.concurrent.BlockingQueue; |
70 | 79 | import java.util.concurrent.Executor; |
71 | 80 | import java.util.concurrent.ScheduledExecutorService; |
72 | 81 | import org.junit.Before; |
@@ -124,6 +133,8 @@ public final class RobolectricBinderTransportTest extends AbstractTransportTest |
124 | 133 | ServiceInfo serviceInfo; |
125 | 134 |
|
126 | 135 | private int nextServerAddress; |
| 136 | + private BlockingBinderDecorator<OneWayBinderProxy> blockingDecorator = |
| 137 | + new BlockingBinderDecorator<>(); |
127 | 138 |
|
128 | 139 | @Parameter(value = 0) |
129 | 140 | public boolean preAuthServersParam; |
@@ -433,4 +444,148 @@ public void flowControlPushBack() {} |
433 | 444 | @Ignore("See BinderTransportTest#serverAlreadyListening") |
434 | 445 | @Override |
435 | 446 | public void serverAlreadyListening() {} |
| 447 | + |
| 448 | + @Test |
| 449 | + public void singleTxnMsgsDeliveredToServerOutOfOrder() throws Exception { |
| 450 | + server.start(serverListener); |
| 451 | + client = |
| 452 | + newClientTransportBuilder() |
| 453 | + .setFactory( |
| 454 | + newClientTransportFactoryBuilder() |
| 455 | + .setBinderDecorator(blockingDecorator) |
| 456 | + .buildClientTransportFactory()) |
| 457 | + .build(); |
| 458 | + runIfNotNull(client.start(mockClientTransportListener)); |
| 459 | + blockingDecorator.putNextResult(takeNextBinder(blockingDecorator)); // Endpoint binder. |
| 460 | + QueueingOneWayBinderProxy queueingServerProxy = |
| 461 | + new QueueingOneWayBinderProxy(takeNextBinder(blockingDecorator)); // Server binder. |
| 462 | + blockingDecorator.putNextResult(queueingServerProxy); |
| 463 | + |
| 464 | + verify(mockClientTransportListener, timeout(TIMEOUT_MS)).transportReady(); |
| 465 | + |
| 466 | + ClientStream stream = |
| 467 | + client.newStream(methodDescriptor, new Metadata(), CallOptions.DEFAULT, noopTracers); |
| 468 | + ClientStreamListenerBase clientStreamListener = new ClientStreamListenerBase(); |
| 469 | + stream.start(clientStreamListener); |
| 470 | + stream.writeMessage(methodDescriptor.streamRequest("one")); |
| 471 | + stream.writeMessage(methodDescriptor.streamRequest("two")); |
| 472 | + stream.halfClose(); |
| 473 | + |
| 474 | + // Expect one transaction for headers, one for each message, and one for half-close. |
| 475 | + QueueingOneWayBinderProxy.Transaction txHeaders = takeNextTransaction(queueingServerProxy); |
| 476 | + QueueingOneWayBinderProxy.Transaction tx1 = takeNextTransaction(queueingServerProxy); |
| 477 | + QueueingOneWayBinderProxy.Transaction tx2 = takeNextTransaction(queueingServerProxy); |
| 478 | + QueueingOneWayBinderProxy.Transaction txHalfClose = takeNextTransaction(queueingServerProxy); |
| 479 | + |
| 480 | + // Deliver messages out of order! |
| 481 | + queueingServerProxy.deliver(txHeaders); |
| 482 | + queueingServerProxy.deliver(tx2); |
| 483 | + queueingServerProxy.deliver(tx1); |
| 484 | + queueingServerProxy.deliver(txHalfClose); |
| 485 | + |
| 486 | + MockServerTransportListener serverTransportListener = |
| 487 | + serverListener.takeListenerOrFail(TIMEOUT_MS, MILLISECONDS); |
| 488 | + MockServerTransportListener.StreamCreation serverStreamCreation = |
| 489 | + serverTransportListener.takeStreamOrFail(TIMEOUT_MS, MILLISECONDS); |
| 490 | + serverStreamCreation.stream.request(2); |
| 491 | + |
| 492 | + // Expect the server to deliver the messages in the order they were originally sent. |
| 493 | + InputStream msg1 = takeNextMessage(serverStreamCreation.listener.messageQueue); |
| 494 | + assertThat(methodDescriptor.parseResponse(msg1)).isEqualTo("one"); |
| 495 | + |
| 496 | + InputStream msg2 = takeNextMessage(serverStreamCreation.listener.messageQueue); |
| 497 | + assertThat(methodDescriptor.parseResponse(msg2)).isEqualTo("two"); |
| 498 | + |
| 499 | + assertThat(serverStreamCreation.listener.awaitHalfClosed(TIMEOUT_MS, MILLISECONDS)).isTrue(); |
| 500 | + serverStreamCreation.stream.close(Status.OK, new Metadata()); |
| 501 | + |
| 502 | + assertAbout(status()).that(clientStreamListener.awaitClose(TIMEOUT_MS, MILLISECONDS)).isOk(); |
| 503 | + assertAbout(status()) |
| 504 | + .that(serverStreamCreation.listener.awaitClose(TIMEOUT_MS, MILLISECONDS)) |
| 505 | + .isOk(); |
| 506 | + } |
| 507 | + |
| 508 | + @Test |
| 509 | + public void msgFragmentsDeliveredToServerOutOfOrder() throws Exception { |
| 510 | + server.start(serverListener); |
| 511 | + client = |
| 512 | + newClientTransportBuilder() |
| 513 | + .setFactory( |
| 514 | + newClientTransportFactoryBuilder() |
| 515 | + .setBinderDecorator(blockingDecorator) |
| 516 | + .buildClientTransportFactory()) |
| 517 | + .build(); |
| 518 | + runIfNotNull(client.start(mockClientTransportListener)); |
| 519 | + blockingDecorator.putNextResult(takeNextBinder(blockingDecorator)); // Endpoint binder. |
| 520 | + QueueingOneWayBinderProxy queueingServerProxy = |
| 521 | + new QueueingOneWayBinderProxy(takeNextBinder(blockingDecorator)); // Server binder. |
| 522 | + blockingDecorator.putNextResult(queueingServerProxy); |
| 523 | + |
| 524 | + verify(mockClientTransportListener, timeout(TIMEOUT_MS)).transportReady(); |
| 525 | + |
| 526 | + ClientStream stream = |
| 527 | + client.newStream(methodDescriptor, new Metadata(), CallOptions.DEFAULT, noopTracers); |
| 528 | + ClientStreamListenerBase clientStreamListener = new ClientStreamListenerBase(); |
| 529 | + stream.start(clientStreamListener); |
| 530 | + |
| 531 | + String largeMessage = newStringOfLength(BlockPool.BLOCK_SIZE + 1); |
| 532 | + stream.writeMessage(methodDescriptor.streamRequest(largeMessage)); |
| 533 | + stream.halfClose(); |
| 534 | + |
| 535 | + // Expect the client to split largeMessage into two transactions, plus headers and half-close. |
| 536 | + QueueingOneWayBinderProxy.Transaction txHeaders = takeNextTransaction(queueingServerProxy); |
| 537 | + QueueingOneWayBinderProxy.Transaction tx1 = takeNextTransaction(queueingServerProxy); |
| 538 | + QueueingOneWayBinderProxy.Transaction tx2 = takeNextTransaction(queueingServerProxy); |
| 539 | + QueueingOneWayBinderProxy.Transaction txHalfClose = takeNextTransaction(queueingServerProxy); |
| 540 | + |
| 541 | + // Deliver fragments out of order! |
| 542 | + queueingServerProxy.deliver(txHeaders); |
| 543 | + queueingServerProxy.deliver(tx2); |
| 544 | + queueingServerProxy.deliver(tx1); |
| 545 | + queueingServerProxy.deliver(txHalfClose); |
| 546 | + |
| 547 | + // Verify that the server reassembles the transactions correctly. |
| 548 | + MockServerTransportListener serverTransportListener = |
| 549 | + serverListener.takeListenerOrFail(TIMEOUT_MS, MILLISECONDS); |
| 550 | + MockServerTransportListener.StreamCreation serverStreamCreation = |
| 551 | + serverTransportListener.takeStreamOrFail(TIMEOUT_MS, MILLISECONDS); |
| 552 | + serverStreamCreation.stream.request(1); |
| 553 | + InputStream msg = takeNextMessage(serverStreamCreation.listener.messageQueue); |
| 554 | + assertThat(methodDescriptor.parseResponse(msg)).isEqualTo(largeMessage); |
| 555 | + |
| 556 | + assertThat(serverStreamCreation.listener.awaitHalfClosed(TIMEOUT_MS, MILLISECONDS)).isTrue(); |
| 557 | + serverStreamCreation.stream.close(Status.OK, new Metadata()); |
| 558 | + |
| 559 | + assertAbout(status()).that(clientStreamListener.awaitClose(TIMEOUT_MS, MILLISECONDS)).isOk(); |
| 560 | + assertAbout(status()) |
| 561 | + .that(serverStreamCreation.listener.awaitClose(TIMEOUT_MS, MILLISECONDS)) |
| 562 | + .isOk(); |
| 563 | + } |
| 564 | + |
| 565 | + private static OneWayBinderProxy takeNextBinder( |
| 566 | + BlockingBinderDecorator<OneWayBinderProxy> decorator) throws InterruptedException { |
| 567 | + OneWayBinderProxy proxy = decorator.takeNextRequest(TIMEOUT_MS, MILLISECONDS); |
| 568 | + assertThat(proxy).isNotNull(); |
| 569 | + return proxy; |
| 570 | + } |
| 571 | + |
| 572 | + private static QueueingOneWayBinderProxy.Transaction takeNextTransaction( |
| 573 | + QueueingOneWayBinderProxy proxy) throws InterruptedException { |
| 574 | + QueueingOneWayBinderProxy.Transaction tx = proxy.pollNextTransaction(TIMEOUT_MS, MILLISECONDS); |
| 575 | + assertThat(tx).isNotNull(); |
| 576 | + return tx; |
| 577 | + } |
| 578 | + |
| 579 | + private static InputStream takeNextMessage(BlockingQueue<InputStream> messageQueue) |
| 580 | + throws InterruptedException { |
| 581 | + InputStream msg = messageQueue.poll(TIMEOUT_MS, MILLISECONDS); |
| 582 | + assertThat(msg).isNotNull(); |
| 583 | + return msg; |
| 584 | + } |
| 585 | + |
| 586 | + private static String newStringOfLength(int numChars) { |
| 587 | + char[] chars = new char[numChars]; |
| 588 | + java.util.Arrays.fill(chars, 'x'); |
| 589 | + return new String(chars); |
| 590 | + } |
436 | 591 | } |
0 commit comments