package io.vertx.ext.web.handler.sockjs.impl;
import io.vertx.core.AsyncResult;
import io.vertx.core.Handler;
import io.vertx.core.Vertx;
import io.vertx.core.http.HttpHeaders;
import io.vertx.core.http.HttpServerRequest;
import io.vertx.core.impl.logging.Logger;
import io.vertx.core.impl.logging.LoggerFactory;
import io.vertx.core.net.impl.URIDecoder;
import io.vertx.core.shareddata.LocalMap;
import io.vertx.ext.web.Router;
import io.vertx.ext.web.RoutingContext;
import io.vertx.ext.web.handler.sockjs.SockJSHandlerOptions;
import io.vertx.ext.web.handler.sockjs.SockJSSocket;
import java.util.regex.Pattern;
class JsonPTransport extends BaseTransport {
private static final Logger log = LoggerFactory.getLogger(JsonPTransport.class);
private static final Pattern CALLBACK_VALIDATION = Pattern.compile("[^a-zA-Z0-9-_.]");
JsonPTransport(Vertx vertx, Router router, LocalMap<String, SockJSSession> sessions, SockJSHandlerOptions options,
Handler<SockJSSocket> sockHandler) {
super(vertx, sessions, options);
String jsonpRE = COMMON_PATH_ELEMENT_RE + "jsonp";
router.getWithRegex(jsonpRE).handler(rc -> {
if (log.isTraceEnabled()) log.trace("JsonP, get: " + rc.request().uri());
String callback = rc.request().getParam("callback");
if (callback == null) {
callback = rc.request().getParam("c");
if (callback == null) {
rc.response().setStatusCode(500);
rc.response().end("\"callback\" parameter required\n");
return;
}
}
if (callback.length() > 32 || CALLBACK_VALIDATION.matcher(callback).find()) {
rc.response().setStatusCode(500);
rc.response().end("invalid \"callback\" parameter\n");
return;
}
HttpServerRequest req = rc.request();
String sessionID = req.params().get("param0");
SockJSSession session = getSession(rc, options, sessionID, sockHandler);
session.register(req, new JsonPListener(rc, session, callback));
});
String jsonpSendRE = COMMON_PATH_ELEMENT_RE + "jsonp_send";
router.postWithRegex(jsonpSendRE).handler(rc -> {
if (log.isTraceEnabled()) log.trace("JsonP, post: " + rc.request().uri());
String sessionID = rc.request().getParam("param0");
final SockJSSession session = sessions.get(sessionID);
if (session != null && !session.isClosed()) {
handleSend(rc, session);
} else {
rc.response().setStatusCode(404);
setJSESSIONID(options, rc);
rc.response().end();
}
});
}
private void handleSend(RoutingContext rc, SockJSSession session) {
rc.request().bodyHandler(buff -> {
String body = buff.toString();
boolean urlEncoded;
String ct = rc.request().getHeader(HttpHeaders.CONTENT_TYPE);
if ("application/x-www-form-urlencoded".equalsIgnoreCase(ct)) {
urlEncoded = true;
} else if ("text/plain".equalsIgnoreCase(ct)) {
urlEncoded = false;
} else {
rc.response().setStatusCode(500);
rc.response().end("Invalid Content-Type");
return;
}
if (body.equals("") || urlEncoded && (!body.startsWith("d=") || body.length() <= 2)) {
rc.response().setStatusCode(500).end("Payload expected.");
return;
}
if (urlEncoded) {
body = URIDecoder.decodeURIComponent(body, true).substring(2);
}
if (!session.handleMessages(body)) {
sendInvalidJSON(rc.response());
} else {
setJSESSIONID(options, rc);
rc.response().putHeader(HttpHeaders.CONTENT_TYPE, "text/plain; charset=UTF-8");
setNoCacheHeaders(rc);
rc.response().end("ok");
if (log.isTraceEnabled()) log.trace("send handled ok");
}
});
}
private class JsonPListener extends BaseListener {
final String callback;
boolean headersWritten;
boolean closed;
JsonPListener(RoutingContext rc, SockJSSession session, String callback) {
super(rc, session);
this.callback = callback;
addCloseHandler(rc.response(), session);
}
@Override
public void sendFrame(String body, Handler<AsyncResult<Void>> handler) {
if (log.isTraceEnabled()) log.trace("JsonP, sending frame");
if (!headersWritten) {
rc.response()
.setChunked(true)
.putHeader("X-Content-Type-Options", "nosniff")
.putHeader(HttpHeaders.CONTENT_TYPE, "application/javascript; charset=UTF-8");
setNoCacheHeaders(rc);
setJSESSIONID(options, rc);
headersWritten = true;
}
body = escapeForJavaScript(body);
String sb = "/**/" + callback + "(\"" +
body +
"\");\r\n";
rc.response().write(sb, handler);
close();
}
public void close() {
if (!closed) {
try {
session.resetListener();
rc.response().end();
rc.response().close();
closed = true;
} catch (IllegalStateException e) {
}
}
}
}
}