Fixed data frame parser in case of frames with length == 0.

This commit is contained in:
Simone Bordet 2012-02-14 12:53:35 +01:00
parent dcdf54f557
commit 4af0a3e937
3 changed files with 115 additions and 6 deletions

View File

@ -71,12 +71,14 @@ public abstract class DataFrameParser
byte currByte = buffer.get();
--cursor;
length += (currByte & 0xFF) << 8 * cursor;
if (cursor == 0)
{
remaining = length;
state = State.DATA;
}
break;
if (cursor > 0)
break;
remaining = length;
state = State.DATA;
// Fall down if length == 0: we can't loop because the buffer
// may be empty but we need to invoke the application anyway
if (length > 0)
break;
}
case DATA:
{
@ -86,6 +88,8 @@ public abstract class DataFrameParser
// However, TCP may further split the flow control window, so we may
// only have part of the data at this point.
// TODO: introduce synthetic frames instead of accumulating data
int length = Math.min(remaining, buffer.remaining());
int limit = buffer.limit();
buffer.limit(buffer.position() + length);

View File

@ -0,0 +1,91 @@
/*
* Copyright (c) 2012 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.eclipse.jetty.spdy.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.spdy.StandardCompressionFactory;
import org.eclipse.jetty.spdy.api.DataInfo;
import org.eclipse.jetty.spdy.api.StringDataInfo;
import org.eclipse.jetty.spdy.generator.Generator;
import org.eclipse.jetty.spdy.parser.Parser;
import org.junit.Assert;
import org.junit.Test;
public class DataGenerateParseTest
{
@Test
public void testGenerateParse() throws Exception
{
testGenerateParse("test1");
}
@Test
public void testGenerateParseZeroLength() throws Exception
{
testGenerateParse("");
}
private void testGenerateParse(String content) throws Exception
{
int length = content.length();
DataInfo data = new StringDataInfo(content, true);
int streamId = 13;
Generator generator = new Generator(new StandardCompressionFactory().newCompressor());
ByteBuffer buffer = generator.data(streamId, 2 * length, data);
Assert.assertNotNull(buffer);
TestSPDYParserListener listener = new TestSPDYParserListener();
Parser parser = new Parser(new StandardCompressionFactory().newDecompressor());
parser.addListener(listener);
parser.parse(buffer);
DataFrame frame2 = listener.getDataFrame();
Assert.assertNotNull(frame2);
Assert.assertEquals(streamId, frame2.getStreamId());
Assert.assertEquals(DataInfo.FLAG_FIN, frame2.getFlags());
Assert.assertEquals(length, frame2.getLength());
Assert.assertEquals(length, listener.getData().remaining());
}
@Test
public void testGenerateParseOneByteAtATime() throws Exception
{
String content = "test2";
int length = content.length();
DataInfo data = new StringDataInfo(content, true);
int streamId = 13;
Generator generator = new Generator(new StandardCompressionFactory().newCompressor());
ByteBuffer buffer = generator.data(streamId, 2 * length, data);
Assert.assertNotNull(buffer);
TestSPDYParserListener listener = new TestSPDYParserListener();
Parser parser = new Parser(new StandardCompressionFactory().newDecompressor());
parser.addListener(listener);
while (buffer.hasRemaining())
parser.parse(ByteBuffer.wrap(new byte[]{buffer.get()}));
DataFrame frame2 = listener.getDataFrame();
Assert.assertNotNull(frame2);
Assert.assertEquals(streamId, frame2.getStreamId());
Assert.assertEquals(DataInfo.FLAG_FIN, frame2.getFlags());
Assert.assertEquals(length, frame2.getLength());
Assert.assertEquals(length, listener.getData().remaining());
}
}

View File

@ -25,6 +25,8 @@ import org.eclipse.jetty.spdy.parser.Parser;
public class TestSPDYParserListener implements Parser.Listener
{
private ControlFrame controlFrame;
private DataFrame dataFrame;
private ByteBuffer data;
@Override
public void onControlFrame(ControlFrame frame)
@ -35,6 +37,8 @@ public class TestSPDYParserListener implements Parser.Listener
@Override
public void onDataFrame(DataFrame frame, ByteBuffer data)
{
this.dataFrame = frame;
this.data = data;
}
@Override
@ -51,4 +55,14 @@ public class TestSPDYParserListener implements Parser.Listener
{
return controlFrame;
}
public DataFrame getDataFrame()
{
return dataFrame;
}
public ByteBuffer getData()
{
return data;
}
}