All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.jetty.http2.parser.HeaderParser Maven / Gradle / Ivy

The newest version!
//
// ========================================================================
// Copyright (c) 1995 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.http2.parser;

import java.nio.ByteBuffer;

import org.eclipse.jetty.http2.Flags;
import org.eclipse.jetty.http2.RateControl;
import org.eclipse.jetty.http2.frames.Frame;
import org.eclipse.jetty.http2.frames.FrameType;

/**
 * 

The parser for the frame header of HTTP/2 frames.

* * @see Parser */ public class HeaderParser { private final RateControl rateControl; private State state = State.LENGTH; private int cursor; private int length; private int type; private int flags; private int streamId; public HeaderParser(RateControl rateControl) { this.rateControl = rateControl; } public RateControl getRateControl() { return rateControl; } protected void reset() { state = State.LENGTH; cursor = 0; length = 0; type = 0; flags = 0; streamId = 0; } /** *

Parses the header bytes in the given {@code buffer}; only the header * bytes are consumed, therefore when this method returns, the buffer may * contain unconsumed bytes.

* * @param buffer the buffer to parse * @return true if the whole header bytes were parsed, false if not enough * header bytes were present in the buffer */ public boolean parse(ByteBuffer buffer) { while (buffer.hasRemaining()) { switch (state) { case LENGTH: { int octet = buffer.get() & 0xFF; length = (length << 8) + octet; if (++cursor == 3) { length &= Frame.MAX_MAX_SIZE; state = State.TYPE; } break; } case TYPE: { type = buffer.get() & 0xFF; state = State.FLAGS; break; } case FLAGS: { flags = buffer.get() & 0xFF; state = State.STREAM_ID; break; } case STREAM_ID: { if (buffer.remaining() >= 4) { streamId = buffer.getInt(); // Most significant bit MUST be ignored as per specification. streamId &= 0x7F_FF_FF_FF; return true; } else { state = State.STREAM_ID_BYTES; cursor = 4; } break; } case STREAM_ID_BYTES: { int currByte = buffer.get() & 0xFF; --cursor; streamId += currByte << (8 * cursor); if (cursor == 0) { // Most significant bit MUST be ignored as per specification. streamId &= 0x7F_FF_FF_FF; return true; } break; } default: { throw new IllegalStateException(); } } } return false; } public int getLength() { return length; } public int getFrameType() { return type; } /** *

Returns whether the given header {@link Flags flag} is present.

* * @param bit the flag to test for presence * @return whether the flag is present */ public boolean hasFlag(int bit) { return (flags & bit) == bit; } /** *

Returns the raw value of the header flag bits.

*

Use {@link #hasFlag(int)} to test for flag presence. * This method should be used just to retrieve the raw value * of all the flag bits.

* * @return the raw value of the headers flag bits * @see #hasFlag(int) */ public int getFlags() { return flags; } public int getStreamId() { return streamId; } @Override public String toString() { return String.format("[%s|%d|%d|%d]", FrameType.from(getFrameType()), getLength(), flags, getStreamId()); } private enum State { LENGTH, TYPE, FLAGS, STREAM_ID, STREAM_ID_BYTES } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy