io.undertow.websockets.jsr.handshake.JsrHybi13Handshake Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source.
* Copyright 2014 Red Hat, Inc., and individual contributors
* as indicated by the @author tags.
*
* 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 io.undertow.websockets.jsr.handshake;
import io.undertow.websockets.WebSocketExtension;
import io.undertow.websockets.core.WebSocketChannel;
import io.undertow.websockets.core.protocol.version13.Hybi13Handshake;
import io.undertow.websockets.extensions.ExtensionHandshake;
import io.undertow.websockets.jsr.ConfiguredServerEndpoint;
import io.undertow.websockets.jsr.ExtensionImpl;
import io.undertow.websockets.spi.WebSocketHttpExchange;
import io.undertow.connector.ByteBufferPool;
import org.xnio.StreamConnection;
import javax.websocket.Extension;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* {@link Hybi13Handshake} sub-class which takes care of match against the {@link javax.websocket.server.ServerEndpointConfig} and
* stored the config in the attributes for later usage.
*
* @author Norman Maurer
*/
public final class JsrHybi13Handshake extends Hybi13Handshake {
private final ConfiguredServerEndpoint config;
public JsrHybi13Handshake(ConfiguredServerEndpoint config) {
super(Collections.emptySet(), false);
this.config = config;
}
@Override
protected void upgradeChannel(final WebSocketHttpExchange exchange, byte[] data) {
HandshakeUtil.prepareUpgrade(config.getEndpointConfiguration(), exchange);
super.upgradeChannel(exchange, data);
}
@Override
public WebSocketChannel createChannel(WebSocketHttpExchange exchange, final StreamConnection c, final ByteBufferPool buffers) {
WebSocketChannel channel = super.createChannel(exchange, c, buffers);
HandshakeUtil.setConfig(channel, config);
return channel;
}
@Override
public boolean matches(WebSocketHttpExchange exchange) {
return super.matches(exchange) && HandshakeUtil.checkOrigin(config.getEndpointConfiguration(), exchange);
}
@Override
protected String supportedSubprotols(String[] requestedSubprotocolArray) {
return HandshakeUtil.selectSubProtocol(config, requestedSubprotocolArray);
}
@Override
protected List selectedExtension(List extensionList) {
List ext = new ArrayList<>();
for(WebSocketExtension i : extensionList) {
ext.add(ExtensionImpl.create(i));
}
List selected = HandshakeUtil.selectExtensions(config, ext);
if(selected == null) {
return Collections.emptyList();
}
Map extensionMap = new HashMap<>();
for(ExtensionHandshake availible : availableExtensions) {
extensionMap.put(availible.getName(), availible);
}
List ret = new ArrayList<>();
List accepted = new ArrayList<>();
for(Extension i : selected) {
ExtensionHandshake handshake = extensionMap.get(i.getName());
if(handshake == null) {
continue; //should not happen
}
List parameters = new ArrayList<>();
for(Extension.Parameter p : i.getParameters()) {
parameters.add(new WebSocketExtension.Parameter(p.getName(), p.getValue()));
}
if(!handshake.isIncompatible(accepted)) {
WebSocketExtension accept = handshake.accept(new WebSocketExtension(i.getName(), parameters));
if (accept != null) {
ret.add(accept);
accepted.add(handshake);
}
}
}
return ret;
}
}