Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix DoubleByteBuf to send large size messages in TLS mode #447

Merged
merged 4 commits into from
Jun 5, 2017

Conversation

nkurihar
Copy link
Contributor

@nkurihar nkurihar commented Jun 2, 2017

Motivation

Issue

When TLS is enabled, messages whose size is larger than 2^14(16384) bytes can not be sent by the following error:

java.lang.IndexOutOfBoundsException: index: 16384, length: 1000057 (expected: range(0, 1000057))
    at io.netty.buffer.AbstractByteBuf.checkIndex0(AbstractByteBuf.java:1143) ~[netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.buffer.AbstractByteBuf.checkIndex(AbstractByteBuf.java:1138) ~[netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.buffer.AbstractByteBuf.checkDstIndex(AbstractByteBuf.java:1157) ~[netty-all-4.0.40.Final.jar:4.0.40.Final]
    at com.yahoo.pulsar.common.api.DoubleByteBuf.getBytes(DoubleByteBuf.java:206) ~[pulsar-common-1.17.1-SNAPSHOT.jar:na]
    at com.yahoo.pulsar.common.api.DoubleByteBuf.getBytes(DoubleByteBuf.java:60) ~[pulsar-common-1.17.1-SNAPSHOT.jar:na]
    at io.netty.buffer.AbstractByteBuf.getBytes(AbstractByteBuf.java:452) ~[netty-all-4.0.40.Final.jar:4.0.40.Final]
    at com.yahoo.pulsar.common.api.DoubleByteBuf.nioBuffer(DoubleByteBuf.java:333) ~[pulsar-common-1.17.1-SNAPSHOT.jar:na]
    at com.yahoo.pulsar.common.api.DoubleByteBuf.nioBuffers(DoubleByteBuf.java:339) ~[pulsar-common-1.17.1-SNAPSHOT.jar:na]
    at com.yahoo.pulsar.common.api.DoubleByteBuf.nioBuffers(DoubleByteBuf.java:356) ~[pulsar-common-1.17.1-SNAPSHOT.jar:na]
    at io.netty.buffer.WrappedByteBuf.nioBuffers(WrappedByteBuf.java:739) ~[netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.handler.ssl.SslHandler.wrap(SslHandler.java:682) ~[netty-all-4.0.40.Final.jar:na]
    at io.netty.handler.ssl.SslHandler.wrap(SslHandler.java:522) ~[netty-all-4.0.40.Final.jar:na]
    at io.netty.handler.ssl.SslHandler.flush(SslHandler.java:497) ~[netty-all-4.0.40.Final.jar:na]
    at io.netty.channel.AbstractChannelHandlerContext.invokeFlush0(AbstractChannelHandlerContext.java:780) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.channel.AbstractChannelHandlerContext.invokeWriteAndFlush(AbstractChannelHandlerContext.java:806) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.channel.AbstractChannelHandlerContext.write(AbstractChannelHandlerContext.java:817) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.channel.AbstractChannelHandlerContext.writeAndFlush(AbstractChannelHandlerContext.java:798) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at com.yahoo.pulsar.client.impl.ProducerImpl$WriteInEventLoopCallback.run(ProducerImpl.java:358) [pulsar-client-1.17.1-SNAPSHOT.jar:na]
    at io.netty.util.concurrent.SingleThreadEventExecutor.runAllTasks(SingleThreadEventExecutor.java:408) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.channel.epoll.EpollEventLoop.run(EpollEventLoop.java:309) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.util.concurrent.SingleThreadEventExecutor$2.run(SingleThreadEventExecutor.java:140) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at io.netty.util.concurrent.DefaultThreadFactory$DefaultRunnableDecorator.run(DefaultThreadFactory.java:144) [netty-all-4.0.40.Final.jar:4.0.40.Final]
    at java.lang.Thread.run(Thread.java:745) [na:1.8.0_112]

Cause

DoubleByteBuf#readableBytes returns wrong value.

Detail

DoubleByteBuf has 2 buffers (b1, b2) internally.
DoubleByteBuf#readableBytes returns b1.readableBytes() + b2.readableBytes(), where readableBytes is defined as writerIndex(default: end of the buffer) - readerIndex(default: 0).

However, since b1.readerIndex and b2.readerIndex are not updated through any methods of DoubleByteBuf, it always returns the sum of the length of b1 and b2.

In TLS, messages whose size is larger than 2^14 bytes are divided into several chunks (c.f. 6.2.1. Fragmentation on https://www.ietf.org/rfc/rfc5246.txt).

When sending the second chunk, DoubleByteBuf#readerIndex is 16384 while DoubleByteBuf#readableBytes returns the length of buffer, IndexOutOfBoundsException occurs in DoubleByteBuf#nioBuffers which attempts to read the buffer from 16384 to 16384 + length.

Modifications

  • Delete DoubleByteBuf#readableBytes in order to use AbstractByteBuf#readableBytes instead
  • Added TlsProducerConsumerTest to verify that messages whose size is larger than 2^14 bytes can be sent.
  • Fixed typo in AuthenticatedProducerConsumerTest

Note:
In this PR I just deleted DoubleByteBuf#readableBytes.
Another approach is to update b1.readerIndex and b2.readerIndex when DoubleByteBuf#readerIndex is updated, but it seems unnecessary for now.
Should we do so?

Result

Messages whose size is larger than 2^14 bytes can be sent when TLS is enabled.

@merlimat merlimat added the type/bug The PR fixed a bug or issue reported a bug label Jun 2, 2017
@merlimat merlimat added this to the 1.18 milestone Jun 2, 2017
@merlimat
Copy link
Contributor

merlimat commented Jun 2, 2017

@nkurihar The change looks good. I just have a couple of questions:

  • What are the values of b1.readableBytes() and b2.readableBytes() ?
  • Why is this happening only for TLS and not regular messages?
  • Can you add a test method in DoubleByteBuf to demonstrate this issue just with the buffer itself?

@nkurihar
Copy link
Contributor Author

nkurihar commented Jun 5, 2017

@merlimat

What are the values of b1.readableBytes() and b2.readableBytes() ?

They are defined as b1(b2).writerIndex - b1(b2).readerIndex.
In this case, readerIndex and writerIndex are the start and the end of the buffer, respectively.
Since no method in DoubleByteBuf updates readerIndex and writerIndex, b1(b2).readableBytes() always returns the length of the buffer.

Why is this happening only for TLS and not regular messages?

If TLS is disabled, messages larger than 16384 bytes can be sent at one time.
However, if TLS is enabled, such messages are divided into chunks and sent at more than one time.

Can you add a test method in DoubleByteBuf to demonstrate this issue just with the buffer itself?

I added a test code for DoubleByteBuf#readableBytes.

@nkurihar nkurihar force-pushed the tls_message_size branch from 6d0f368 to 2b6d6aa Compare June 5, 2017 12:52
Copy link
Contributor

@merlimat merlimat left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Got it, thanks

@merlimat merlimat merged commit 049e428 into apache:master Jun 5, 2017
@nkurihar nkurihar deleted the tls_message_size branch November 29, 2017 10:09
hangc0276 pushed a commit to hangc0276/pulsar that referenced this pull request May 26, 2021
## Motivation
Keep the data in the direct memory of nio as much as possible to reduce the damage to performance.

## Modifications
1. modify `ByteBuf` directly instead of `MemoryRecords`
2. delay the release timing of `Entry` until after writing to the network channel

Co-authored-by: wuzhanpeng <wuzhanpeng@bigo.sg>
hangc0276 pushed a commit to hangc0276/pulsar that referenced this pull request May 26, 2021
…pache#454)

Motivation: Remove CompositeBytebuf to avoid on-heap memory copy in `KafkaEntryFormatter#decode`

Related to: apache#447 

Co-authored-by: wuzhanpeng <wuzhanpeng@bigo.sg>
Co-authored-by: Yunze Xu <xyzinfernity@163.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
type/bug The PR fixed a bug or issue reported a bug
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants