org.atmosphere.interceptor.SSEAtmosphereInterceptor 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.interceptor;
import org.atmosphere.cpr.Action;
import org.atmosphere.cpr.ApplicationConfig;
import org.atmosphere.cpr.AsyncIOInterceptorAdapter;
import org.atmosphere.cpr.AsyncIOWriter;
import org.atmosphere.cpr.AtmosphereConfig;
import org.atmosphere.cpr.AtmosphereInterceptorAdapter;
import org.atmosphere.cpr.AtmosphereInterceptorWriter;
import org.atmosphere.cpr.AtmosphereRequest;
import org.atmosphere.cpr.AtmosphereResource;
import org.atmosphere.cpr.AtmosphereResourceEvent;
import org.atmosphere.cpr.AtmosphereResponse;
import org.atmosphere.util.Utils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter;
import static org.atmosphere.cpr.ApplicationConfig.PROPERTY_USE_STREAM;
import static org.atmosphere.cpr.AtmosphereResourceEventListenerAdapter.OnPreSuspend;
import static org.atmosphere.cpr.FrameworkConfig.CALLBACK_JAVASCRIPT_PROTOCOL;
import static org.atmosphere.cpr.FrameworkConfig.CONTAINER_RESPONSE;
/**
* HTML 5 Server-Sent Events implementation.
*
* @author Jeanfrancois Arcand
*/
public class SSEAtmosphereInterceptor extends AtmosphereInterceptorAdapter {
private static final Logger logger = LoggerFactory.getLogger(SSEAtmosphereInterceptor.class);
private static final byte[] padding;
private static final String paddingText;
private static final byte[] DATA = "data:".getBytes();
private static final byte[] NEWLINE = "\r\n".getBytes();
private static final byte[] END = "\r\n\r\n".getBytes();
private String contentType = "text/event-stream";
static {
StringBuffer whitespace = new StringBuffer();
for (int i = 0; i < 2000; i++) {
whitespace.append(" ");
}
whitespace.append("\n");
paddingText = whitespace.toString();
padding = paddingText.getBytes();
}
@Override
public void configure(AtmosphereConfig config) {
String s = config.getInitParameter(ApplicationConfig.SSE_DEFAULT_CONTENTTYPE);
if (s != null) {
contentType = s;
}
}
private boolean writePadding(AtmosphereResponse response) {
if (response.request() != null && response.request().getAttribute("paddingWritten") != null) return false;
response.setContentType(contentType);
response.setCharacterEncoding("utf-8");
boolean isUsingStream = (Boolean) response.request().getAttribute(PROPERTY_USE_STREAM);
if (isUsingStream) {
try {
OutputStream stream = response.getResponse().getOutputStream();
try {
stream.write(padding);
stream.flush();
} catch (IOException ex) {
logger.warn("SSE may not work", ex);
}
} catch (IOException e) {
logger.trace("", e);
}
} else {
try {
PrintWriter w = response.getResponse().getWriter();
w.println(paddingText);
w.flush();
} catch (IOException e) {
logger.trace("", e);
}
}
response.resource().getRequest().setAttribute("paddingWritten", "true");
return true;
}
private final class P extends OnPreSuspend implements AllowInterceptor {
private final AtmosphereResponse response;
private P(AtmosphereResponse response) {
this.response = response;
}
@Override
public void onPreSuspend(AtmosphereResourceEvent event) {
writePadding(response);
}
}
@Override
public Action inspect(final AtmosphereResource r) {
if (Utils.webSocketMessage(r)) return Action.CONTINUE;
final AtmosphereResponse response = r.getResponse();
final AtmosphereRequest request = r.getRequest();
String accept = request.getHeader("Accept") == null ? "text/plain" : request.getHeader("Accept").trim();
if (r.transport().equals(AtmosphereResource.TRANSPORT.SSE) || contentType.equalsIgnoreCase(accept)) {
super.inspect(r);
r.addEventListener(new P(response));
AsyncIOWriter writer = response.getAsyncIOWriter();
if (AtmosphereInterceptorWriter.class.isAssignableFrom(writer.getClass())) {
AtmosphereInterceptorWriter.class.cast(writer).interceptor(new AsyncIOInterceptorAdapter() {
private boolean padding() {
if (!r.isSuspended()) {
return writePadding(response);
}
return false;
}
@Override
public void prePayload(AtmosphereResponse response, byte[] data, int offset, int length) {
boolean noPadding = padding();
// The CALLBACK_JAVASCRIPT_PROTOCOL may be called by a framework running on top of Atmosphere
// In that case, we must pad/protocol indenendently of the state of the AtmosphereResource
if (!noPadding || r.getRequest().getAttribute(CALLBACK_JAVASCRIPT_PROTOCOL) != null) {
// write other meta info such as (id, event, etc)?
response.write(DATA, true);
}
}
@Override
public byte[] transformPayload(AtmosphereResponse response, byte[] responseDraft,
byte[] data) throws IOException {
boolean noPadding = padding();
// The CALLBACK_JAVASCRIPT_PROTOCOL may be called by a framework running on top of Atmosphere
// In that case, we must pad/protocol indenendently of the state of the AtmosphereResource
if (!noPadding || r.getRequest().getAttribute(CALLBACK_JAVASCRIPT_PROTOCOL) != null) {
if (isMultilineData(responseDraft)) {
// return a padded multiline-data
return encodeMultilineData(responseDraft);
}
}
return responseDraft;
}
@Override
public void postPayload(AtmosphereResponse response, byte[] data, int offset, int length) {
// The CALLBACK_JAVASCRIPT_PROTOCOL may be called by a framework running on top of Atmosphere
// In that case, we must pad/protocol indenendently of the state of the AtmosphereResource
if (r.isSuspended() || r.getRequest().getAttribute(CALLBACK_JAVASCRIPT_PROTOCOL) != null
|| r.getRequest().getAttribute(CONTAINER_RESPONSE) != null) {
response.write(END, true);
}
/**
* When used with https://github.com/remy/polyfills/blob/master/EventSource.js , we
* resume after every message.
*/
String ua = r.getRequest().getHeader("User-Agent");
if (ua != null && ua.contains("MSIE")) {
try {
response.flushBuffer();
} catch (IOException e) {
logger.trace("", e);
}
r.resume();
}
}
});
} else {
logger.warn("Unable to apply {}. Your AsyncIOWriter must implement {}", getClass().getName(), AtmosphereInterceptorWriter.class.getName());
}
}
return Action.CONTINUE;
}
@Override
public String toString() {
return "SSE Interceptor Support";
}
// utilities
private static byte[] encodeMultilineData(byte[] data) {
// add "data" field for each line
ByteArrayOutputStream baos = new ByteArrayOutputStream();
int bp = 0;
int ep = 0;
try {
while (ep < data.length) {
int c = data[ep];
if (c == '\r') {
if (baos.size() > 0) {
baos.write(NEWLINE);
baos.write(DATA);
}
baos.write(data, bp, ep - bp);
if (ep + 1 < data.length && data[ep + 1] == '\n') {
ep++;
}
bp = ep + 1;
} else if (c == '\n') {
if (baos.size() > 0) {
baos.write(NEWLINE);
baos.write(DATA);
}
baos.write(data, bp, ep - bp);
bp = ep + 1;
}
ep++;
}
if (baos.size() > 0) {
baos.write(NEWLINE);
baos.write(DATA);
}
baos.write(data, bp, ep - bp);
} catch (IOException e) {
//ignore
}
return baos.toByteArray();
}
private static boolean isMultilineData(byte[] data) {
for (byte b : data) {
if (b == '\r' || b == '\n') {
return true;
}
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy