From 39ff60f59529d6433d0585b8d28b08f94662bb3e Mon Sep 17 00:00:00 2001 From: Lachlan Roberts Date: Mon, 28 Feb 2022 18:47:36 +1100 Subject: [PATCH 1/4] Issue #7635 - implement maxBlockedStreams logic in QpackDecoder Signed-off-by: Lachlan Roberts --- .../jetty/http3/qpack/QpackDecoder.java | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackDecoder.java b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackDecoder.java index 4129979ca68..6fc49bf8091 100644 --- a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackDecoder.java +++ b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackDecoder.java @@ -16,7 +16,11 @@ package org.eclipse.jetty.http3.qpack; import java.io.IOException; import java.nio.ByteBuffer; import java.util.ArrayList; +import java.util.HashMap; +import java.util.Iterator; import java.util.List; +import java.util.Map; +import java.util.concurrent.atomic.AtomicInteger; import org.eclipse.jetty.http.HttpField; import org.eclipse.jetty.http.MetaData; @@ -50,6 +54,7 @@ public class QpackDecoder implements Dumpable private final List _encodedFieldSections = new ArrayList<>(); private final NBitIntegerParser _integerDecoder = new NBitIntegerParser(); private final InstructionHandler _instructionHandler = new InstructionHandler(); + private final Map _blockedStreams = new HashMap<>(); private int _maxHeaderSize; private int _maxBlockedStreams; @@ -100,7 +105,6 @@ public class QpackDecoder implements Dumpable public int getMaxBlockedStreams() { - // TODO: implement logic about blocked streams by calling this method. return _maxBlockedStreams; } @@ -172,6 +176,10 @@ public class QpackDecoder implements Dumpable { if (LOG.isDebugEnabled()) LOG.debug("Deferred Decoding: streamId={}, encodedFieldSection={}", streamId, encodedFieldSection); + AtomicInteger blockedFields = _blockedStreams.computeIfAbsent(streamId, id -> new AtomicInteger(0)); + blockedFields.incrementAndGet(); + if (_blockedStreams.size() > _maxBlockedStreams) + throw new QpackException.SessionException(QPACK_DECOMPRESSION_FAILED, "exceeded max blocked streams"); _encodedFieldSections.add(encodedFieldSection); } @@ -226,6 +234,7 @@ public class QpackDecoder implements Dumpable public void streamCancellation(long streamId) { _encodedFieldSections.removeIf(encodedFieldSection -> encodedFieldSection.getStreamId() == streamId); + _blockedStreams.remove(streamId); _metaDataNotifications.removeIf(notification -> notification._streamId == streamId); _instructions.add(new StreamCancellationInstruction(streamId)); notifyInstructionHandler(); @@ -234,12 +243,17 @@ public class QpackDecoder implements Dumpable private void checkEncodedFieldSections() throws QpackException { int insertCount = _context.getDynamicTable().getInsertCount(); - for (EncodedFieldSection encodedFieldSection : _encodedFieldSections) + Iterator iterator = _encodedFieldSections.iterator(); + while (iterator.hasNext()) { + EncodedFieldSection encodedFieldSection = iterator.next(); if (encodedFieldSection.getRequiredInsertCount() <= insertCount) { + iterator.remove(); long streamId = encodedFieldSection.getStreamId(); MetaData metaData = encodedFieldSection.decode(_context, _maxHeaderSize); + if (_blockedStreams.get(streamId).decrementAndGet() <= 0) + _blockedStreams.remove(streamId); if (LOG.isDebugEnabled()) LOG.debug("Decoded: streamId={}, metadata={}", streamId, metaData); From 0b00c0628ea04000e8758497c1afb03bd0c0d71e Mon Sep 17 00:00:00 2001 From: Lachlan Roberts Date: Fri, 13 May 2022 16:11:15 +1000 Subject: [PATCH 2/4] Issue #7635 - add extra testing for QPACK blocked streams Signed-off-by: Lachlan Roberts --- .../jetty/http3/qpack/QpackEncoder.java | 17 ++ .../http3/qpack/internal/StreamInfo.java | 1 + .../InsertCountIncrementInstruction.java | 5 + .../SectionAcknowledgmentInstruction.java | 5 + .../jetty/http3/qpack/BlockedStreamsTest.java | 213 ++++++++++++++++++ .../jetty/http3/qpack/QpackTestUtil.java | 71 ++++++ 6 files changed, 312 insertions(+) create mode 100644 jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java diff --git a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackEncoder.java b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackEncoder.java index 7105b5257bf..d8427466f9c 100644 --- a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackEncoder.java +++ b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/QpackEncoder.java @@ -106,6 +106,11 @@ public class QpackEncoder implements Dumpable _parser = new EncoderInstructionParser(_instructionHandler); } + Map getStreamInfoMap() + { + return _streamInfoMap; + } + public int getMaxBlockedStreams() { return _maxBlockedStreams; @@ -193,7 +198,15 @@ public class QpackEncoder implements Dumpable requiredInsertCount = entryRequiredInsertCount; } + // We should not expect section acknowledgements for 0 required insert count. sectionInfo.setRequiredInsertCount(requiredInsertCount); + if (requiredInsertCount == 0) + { + streamInfo.remove(sectionInfo); + if (streamInfo.isEmpty()) + _streamInfoMap.remove(streamId); + } + int base = dynamicTable.getBase(); int encodedInsertCount = encodeInsertCount(requiredInsertCount, dynamicTable.getCapacity()); boolean signBit = base < requiredInsertCount; @@ -472,9 +485,13 @@ public class QpackEncoder implements Dumpable // The KnownInsertCount should be updated to the earliest sent RequiredInsertCount on that stream. StreamInfo.SectionInfo sectionInfo = streamInfo.acknowledge(); + boolean wasBlocked = sectionInfo.isBlocking(); sectionInfo.release(); _knownInsertCount = Math.max(_knownInsertCount, sectionInfo.getRequiredInsertCount()); + if (wasBlocked && !streamInfo.isBlocked()) + _blockedStreams--; + // If we have no more outstanding section acknowledgments remove the StreamInfo. if (streamInfo.isEmpty()) _streamInfoMap.remove(streamId); diff --git a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/StreamInfo.java b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/StreamInfo.java index 08e7dc09af8..4d229e094b3 100644 --- a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/StreamInfo.java +++ b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/StreamInfo.java @@ -106,6 +106,7 @@ public class StreamInfo implements Iterable { entry.release(); } + _block = false; _entries.clear(); } diff --git a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/InsertCountIncrementInstruction.java b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/InsertCountIncrementInstruction.java index 00421c71558..1a57fe1507f 100644 --- a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/InsertCountIncrementInstruction.java +++ b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/InsertCountIncrementInstruction.java @@ -29,6 +29,11 @@ public class InsertCountIncrementInstruction implements Instruction _increment = increment; } + public int getIncrement() + { + return _increment; + } + @Override public void encode(ByteBufferPool.Lease lease) { diff --git a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/SectionAcknowledgmentInstruction.java b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/SectionAcknowledgmentInstruction.java index 24ff83265fc..9d6e916591c 100644 --- a/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/SectionAcknowledgmentInstruction.java +++ b/jetty-http3/http3-qpack/src/main/java/org/eclipse/jetty/http3/qpack/internal/instruction/SectionAcknowledgmentInstruction.java @@ -29,6 +29,11 @@ public class SectionAcknowledgmentInstruction implements Instruction _streamId = streamId; } + public long getStreamId() + { + return _streamId; + } + @Override public void encode(ByteBufferPool.Lease lease) { diff --git a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java new file mode 100644 index 00000000000..fa68def9404 --- /dev/null +++ b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java @@ -0,0 +1,213 @@ +package org.eclipse.jetty.http3.qpack; + +import java.nio.ByteBuffer; + +import org.eclipse.jetty.http.HttpField; +import org.eclipse.jetty.http.MetaData; +import org.eclipse.jetty.http3.qpack.QpackException.SessionException; +import org.eclipse.jetty.http3.qpack.internal.instruction.IndexedNameEntryInstruction; +import org.eclipse.jetty.http3.qpack.internal.instruction.InsertCountIncrementInstruction; +import org.eclipse.jetty.http3.qpack.internal.instruction.LiteralNameEntryInstruction; +import org.eclipse.jetty.http3.qpack.internal.instruction.SectionAcknowledgmentInstruction; +import org.eclipse.jetty.http3.qpack.internal.instruction.SetCapacityInstruction; +import org.eclipse.jetty.util.BufferUtil; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.eclipse.jetty.http3.qpack.QpackTestUtil.encode; +import static org.eclipse.jetty.http3.qpack.QpackTestUtil.toBuffer; +import static org.eclipse.jetty.http3.qpack.QpackTestUtil.toMetaData; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThan; +import static org.hamcrest.Matchers.instanceOf; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +public class BlockedStreamsTest +{ + private static final int MAX_BLOCKED_STREAMS = 5; + private static final int MAX_HEADER_SIZE = 1024; + + private QpackEncoder _encoder; + private QpackDecoder _decoder; + private TestDecoderHandler _decoderHandler; + private TestEncoderHandler _encoderHandler; + + @BeforeEach + public void before() + { + _encoderHandler = new TestEncoderHandler(); + _decoderHandler = new TestDecoderHandler(); + _encoder = new QpackEncoder(_encoderHandler, MAX_BLOCKED_STREAMS); + _decoder = new QpackDecoder(_decoderHandler, MAX_HEADER_SIZE); + } + + @Test + public void testBlockedStreams() throws Exception + { + // These settings are determined by HTTP/3 settings frames. + _encoder.setMaxBlockedStreams(2); + _decoder.setMaxBlockedStreams(2); + + // Set capacity of the encoder & decoder to allow entries to be added to the table. + int capacity = 1024; + _encoder.setCapacity(capacity); + Instruction instruction = _encoderHandler.getInstruction(); + assertThat(instruction, instanceOf(SetCapacityInstruction.class)); + _decoder.parseInstructions(QpackTestUtil.toBuffer(instruction)); + + // Encode a new field, which will be added to table. But do not forward insertion instruction to decoder, + // this will cause decoder to become "blocked" on stream 0 until receives the instruction. + HttpField entry1 = new HttpField("name1", "value1"); + ByteBuffer buffer = encode(_encoder, 0, toMetaData("GET", "/", "http", entry1)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + Instruction instruction1 = _encoderHandler.getInstruction(); + assertThat(instruction1, instanceOf(LiteralNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + // Decoder will not be able to decode this header until it receives instruction. + boolean decoded = _decoder.decode(0, buffer, _decoderHandler); + assertFalse(decoded); + assertThat(BufferUtil.remaining(buffer), equalTo(0L)); + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // Encode second field with dynamic table, do not forward instruction to decoder. + HttpField entry2 = new HttpField("name1", "value2"); + buffer = encode(_encoder, 1, toMetaData("GET", "/", "http", entry2)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + Instruction instruction2 = _encoderHandler.getInstruction(); + assertThat(instruction2, instanceOf(IndexedNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + // Decoder will not be able to decode this header until it receives instruction. + decoded = _decoder.decode(1, buffer, _decoderHandler); + assertFalse(decoded); + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // Give first instruction to get first metadata. + _decoder.parseInstructions(QpackTestUtil.toBuffer(instruction1)); + MetaData metaData = _decoderHandler.getMetaData(); + assertThat(metaData.getFields().size(), equalTo(1)); + assertThat(metaData.getFields().get(entry1.getHeader()), equalTo(entry1.getValue())); + + Instruction inc1 = _decoderHandler.getInstruction(); + assertThat(inc1, instanceOf(InsertCountIncrementInstruction.class)); + assertThat(((InsertCountIncrementInstruction)inc1).getIncrement(), equalTo(1)); + + Instruction ack1 = _decoderHandler.getInstruction(); + assertThat(ack1, instanceOf(SectionAcknowledgmentInstruction.class)); + assertThat(((SectionAcknowledgmentInstruction)ack1).getStreamId(), equalTo(0L)); + + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // Give second instruction to get second metadata. + _decoder.parseInstructions(QpackTestUtil.toBuffer(instruction2)); + metaData = _decoderHandler.getMetaData(); + assertThat(metaData.getFields().size(), equalTo(1)); + assertThat(metaData.getFields().get(entry2.getHeader()), equalTo(entry2.getValue())); + + Instruction inc2 = _decoderHandler.getInstruction(); + assertThat(inc2, instanceOf(InsertCountIncrementInstruction.class)); + assertThat(((InsertCountIncrementInstruction)inc2).getIncrement(), equalTo(1)); + + Instruction ack2 = _decoderHandler.getInstruction(); + assertThat(ack2, instanceOf(SectionAcknowledgmentInstruction.class)); + assertThat(((SectionAcknowledgmentInstruction)ack2).getStreamId(), equalTo(1L)); + + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // The encoder hasn't received any InsertCountIncrementInstruction and so it thinks there are two streams blocked. + // It should only encode literal entries to not risk blocking another stream on the decoder. + HttpField entry3 = new HttpField("name3", "value3"); + buffer = encode(_encoder, 3, toMetaData("GET", "/", "http", entry3)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + instruction = _encoderHandler.getInstruction(); + assertThat(instruction, instanceOf(LiteralNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + // Can decode literal entry immediately without any further instructions. + decoded = _decoder.decode(3, buffer, _decoderHandler); + assertTrue(decoded); + metaData = _decoderHandler.getMetaData(); + assertThat(metaData.getFields().size(), equalTo(1)); + assertThat(metaData.getFields().get(entry3.getHeader()), equalTo(entry3.getValue())); + + // No longer referencing any streams that have been acknowledged. + buffer = toBuffer(inc1, ack1, inc2, ack2); + _encoder.parseInstructions(buffer); + assertThat(BufferUtil.remaining(buffer), equalTo(0L)); + assertThat(_encoder.getStreamInfoMap().size(), equalTo(0)); + + // Encoder can now reference entries not acknowledged by the decoder again. + HttpField entry4 = new HttpField("name4", "value4"); + buffer = encode(_encoder, 4, toMetaData("GET", "/", "http", entry4)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + instruction = _encoderHandler.getInstruction(); + assertThat(instruction, instanceOf(LiteralNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + decoded = _decoder.decode(4, buffer, _decoderHandler); + assertFalse(decoded); + } + + @Test + public void testMaxBlockedStreams() throws Exception + { + // Encoder will risk blocking 1 more stream than the decoder will allow. + _encoder.setMaxBlockedStreams(3); + _decoder.setMaxBlockedStreams(2); + + // Set capacity of the encoder & decoder to allow entries to be added to the table. + int capacity = 1024; + _encoder.setCapacity(capacity); + Instruction instruction = _encoderHandler.getInstruction(); + assertThat(instruction, instanceOf(SetCapacityInstruction.class)); + _decoder.parseInstructions(QpackTestUtil.toBuffer(instruction)); + + // Encode a new field, which will be added to table. But do not forward insertion instruction to decoder, + // this will cause decoder to become "blocked" on stream 0 until receives the instruction. + HttpField entry1 = new HttpField("name1", "value1"); + ByteBuffer buffer = encode(_encoder, 0, toMetaData("GET", "/", "http", entry1)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + Instruction instruction1 = _encoderHandler.getInstruction(); + assertThat(instruction1, instanceOf(LiteralNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + // Decoder will not be able to decode this header until it receives instruction. + boolean decoded = _decoder.decode(0, buffer, _decoderHandler); + assertFalse(decoded); + assertThat(BufferUtil.remaining(buffer), equalTo(0L)); + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // Encode second field with dynamic table, do not forward instruction to decoder. + HttpField entry2 = new HttpField("name1", "value2"); + buffer = encode(_encoder, 1, toMetaData("GET", "/", "http", entry2)); + assertThat(BufferUtil.remaining(buffer), greaterThan(0L)); + Instruction instruction2 = _encoderHandler.getInstruction(); + assertThat(instruction2, instanceOf(IndexedNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + // Decoder will not be able to decode this header until it receives instruction. + decoded = _decoder.decode(1, buffer, _decoderHandler); + assertFalse(decoded); + assertNull(_decoderHandler.getMetaData()); + assertNull(_decoderHandler.getInstruction()); + + // This entry will block a 3rd stream which the decoder must not allow. + HttpField entry3 = new HttpField("name3", "value3"); + ByteBuffer encodedMetadata = encode(_encoder, 3, toMetaData("GET", "/", "http", entry3)); + assertThat(BufferUtil.remaining(encodedMetadata), greaterThan(0L)); + instruction = _encoderHandler.getInstruction(); + assertThat(instruction, instanceOf(LiteralNameEntryInstruction.class)); + assertNull(_encoderHandler.getInstruction()); + + assertThrows(SessionException.class, () -> _decoder.decode(3, encodedMetadata, _decoderHandler)); + } +} diff --git a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java index 5645bdfd605..f7ef407ae63 100644 --- a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java +++ b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java @@ -16,6 +16,10 @@ package org.eclipse.jetty.http3.qpack; import java.nio.ByteBuffer; import java.util.List; +import org.eclipse.jetty.http.HttpField; +import org.eclipse.jetty.http.HttpFields; +import org.eclipse.jetty.http.HttpVersion; +import org.eclipse.jetty.http.MetaData; import org.eclipse.jetty.io.ByteBufferPool; import org.eclipse.jetty.io.NullByteBufferPool; import org.eclipse.jetty.util.BufferUtil; @@ -27,6 +31,23 @@ import static org.hamcrest.Matchers.is; public class QpackTestUtil { + public static ByteBuffer toBuffer(Instruction... instructions) + { + ByteBufferPool.Lease lease = new ByteBufferPool.Lease(new NullByteBufferPool()); + for (Instruction instruction : instructions) + { + instruction.encode(lease); + } + ByteBuffer combinedBuffer = BufferUtil.allocate(Math.toIntExact(lease.getTotalLength())); + BufferUtil.clearToFill(combinedBuffer); + for (ByteBuffer buffer : lease.getByteBuffers()) + { + combinedBuffer.put(buffer); + } + BufferUtil.flipToFlush(combinedBuffer, 0); + return combinedBuffer; + } + public static Matcher equalsHex(String expectedString) { expectedString = expectedString.replaceAll("\\s+", ""); @@ -56,4 +77,54 @@ public class QpackTestUtil { return BufferUtil.toHexString(toBuffer(List.of(instruction))); } + + public static ByteBuffer encode(QpackEncoder encoder, long streamId, MetaData metaData) throws QpackException + { + ByteBuffer buffer = BufferUtil.allocate(1024); + BufferUtil.clearToFill(buffer); + encoder.encode(buffer, streamId, metaData); + BufferUtil.flipToFlush(buffer, 0); + return buffer; + } + + public static HttpFields.Mutable toHttpFields(HttpField field) + { + return HttpFields.build().add(field); + } + + public static MetaData toMetaData(String name, String value) + { + return toMetaData(toHttpFields(new HttpField(name, value))); + } + + public static MetaData toMetaData(String method, String path, String scheme) + { + return toMetaData(method, path, scheme, (HttpField)null); + } + + public static MetaData toMetaData(String method, String path, String scheme, HttpField ...fields) + { + HttpFields.Mutable httpFields = HttpFields.build(); + for (HttpField field : fields) + { + httpFields.add(field); + } + + return toMetaData(method, path, scheme, httpFields); + } + + public static MetaData toMetaData(String method, String path, String scheme, HttpFields.Mutable fields) + { + fields = HttpFields.build() + .put(":scheme", scheme) + .put(":method", method) + .put(":path", path) + .add(fields); + return new MetaData(HttpVersion.HTTP_3, fields); + } + + public static MetaData toMetaData(HttpFields fields) + { + return new MetaData(HttpVersion.HTTP_3, fields); + } } From b9689c2f2703d6c5b67454e7373e9846b915567f Mon Sep 17 00:00:00 2001 From: Lachlan Roberts Date: Fri, 13 May 2022 16:15:03 +1000 Subject: [PATCH 3/4] add missing licence header Signed-off-by: Lachlan Roberts --- .../jetty/http3/qpack/BlockedStreamsTest.java | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java index fa68def9404..2584c43d52a 100644 --- a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java +++ b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/BlockedStreamsTest.java @@ -1,3 +1,16 @@ +// +// ======================================================================== +// Copyright (c) 1995-2022 Mort Bay Consulting Pty Ltd and others. +// +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License v. 2.0 which is available at +// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0 +// which is available at https://www.apache.org/licenses/LICENSE-2.0. +// +// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0 +// ======================================================================== +// + package org.eclipse.jetty.http3.qpack; import java.nio.ByteBuffer; From ac56b95eba8441f2ee4896355e5a0179d96728d5 Mon Sep 17 00:00:00 2001 From: Lachlan Roberts Date: Fri, 13 May 2022 16:21:07 +1000 Subject: [PATCH 4/4] fix checkstyle violation Signed-off-by: Lachlan Roberts --- .../test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java index f7ef407ae63..fc12a5906f8 100644 --- a/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java +++ b/jetty-http3/http3-qpack/src/test/java/org/eclipse/jetty/http3/qpack/QpackTestUtil.java @@ -102,7 +102,7 @@ public class QpackTestUtil return toMetaData(method, path, scheme, (HttpField)null); } - public static MetaData toMetaData(String method, String path, String scheme, HttpField ...fields) + public static MetaData toMetaData(String method, String path, String scheme, HttpField... fields) { HttpFields.Mutable httpFields = HttpFields.build(); for (HttpField field : fields)