com.canoo.dp.impl.server.context.DolphinContextCommunicationHandler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of dolphin-platform-remoting-server Show documentation
Show all versions of dolphin-platform-remoting-server Show documentation
The Dolphin Platform is a framework that implements the presentation model pattern and provides amodern way to create enterprise applications. The Platform provides several client implementations that all canbe used in combination with a general sever API.
The newest version!
/*
* Copyright 2015-2018 Canoo Engineering AG.
*
* 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 com.canoo.dp.impl.server.context;
import com.canoo.dp.impl.platform.core.Assert;
import com.canoo.dp.impl.remoting.codec.OptimizedJsonCodec;
import com.canoo.dp.impl.remoting.commands.CreateContextCommand;
import com.canoo.dp.impl.remoting.legacy.communication.Codec;
import com.canoo.dp.impl.remoting.legacy.communication.Command;
import com.canoo.dp.impl.server.client.ClientSessionProvider;
import com.canoo.platform.server.client.ClientSession;
import org.apiguardian.api.API;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.function.Consumer;
import static org.apiguardian.api.API.Status.INTERNAL;
@API(since = "0.x", status = INTERNAL)
public class DolphinContextCommunicationHandler {
private static final Logger LOG = LoggerFactory.getLogger(DolphinContextCommunicationHandler.class);
private final static String DOLPHIN_CONTEXT_ATTRIBUTE_NAME = "DolphinContext";
private final ClientSessionProvider sessionProvider;
private final Codec codec = OptimizedJsonCodec.getInstance();
private final DolphinContextFactory contextFactory;
private static final HashMap> weakContextMap = new HashMap<>();
public DolphinContextCommunicationHandler(final ClientSessionProvider sessionProvider, DolphinContextFactory contextFactory) {
this.sessionProvider = Assert.requireNonNull(sessionProvider, "sessionProvider");
this.contextFactory = contextFactory;
}
public void handle(final HttpServletRequest request, final HttpServletResponse response) {
Assert.requireNonNull(request, "request");
Assert.requireNonNull(response, "response");
final HttpSession httpSession = Assert.requireNonNull(request.getSession(), "request.getSession()");
final ClientSession clientSession = sessionProvider.getCurrentClientSession();
if (clientSession == null) {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
LOG.error("No client session provided for request in http session {}", httpSession.getId());
return;
}
final String userAgent = request.getHeader("user-agent");
LOG.trace("receiving RPM request for client session {} in http session {} from client with user-agent {}", clientSession.getId(), httpSession.getId(), userAgent);
final List commands = new ArrayList<>();
try {
commands.addAll(readCommands(request));
} catch (final Exception e) {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
LOG.error("Can not parse request! (DolphinContext " + clientSession.getId() + ")", e);
return;
}
LOG.trace("Request for DolphinContext {} in http session {} contains {} commands", clientSession.getId(), httpSession.getId(), commands.size());
try {
DolphinContext context = getOrCreateContext(clientSession, commands);
final List results = new ArrayList<>();
try {
results.addAll(handle(context, commands));
} catch (final Exception e) {
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
LOG.error("Can not withoutResult the the received commands (DolphinContext " + context.getId() + ")", e);
return;
}
LOG.trace("Sending RPM response for client session {} in http session {} from client with user-agent {}", context.getId(), httpSession.getId(), userAgent);
LOG.trace("RPM response for client session {} in http session {} contains {} commands", context.getId(), httpSession.getId(), results.size());
try {
writeCommands(results, response);
} catch (final Exception e) {
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
LOG.error("Can not writeRequestContent response!", e);
return;
}
} catch (final Exception e) {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
LOG.error("Can not find or create matching dolphin context in session " + httpSession.getId(), e);
return;
}
}
public DolphinContext getContext(final ClientSession clientSession) {
Assert.requireNonNull(clientSession, "clientSession");
return clientSession.getAttribute(DOLPHIN_CONTEXT_ATTRIBUTE_NAME);
}
public DolphinContext getContextById(String clientSessionId) {
Assert.requireNonBlank(clientSessionId, "clientSessionId");
WeakReference ref = weakContextMap.get(clientSessionId);
DolphinContext dolphinContext = ref.get();
Assert.requireNonNull(dolphinContext, "dolphinContext");
return dolphinContext;
}
public DolphinContext getCurrentDolphinContext() {
final ClientSession clientSession = sessionProvider.getCurrentClientSession();
if (clientSession == null) {
return null;
}
return getContext(clientSession);
}
private DolphinContext getOrCreateContext(final ClientSession clientSession, final List commands) {
final DolphinContext context = getContext(clientSession);
if (context != null) {
return context;
}
if (containsInitCommand(commands)) {
final Consumer onDestroyCallback = (dolphinContext) -> {
Assert.requireNonNull(dolphinContext, "dolphinContext");
LOG.trace("Destroying DolphinContext {}", dolphinContext.getId());
remove(clientSession);
};
DolphinContext createdContext = contextFactory.create(clientSession, onDestroyCallback);
add(clientSession, createdContext);
return createdContext;
}
throw new IllegalStateException("No dolphin context is defined and no init command is send.");
}
private boolean containsInitCommand(final List commands) {
for (Command command : commands) {
if (command instanceof CreateContextCommand) {
return true;
}
}
return false;
}
private List readCommands(final HttpServletRequest request) throws IOException {
final List commands = new ArrayList<>();
final StringBuilder requestJson = new StringBuilder();
String line;
while ((line = request.getReader().readLine()) != null) {
requestJson.append(line).append("\n");
}
commands.addAll(codec.decode(requestJson.toString()));
return commands;
}
private void writeCommands(final List commands, final HttpServletResponse response) throws IOException {
response.setHeader("Content-Type", "application/json");
response.setCharacterEncoding("UTF-8");
final String jsonResponse = codec.encode(commands);
response.getWriter().print(jsonResponse);
}
private List handle(final DolphinContext context, List commands) {
final List results = new ArrayList<>();
results.addAll(context.handle(commands));
return results;
}
private void add(final ClientSession clientSession, final DolphinContext context) {
Assert.requireNonNull(clientSession, "clientSession");
Assert.requireNonNull(context, "context");
clientSession.setAttribute(DOLPHIN_CONTEXT_ATTRIBUTE_NAME, context);
weakContextMap.put(clientSession.getId(), new WeakReference<>(context));
}
private void remove(final ClientSession clientSession) {
Assert.requireNonNull(clientSession, "clientSession");
clientSession.removeAttribute(DOLPHIN_CONTEXT_ATTRIBUTE_NAME);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy