|
47 | 47 | import com.google.common.collect.ImmutableList; |
48 | 48 | import com.google.common.truth.TruthJUnit; |
49 | 49 | import io.grpc.Attributes; |
| 50 | +import io.grpc.CallOptions; |
50 | 51 | import io.grpc.InternalChannelz.SocketStats; |
| 52 | +import io.grpc.Metadata; |
51 | 53 | import io.grpc.ServerStreamTracer; |
52 | 54 | import io.grpc.Status; |
53 | 55 | import io.grpc.binder.AndroidComponentAddress; |
54 | 56 | import io.grpc.binder.ApiConstants; |
55 | 57 | import io.grpc.binder.AsyncSecurityPolicy; |
56 | 58 | import io.grpc.binder.SecurityPolicies; |
| 59 | +import io.grpc.binder.internal.OneWayBinderProxies.*; |
57 | 60 | import io.grpc.binder.internal.SettableAsyncSecurityPolicy.AuthRequest; |
58 | 61 | import io.grpc.internal.AbstractTransportTest; |
| 62 | +import io.grpc.internal.ClientStream; |
59 | 63 | import io.grpc.internal.ClientTransport; |
60 | 64 | import io.grpc.internal.ClientTransportFactory.ClientTransportOptions; |
61 | 65 | import io.grpc.internal.ConnectionClientTransport; |
|
66 | 70 | import io.grpc.internal.MockServerTransportListener; |
67 | 71 | import io.grpc.internal.ObjectPool; |
68 | 72 | import io.grpc.internal.SharedResourcePool; |
| 73 | +import java.io.InputStream; |
69 | 74 | import java.util.List; |
| 75 | +import java.util.concurrent.BlockingQueue; |
70 | 76 | import java.util.concurrent.Executor; |
71 | 77 | import java.util.concurrent.ScheduledExecutorService; |
72 | 78 | import org.junit.Before; |
@@ -124,6 +130,8 @@ public final class RobolectricBinderTransportTest extends AbstractTransportTest |
124 | 130 | ServiceInfo serviceInfo; |
125 | 131 |
|
126 | 132 | private int nextServerAddress; |
| 133 | + private BlockingBinderDecorator<OneWayBinderProxy> blockingDecorator = |
| 134 | + new BlockingBinderDecorator<>(); |
127 | 135 |
|
128 | 136 | @Parameter(value = 0) |
129 | 137 | public boolean preAuthServersParam; |
@@ -433,4 +441,120 @@ public void flowControlPushBack() {} |
433 | 441 | @Ignore("See BinderTransportTest#serverAlreadyListening") |
434 | 442 | @Override |
435 | 443 | public void serverAlreadyListening() {} |
| 444 | + |
| 445 | + @Test |
| 446 | + public void singleTxnMsgsDeliveredToServerOutOfOrder() throws Exception { |
| 447 | + server.start(serverListener); |
| 448 | + client = |
| 449 | + newClientTransportBuilder() |
| 450 | + .setFactory( |
| 451 | + newClientTransportFactoryBuilder() |
| 452 | + .setBinderDecorator(blockingDecorator) |
| 453 | + .buildClientTransportFactory()) |
| 454 | + .build(); |
| 455 | + runIfNotNull(client.start(mockClientTransportListener)); |
| 456 | + blockingDecorator.putNextResult(takeNextBinder(blockingDecorator)); // Endpoint binder. |
| 457 | + QueueingOneWayBinderProxy queueingServerProxy = |
| 458 | + new QueueingOneWayBinderProxy(takeNextBinder(blockingDecorator)); // Server binder. |
| 459 | + blockingDecorator.putNextResult(queueingServerProxy); |
| 460 | + |
| 461 | + verify(mockClientTransportListener, timeout(TIMEOUT_MS)).transportReady(); |
| 462 | + |
| 463 | + ClientStream stream = |
| 464 | + client.newStream(methodDescriptor, new Metadata(), CallOptions.DEFAULT, noopTracers); |
| 465 | + stream.writeMessage(methodDescriptor.streamRequest("one")); |
| 466 | + stream.writeMessage(methodDescriptor.streamRequest("two")); |
| 467 | + stream.flush(); |
| 468 | + |
| 469 | + // Expect one transaction from the client for each message. |
| 470 | + QueueingOneWayBinderProxy.Transaction tx1 = takeNextTransaction(queueingServerProxy); |
| 471 | + QueueingOneWayBinderProxy.Transaction tx2 = takeNextTransaction(queueingServerProxy); |
| 472 | + |
| 473 | + // Deliver them to the server out of order! |
| 474 | + queueingServerProxy.deliver(tx2); |
| 475 | + queueingServerProxy.deliver(tx1); |
| 476 | + |
| 477 | + MockServerTransportListener serverTransportListener = |
| 478 | + serverListener.takeListenerOrFail(TIMEOUT_MS, MILLISECONDS); |
| 479 | + MockServerTransportListener.StreamCreation streamCreation = |
| 480 | + serverTransportListener.takeStreamOrFail(TIMEOUT_MS, MILLISECONDS); |
| 481 | + streamCreation.stream.request(2); |
| 482 | + |
| 483 | + // Expect the server to deliver the messages in the order they were originally sent. |
| 484 | + InputStream msg1 = takeNextMessage(streamCreation.listener.messageQueue); |
| 485 | + assertThat(methodDescriptor.parseResponse(msg1)).isEqualTo("one"); |
| 486 | + |
| 487 | + InputStream msg2 = takeNextMessage(streamCreation.listener.messageQueue); |
| 488 | + assertThat(methodDescriptor.parseResponse(msg2)).isEqualTo("two"); |
| 489 | + } |
| 490 | + |
| 491 | + @Test |
| 492 | + public void msgFragmentsDeliveredToServerOutOfOrder() throws Exception { |
| 493 | + server.start(serverListener); |
| 494 | + client = |
| 495 | + newClientTransportBuilder() |
| 496 | + .setFactory( |
| 497 | + newClientTransportFactoryBuilder() |
| 498 | + .setBinderDecorator(blockingDecorator) |
| 499 | + .buildClientTransportFactory()) |
| 500 | + .build(); |
| 501 | + runIfNotNull(client.start(mockClientTransportListener)); |
| 502 | + blockingDecorator.putNextResult(takeNextBinder(blockingDecorator)); // Endpoint binder. |
| 503 | + QueueingOneWayBinderProxy queueingServerProxy = |
| 504 | + new QueueingOneWayBinderProxy(takeNextBinder(blockingDecorator)); // Server binder. |
| 505 | + blockingDecorator.putNextResult(queueingServerProxy); |
| 506 | + |
| 507 | + verify(mockClientTransportListener, timeout(TIMEOUT_MS)).transportReady(); |
| 508 | + |
| 509 | + ClientStream stream = |
| 510 | + client.newStream(methodDescriptor, new Metadata(), CallOptions.DEFAULT, noopTracers); |
| 511 | + |
| 512 | + String largeMessage = newStringOfLength(BlockPool.BLOCK_SIZE + 1); |
| 513 | + stream.writeMessage(methodDescriptor.streamRequest(largeMessage)); |
| 514 | + stream.flush(); |
| 515 | + |
| 516 | + // Expect the client to split largeMessage into two transactions. |
| 517 | + QueueingOneWayBinderProxy.Transaction tx1 = takeNextTransaction(queueingServerProxy); |
| 518 | + QueueingOneWayBinderProxy.Transaction tx2 = takeNextTransaction(queueingServerProxy); |
| 519 | + |
| 520 | + // Deliver them out of order! |
| 521 | + queueingServerProxy.deliver(tx2); |
| 522 | + queueingServerProxy.deliver(tx1); |
| 523 | + |
| 524 | + // Verify that the server reassembles the transactions correctly. |
| 525 | + MockServerTransportListener serverTransportListener = |
| 526 | + serverListener.takeListenerOrFail(TIMEOUT_MS, MILLISECONDS); |
| 527 | + MockServerTransportListener.StreamCreation streamCreation = |
| 528 | + serverTransportListener.takeStreamOrFail(TIMEOUT_MS, MILLISECONDS); |
| 529 | + streamCreation.stream.request(1); |
| 530 | + InputStream msg = takeNextMessage(streamCreation.listener.messageQueue); |
| 531 | + assertThat(methodDescriptor.parseResponse(msg)).isEqualTo(largeMessage); |
| 532 | + } |
| 533 | + |
| 534 | + private static OneWayBinderProxy takeNextBinder( |
| 535 | + BlockingBinderDecorator<OneWayBinderProxy> decorator) throws InterruptedException { |
| 536 | + OneWayBinderProxy proxy = decorator.takeNextRequest(TIMEOUT_MS, MILLISECONDS); |
| 537 | + assertThat(proxy).isNotNull(); |
| 538 | + return proxy; |
| 539 | + } |
| 540 | + |
| 541 | + private static QueueingOneWayBinderProxy.Transaction takeNextTransaction( |
| 542 | + QueueingOneWayBinderProxy proxy) throws InterruptedException { |
| 543 | + QueueingOneWayBinderProxy.Transaction tx = proxy.pollNextTransaction(TIMEOUT_MS, MILLISECONDS); |
| 544 | + assertThat(tx).isNotNull(); |
| 545 | + return tx; |
| 546 | + } |
| 547 | + |
| 548 | + private static InputStream takeNextMessage(BlockingQueue<InputStream> messageQueue) |
| 549 | + throws InterruptedException { |
| 550 | + InputStream msg = messageQueue.poll(TIMEOUT_MS, MILLISECONDS); |
| 551 | + assertThat(msg).isNotNull(); |
| 552 | + return msg; |
| 553 | + } |
| 554 | + |
| 555 | + private static String newStringOfLength(int numChars) { |
| 556 | + char[] chars = new char[numChars]; |
| 557 | + java.util.Arrays.fill(chars, 'x'); |
| 558 | + return new String(chars); |
| 559 | + } |
436 | 560 | } |
0 commit comments