2020-03-03 09:36:43 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 ETH Zürich, Educational Development and Technology (LET)
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package ch.ethz.seb.sebserver.gui;
|
|
|
|
|
2021-04-28 17:18:30 +02:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
2020-03-03 09:36:43 +01:00
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
import javax.servlet.ServletContext;
|
2020-10-07 15:07:08 +02:00
|
|
|
import javax.servlet.http.HttpServletResponse;
|
2020-03-03 09:36:43 +01:00
|
|
|
import javax.servlet.http.HttpSession;
|
|
|
|
|
|
|
|
import org.eclipse.rap.rwt.RWT;
|
|
|
|
import org.eclipse.rap.rwt.application.AbstractEntryPoint;
|
|
|
|
import org.eclipse.rap.rwt.application.Application;
|
|
|
|
import org.eclipse.rap.rwt.application.ApplicationConfiguration;
|
|
|
|
import org.eclipse.rap.rwt.application.EntryPoint;
|
|
|
|
import org.eclipse.rap.rwt.application.EntryPointFactory;
|
|
|
|
import org.eclipse.rap.rwt.client.WebClient;
|
|
|
|
import org.eclipse.rap.rwt.internal.theme.ThemeUtil;
|
2021-04-28 17:18:30 +02:00
|
|
|
import org.eclipse.rap.rwt.service.ResourceLoader;
|
2020-03-03 09:36:43 +01:00
|
|
|
import org.eclipse.rap.rwt.service.ServiceManager;
|
|
|
|
import org.eclipse.swt.widgets.Composite;
|
|
|
|
import org.slf4j.Logger;
|
|
|
|
import org.slf4j.LoggerFactory;
|
2022-01-19 16:06:31 +01:00
|
|
|
import org.springframework.core.env.Environment;
|
2020-10-07 15:07:08 +02:00
|
|
|
import org.springframework.http.HttpStatus;
|
2020-03-03 09:36:43 +01:00
|
|
|
import org.springframework.web.context.WebApplicationContext;
|
|
|
|
import org.springframework.web.context.support.WebApplicationContextUtils;
|
2021-06-16 13:27:58 +02:00
|
|
|
|
|
|
|
//import com.eclipsesource.rap.aria.Aria;
|
2020-03-03 09:36:43 +01:00
|
|
|
|
|
|
|
import ch.ethz.seb.sebserver.gui.service.remote.download.DownloadService;
|
|
|
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.AuthorizationContextHolder;
|
|
|
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.SEBServerAuthorizationContext;
|
|
|
|
|
|
|
|
public class RAPConfiguration implements ApplicationConfiguration {
|
|
|
|
|
2022-01-19 16:06:31 +01:00
|
|
|
public static final String ATTR_USER_SESSION_TIMEOUT = "sebserver.gui.session.timeout";
|
2020-03-03 09:36:43 +01:00
|
|
|
private static final String DEFAULT_THEME_NAME = "sebserver";
|
|
|
|
private static final Logger log = LoggerFactory.getLogger(RAPConfiguration.class);
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void configure(final Application application) {
|
|
|
|
try {
|
|
|
|
|
2020-10-06 11:50:06 +02:00
|
|
|
final String guiEntrypoint = StaticApplicationPropertyResolver
|
|
|
|
.getProperty("sebserver.gui.entrypoint", "/gui");
|
2020-10-07 15:07:08 +02:00
|
|
|
final String proctoringEntrypoint = StaticApplicationPropertyResolver
|
|
|
|
.getProperty("sebserver.gui.remote.proctoring.entrypoint", "/remote-proctoring");
|
2020-10-06 11:50:06 +02:00
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
application.addStyleSheet(DEFAULT_THEME_NAME, "resource/theme/default.css");
|
|
|
|
application.addStyleSheet(DEFAULT_THEME_NAME, "static/css/sebserver.css");
|
|
|
|
application.addStyleSheet("sms", "resource/theme/default.css");
|
|
|
|
application.addStyleSheet("sms", "static/css/sms.css");
|
|
|
|
|
2021-04-28 17:18:30 +02:00
|
|
|
application.addResource("fav_icon", new ResourceLoader() {
|
|
|
|
@Override
|
|
|
|
public InputStream getResourceAsStream(final String resourceName) throws IOException {
|
2022-06-27 15:24:21 +02:00
|
|
|
return Thread
|
|
|
|
.currentThread()
|
|
|
|
.getContextClassLoader()
|
|
|
|
.getResourceAsStream("static/images/fav_icon.png");
|
2021-04-28 17:18:30 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
final Map<String, String> properties = new HashMap<>();
|
|
|
|
properties.put(WebClient.PAGE_TITLE, "SEB Server");
|
|
|
|
properties.put(WebClient.BODY_HTML, "<big>Loading Application<big>");
|
|
|
|
properties.put(WebClient.THEME_ID, DEFAULT_THEME_NAME);
|
2021-04-28 17:18:30 +02:00
|
|
|
properties.put(WebClient.FAVICON, "fav_icon");
|
2020-10-06 11:52:25 +02:00
|
|
|
|
2020-10-06 11:50:06 +02:00
|
|
|
application.addEntryPoint(guiEntrypoint, new RAPSpringEntryPointFactory(), properties);
|
2021-07-07 17:20:51 +02:00
|
|
|
|
|
|
|
properties.put(WebClient.PAGE_TITLE, "SEB Server Proctoring");
|
|
|
|
properties.put(WebClient.BODY_HTML, "<big>Loading Application<big>");
|
|
|
|
properties.put(WebClient.THEME_ID, DEFAULT_THEME_NAME);
|
|
|
|
properties.put(WebClient.FAVICON, "fav_icon");
|
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
application.addEntryPoint(proctoringEntrypoint, new RAPRemoteProcotringEntryPointFactory(), properties);
|
2021-06-16 13:27:58 +02:00
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
} catch (final RuntimeException re) {
|
|
|
|
throw re;
|
|
|
|
} catch (final Exception e) {
|
|
|
|
log.error("Error during CSS parsing. Please check the custom CSS files for errors.", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface EntryPointService {
|
|
|
|
|
|
|
|
void loadLoginPage(final Composite parent);
|
|
|
|
|
|
|
|
void loadMainPage(final Composite parent);
|
2020-10-07 15:07:08 +02:00
|
|
|
|
|
|
|
void loadProctoringView(Composite parent);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public static final class RAPRemoteProcotringEntryPointFactory implements EntryPointFactory {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public EntryPoint create() {
|
|
|
|
return new AbstractEntryPoint() {
|
|
|
|
|
|
|
|
private static final long serialVersionUID = -1299125117752916270L;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void createContents(final Composite parent) {
|
2021-06-16 13:27:58 +02:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
final HttpSession httpSession = RWT
|
|
|
|
.getUISession(parent.getDisplay())
|
|
|
|
.getHttpSession();
|
|
|
|
|
|
|
|
final WebApplicationContext webApplicationContext = getWebApplicationContext(httpSession);
|
|
|
|
final boolean authenticated = isAuthenticated(httpSession, webApplicationContext);
|
|
|
|
if (authenticated) {
|
2022-09-01 08:44:26 +02:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
final EntryPointService entryPointService = webApplicationContext
|
|
|
|
.getBean(EntryPointService.class);
|
|
|
|
entryPointService.loadProctoringView(parent);
|
2022-09-01 08:44:26 +02:00
|
|
|
final HttpServletResponse response = RWT.getResponse();
|
|
|
|
setCORS(response);
|
2020-10-07 15:07:08 +02:00
|
|
|
} else {
|
|
|
|
final HttpServletResponse response = RWT.getResponse();
|
|
|
|
response.setStatus(HttpStatus.FORBIDDEN.value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2020-03-03 09:36:43 +01:00
|
|
|
}
|
|
|
|
|
2022-10-27 09:31:06 +02:00
|
|
|
// https://developer.chrome.com/blog/enabling-shared-array-buffer/#cross-origin-isolation
|
2022-09-01 08:44:26 +02:00
|
|
|
public static final void setCORS(final HttpServletResponse resp) {
|
2022-10-27 09:31:06 +02:00
|
|
|
// resp.addHeader("Access-Control-Allow-Origin", "*");
|
|
|
|
// resp.setHeader("Access-Control-Allow-Methods", "GET");
|
|
|
|
// resp.setHeader("Vary", "Origin");
|
|
|
|
|
|
|
|
resp.addHeader("Cross-Origin-Embedder-Policy", "require-corp");
|
|
|
|
resp.addHeader("Cross-Origin-Opener-Policy", "same-origin");
|
2022-09-01 08:44:26 +02:00
|
|
|
}
|
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
public static final class RAPSpringEntryPointFactory implements EntryPointFactory {
|
|
|
|
|
|
|
|
private boolean initialized = false;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public EntryPoint create() {
|
|
|
|
|
|
|
|
return new AbstractEntryPoint() {
|
|
|
|
|
|
|
|
private static final long serialVersionUID = -1299125117752916270L;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void createContents(final Composite parent) {
|
2022-01-19 16:06:31 +01:00
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
final HttpSession httpSession = RWT
|
|
|
|
.getUISession(parent.getDisplay())
|
|
|
|
.getHttpSession();
|
|
|
|
|
|
|
|
log.debug("Create new GUI entrypoint. HttpSession: " + httpSession);
|
|
|
|
if (httpSession == null) {
|
|
|
|
log.error("HttpSession not available from RWT.getUISession().getHttpSession()");
|
|
|
|
throw new IllegalStateException(
|
|
|
|
"HttpSession not available from RWT.getUISession().getHttpSession()");
|
|
|
|
}
|
|
|
|
|
|
|
|
final Object themeId = httpSession.getAttribute("themeId");
|
|
|
|
if (themeId != null) {
|
|
|
|
ThemeUtil.setCurrentThemeId(RWT.getUISession(parent.getDisplay()), String.valueOf(themeId));
|
|
|
|
parent.redraw();
|
|
|
|
parent.layout(true);
|
|
|
|
parent.redraw();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
final WebApplicationContext webApplicationContext = getWebApplicationContext(httpSession);
|
|
|
|
initSpringBasedRAPServices(webApplicationContext);
|
|
|
|
|
2022-01-19 16:06:31 +01:00
|
|
|
final Environment environment = webApplicationContext.getBean(Environment.class);
|
|
|
|
if (environment != null) {
|
|
|
|
|
|
|
|
final Integer sessionTimeout = environment.getProperty(
|
|
|
|
ATTR_USER_SESSION_TIMEOUT,
|
|
|
|
Integer.class,
|
2022-02-08 11:18:01 +01:00
|
|
|
18000);
|
2022-01-19 16:06:31 +01:00
|
|
|
|
|
|
|
httpSession.setMaxInactiveInterval(sessionTimeout);
|
|
|
|
} else {
|
2022-02-08 11:18:01 +01:00
|
|
|
httpSession.setMaxInactiveInterval(18000);
|
2022-01-19 16:06:31 +01:00
|
|
|
}
|
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
final EntryPointService entryPointService = webApplicationContext
|
|
|
|
.getBean(EntryPointService.class);
|
|
|
|
|
|
|
|
if (isAuthenticated(httpSession, webApplicationContext)) {
|
|
|
|
entryPointService.loadMainPage(parent);
|
|
|
|
} else {
|
|
|
|
entryPointService.loadLoginPage(parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
private void initSpringBasedRAPServices(final WebApplicationContext webApplicationContext) {
|
|
|
|
if (!this.initialized) {
|
|
|
|
try {
|
|
|
|
final ServiceManager manager = RWT.getServiceManager();
|
|
|
|
final DownloadService downloadService = webApplicationContext.getBean(DownloadService.class);
|
|
|
|
manager.registerServiceHandler(DownloadService.DOWNLOAD_SERVICE_NAME, downloadService);
|
2021-06-16 13:27:58 +02:00
|
|
|
|
2020-03-03 09:36:43 +01:00
|
|
|
this.initialized = true;
|
|
|
|
} catch (final IllegalArgumentException iae) {
|
|
|
|
log.warn("Failed to register DownloadService on ServiceManager. Already registered: ", iae);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-07 15:07:08 +02:00
|
|
|
}
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
private static boolean isAuthenticated(
|
|
|
|
final HttpSession httpSession,
|
|
|
|
final WebApplicationContext webApplicationContext) {
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
final AuthorizationContextHolder authorizationContextHolder = webApplicationContext
|
|
|
|
.getBean(AuthorizationContextHolder.class);
|
|
|
|
final SEBServerAuthorizationContext authorizationContext = authorizationContextHolder
|
|
|
|
.getAuthorizationContext(httpSession);
|
|
|
|
return authorizationContext.isValid() && authorizationContext.isLoggedIn();
|
|
|
|
}
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
private static WebApplicationContext getWebApplicationContext(final HttpSession httpSession) {
|
|
|
|
try {
|
|
|
|
final ServletContext servletContext = httpSession.getServletContext();
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
log.debug("Initialize Spring-Context on Servlet-Context: " + servletContext);
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
return WebApplicationContextUtils
|
|
|
|
.getRequiredWebApplicationContext(servletContext);
|
2020-03-03 09:36:43 +01:00
|
|
|
|
2020-10-07 15:07:08 +02:00
|
|
|
} catch (final RuntimeException e) {
|
|
|
|
log.error("Failed to initialize Spring-Context on HttpSession: " + httpSession);
|
|
|
|
throw e;
|
|
|
|
} catch (final Exception e) {
|
|
|
|
log.error("Failed to initialize Spring-Context on HttpSession: " + httpSession);
|
|
|
|
throw new RuntimeException("Failed to initialize Spring-Context on HttpSession: " + httpSession);
|
2020-03-03 09:36:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|