org.atmosphere.websocket.protocol.SimpleHttpProtocol Maven / Gradle / Ivy
/*
* Copyright 2015 Async-IO.org
*
* 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.atmosphere.websocket.protocol;
import org.atmosphere.cpr.ApplicationConfig;
import org.atmosphere.cpr.AtmosphereConfig;
import org.atmosphere.cpr.AtmosphereRequest;
import org.atmosphere.cpr.AtmosphereResourceImpl;
import org.atmosphere.cpr.FrameworkConfig;
import org.atmosphere.websocket.WebSocket;
import org.atmosphere.websocket.WebSocketProcessor;
import org.atmosphere.websocket.WebSocketProtocol;
import com.vaadin.external.org.slf4j.Logger;
import com.vaadin.external.org.slf4j.LoggerFactory;
import java.io.Serializable;
import java.net.URI;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import static org.atmosphere.websocket.protocol.ProtocolUtil.constructRequest;
/**
* Like the {@link org.atmosphere.cpr.AsynchronousProcessor} class, this class is responsible for dispatching WebSocket messages to the
* proper {@link org.atmosphere.websocket.WebSocket} implementation by wrapping the Websocket message's bytes within
* an {@link javax.servlet.http.HttpServletRequest}.
*
* The content-type is defined using {@link org.atmosphere.cpr.ApplicationConfig#WEBSOCKET_CONTENT_TYPE} property
* The method is defined using {@link org.atmosphere.cpr.ApplicationConfig#WEBSOCKET_METHOD} property
*
*
* @author Jeanfrancois Arcand
*/
public class SimpleHttpProtocol implements WebSocketProtocol, Serializable {
private static final Logger logger = LoggerFactory.getLogger(SimpleHttpProtocol.class);
protected final static String TEXT = "text/plain";
protected String contentType = TEXT;
protected String methodType = "POST";
protected String delimiter = "@@";
protected boolean destroyable;
protected boolean rewriteUri;
@Override
public void configure(AtmosphereConfig config) {
String contentType = config.getInitParameter(ApplicationConfig.WEBSOCKET_CONTENT_TYPE);
if (contentType == null) {
contentType = "text/plain";
}
this.contentType = contentType;
String methodType = config.getInitParameter(ApplicationConfig.WEBSOCKET_METHOD);
if (methodType == null) {
methodType = "POST";
}
this.methodType = methodType;
String delimiter = config.getInitParameter(ApplicationConfig.WEBSOCKET_PATH_DELIMITER);
if (delimiter == null) {
delimiter = "@@";
}
this.delimiter = delimiter;
String s = config.getInitParameter(ApplicationConfig.RECYCLE_ATMOSPHERE_REQUEST_RESPONSE);
destroyable = s != null && Boolean.valueOf(s);
rewriteUri = Boolean.valueOf(config.getInitParameter(ApplicationConfig.REWRITE_WEBSOCKET_REQUESTURI, "true"));
}
@Override
public List onMessage(WebSocket webSocket, String message) {
AtmosphereResourceImpl resource = (AtmosphereResourceImpl) webSocket.resource();
if (resource == null) {
logger.trace("The WebSocket has been closed before the message was processed.");
return null;
}
AtmosphereRequest request = resource.getRequest(false);
request.setAttribute(FrameworkConfig.WEBSOCKET_SUBPROTOCOL, FrameworkConfig.SIMPLE_HTTP_OVER_WEBSOCKET);
if (!resource.isInScope()) return Collections.emptyList();
String pathInfo = request.getPathInfo();
String requestURI = request.getRequestURI();
// This confuse some JAXRS servers like RestEasy
if (rewriteUri && (requestURI.startsWith("http://") || requestURI.startsWith("https://"))) {
logger.debug("Rewriting requestURI {}. To disable, add {} set to true as init-param",
requestURI, ApplicationConfig.REWRITE_WEBSOCKET_REQUESTURI);
requestURI = URI.create(requestURI).getPath();
request.requestURI(requestURI);
}
if (message.startsWith(delimiter)) {
int delimiterLength = delimiter.length();
int bodyBeginIndex = message.indexOf(delimiter, delimiterLength);
if (bodyBeginIndex != -1) {
pathInfo = message.substring(delimiterLength, bodyBeginIndex);
requestURI += pathInfo;
message = message.substring(bodyBeginIndex + delimiterLength);
}
}
List list = new ArrayList();
list.add(constructRequest(webSocket, pathInfo, requestURI, methodType, contentType.equalsIgnoreCase(TEXT) ? null : contentType, destroyable).body(message).build());
return list;
}
@Override
public List onMessage(WebSocket webSocket, byte[] d, final int offset, final int length) {
//Converting to a string and delegating to onMessage(WebSocket webSocket, String d) causes issues because the binary data may not be a valid string.
AtmosphereResourceImpl resource = (AtmosphereResourceImpl) webSocket.resource();
if (resource == null) {
logger.trace("The WebSocket has been closed before the message was processed.");
return null;
}
AtmosphereRequest request = resource.getRequest(false);
request.setAttribute(FrameworkConfig.WEBSOCKET_SUBPROTOCOL, FrameworkConfig.SIMPLE_HTTP_OVER_WEBSOCKET);
if (!resource.isInScope()) return Collections.emptyList();
List list = new ArrayList();
list.add(constructRequest(webSocket, request.getPathInfo(), request.getRequestURI(), methodType, contentType.equalsIgnoreCase(TEXT) ? null : contentType, destroyable).body(d, offset, length).build());
return list;
}
@Override
public void onOpen(WebSocket webSocket) {
}
@Override
public void onClose(WebSocket webSocket) {
}
@Override
public void onError(WebSocket webSocket, WebSocketProcessor.WebSocketException t) {
logger.warn(t.getMessage() + ". Unable to deliver the websocket messages to installed component." +
" Status {} Message {}", t.response().getStatus(), t.response().getStatusMessage());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy