/*
* Copyright (c) 2015, 2017 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.grizzly.http2;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import org.glassfish.grizzly.Connection;
import org.glassfish.grizzly.attributes.Attribute;
import org.glassfish.grizzly.attributes.AttributeBuilder;
import org.glassfish.grizzly.http2.Http2FrameCodec.FrameParsingState;
Author: oleksiys
/**
*
* @author oleksiys
*/
class Http2State {
private static final Attribute<Http2State> http2State =
AttributeBuilder.DEFAULT_ATTRIBUTE_BUILDER.createAttribute(
Http2State.class.getName() + ".state");
private List<ReadyListener> listeners;
static Http2State get(final Connection connection) {
return http2State.get(connection);
}
static boolean isHttp2(final Connection connection) {
final Http2State state = http2State.get(connection);
return state != null && state.isHttp2();
}
static Http2State obtain(final Connection connection) {
Http2State state = http2State.get(connection);
if (state == null) {
state = create(connection);
}
return state;
}
static Http2State create(final Connection connection) {
final Http2State state = new Http2State();
http2State.set(connection, state);
return state;
}
static void remove(final Connection connection) {
http2State.remove(connection);
}
private final AtomicReference<Status> status = new AtomicReference<>();
private final FrameParsingState frameParsingState = new FrameParsingState();
private Http2Session http2Session;
private boolean isClientHttpUpgradeRequestFinished;
private boolean isClientPrefaceSent;
public enum Status {
NEVER_HTTP2, HTTP_UPGRADE, DIRECT_UPGRADE, OPEN
}
public Http2State() {
status.set(Status.HTTP_UPGRADE);
}
public Status getStatus() {
return status.get();
}
Returns: true if this connection is not HTTP2 and never
will be in future, or false otherwise
/**
* @return <tt>true</tt> if this connection is not HTTP2 and never
* will be in future, or <tt>false</tt> otherwise
*/
boolean isNeverHttp2() {
return status.get() == Status.NEVER_HTTP2;
}
Marks the connection as never be used for HTTP2.
/**
* Marks the connection as never be used for HTTP2.
*/
void setNeverHttp2() {
status.set(Status.NEVER_HTTP2);
}
boolean isHttp2() {
return !isNeverHttp2();
}
Returns: true if HTTP2 connection received preface from the peer,
or false otherwise
/**
* @return <tt>true</tt> if HTTP2 connection received preface from the peer,
* or <tt>false</tt> otherwise
*/
public boolean isReady() {
return status.get() == Status.OPEN;
}
public synchronized void addReadyListener(final ReadyListener... readyListeners) {
if (readyListeners == null) {
return;
}
if (isReady()) {
for (int i = 0, len = readyListeners.length; i < len; i++) {
readyListeners[i].ready(http2Session);
}
} else {
if (listeners == null) {
listeners = new ArrayList<>(readyListeners.length + 2);
}
for (int i = 0, len = readyListeners.length; i < len; i++) {
listeners.add(readyListeners[i]);
}
}
}
Confirms that HTTP2 connection received preface from the peer.
/**
* Confirms that HTTP2 connection received preface from the peer.
*/
void setOpen() {
status.set(Status.OPEN);
notifyReadyListeners();
}
boolean isHttpUpgradePhase() {
return status.get() == Status.HTTP_UPGRADE;
}
void finishHttpUpgradePhase() {
status.compareAndSet(Status.HTTP_UPGRADE, Status.DIRECT_UPGRADE);
}
void setDirectUpgradePhase() {
status.set(Status.DIRECT_UPGRADE);
}
FrameParsingState getFrameParsingState() {
return frameParsingState;
}
Http2Session getHttp2Session() {
return http2Session;
}
void setHttp2Session(final Http2Session http2Session) {
this.http2Session = http2Session;
this.http2Session.http2State = this;
}
Client-side only. Invoked, when a client finishes sending plain HTTP/1.x
request containing HTTP2 upgrade headers.
/**
* Client-side only. Invoked, when a client finishes sending plain HTTP/1.x
* request containing HTTP2 upgrade headers.
*/
void onClientHttpUpgradeRequestFinished() {
isClientHttpUpgradeRequestFinished = true;
}
synchronized boolean tryLockClientPreface() {
final Status s = status.get();
if (!isClientPrefaceSent && isClientHttpUpgradeRequestFinished &&
(s == Status.DIRECT_UPGRADE || s == Status.OPEN)) {
isClientPrefaceSent = true;
return true;
}
return false;
}
private synchronized void notifyReadyListeners() {
if (listeners != null && !listeners.isEmpty()) {
for (ReadyListener listener : listeners) {
listener.ready(http2Session);
}
listeners.clear();
listeners = null;
}
}
interface ReadyListener {
void ready(Http2Session http2Session);
}
}