diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/GuiInit.java b/src/main/java/ch/ethz/seb/sebserver/gui/GuiInit.java index 88574d9b..5a139167 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/GuiInit.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/GuiInit.java @@ -1,41 +1,41 @@ -/* - * Copyright (c) 2019 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; - -import org.springframework.boot.context.event.ApplicationReadyEvent; -import org.springframework.context.ApplicationListener; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.SEBServerInit; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; - -@Component -@GuiProfile -public class GuiInit implements ApplicationListener { - - private final SEBServerInit sebServerInit; - - protected GuiInit(final SEBServerInit sebServerInit) { - this.sebServerInit = sebServerInit; - } - - @Override - public void onApplicationEvent(final ApplicationReadyEvent event) { - - this.sebServerInit.init(); - - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> **** GUI Service starting up... ****"); - - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> GUI Service sucessfully successfully started up!"); - SEBServerInit.INIT_LOGGER.info("---->"); - } - -} +/* + * Copyright (c) 2019 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; + +import org.springframework.boot.context.event.ApplicationReadyEvent; +import org.springframework.context.ApplicationListener; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.SEBServerInit; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; + +@Component +@GuiProfile +public class GuiInit implements ApplicationListener { + + private final SEBServerInit sebServerInit; + + protected GuiInit(final SEBServerInit sebServerInit) { + this.sebServerInit = sebServerInit; + } + + @Override + public void onApplicationEvent(final ApplicationReadyEvent event) { + + this.sebServerInit.init(); + + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> **** GUI Service starting up... ****"); + + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> GUI Service successfully successfully started up!"); + SEBServerInit.INIT_LOGGER.info("---->"); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/InstitutionalAuthenticationEntryPoint.java b/src/main/java/ch/ethz/seb/sebserver/gui/InstitutionalAuthenticationEntryPoint.java index 9d02c350..7f06c4e3 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/InstitutionalAuthenticationEntryPoint.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/InstitutionalAuthenticationEntryPoint.java @@ -73,12 +73,12 @@ public final class InstitutionalAuthenticationEntryPoint implements Authenticati this.webserviceURIService = webserviceURIService; this.clientHttpRequestFactoryService = clientHttpRequestFactoryService; - String _defaultLogo = null; + String _defaultLogo; if (!Constants.NO_NAME.equals(defaultLogoFileName)) { try { final String extension = ImageUploadSelection.SUPPORTED_IMAGE_FILES.stream() - .filter(ext -> defaultLogoFileName.endsWith(ext)) + .filter(defaultLogoFileName::endsWith) .findFirst() .orElse(null); @@ -141,7 +141,7 @@ public final class InstitutionalAuthenticationEntryPoint implements Authenticati : null); if (log.isDebugEnabled()) { - log.debug("Known and active gui entrypoint requested:", institutions); + log.debug("Known and active gui entrypoint requested: {}", institutions); } final String logoImageBase64 = requestLogoImage(institutionalEndpoint); @@ -184,9 +184,7 @@ public final class InstitutionalAuthenticationEntryPoint implements Authenticati } try { - return requestURI.substring( - requestURI.lastIndexOf(Constants.SLASH) + 1, - requestURI.length()); + return requestURI.substring(requestURI.lastIndexOf(Constants.SLASH) + 1); } catch (final Exception e) { log.error("Failed to extract institutional URL suffix: {}", e.getMessage()); return null; @@ -231,12 +229,12 @@ public final class InstitutionalAuthenticationEntryPoint implements Authenticati if (exchange.getStatusCodeValue() == HttpStatus.OK.value()) { return exchange.getBody(); } else { - log.warn("Failed to verify insitution from requested entrypoint url: {}, response: {}", + log.warn("Failed to verify institution from requested entrypoint url: {}, response: {}", institutionalEndpoint, exchange); } } catch (final Exception e) { - log.warn("Failed to verify insitution from requested entrypoint url: {}", + log.warn("Failed to verify institution from requested entrypoint url: {}", institutionalEndpoint, e); } @@ -245,7 +243,7 @@ public final class InstitutionalAuthenticationEntryPoint implements Authenticati } /** TODO this seems not to work as expected. Different Theme is only possible in RAP on different - * entry-points and since entry-points are statically defined within the RAPConficuration + * entry-points and since entry-points are statically defined within the RAPConfiguration * there is no possibility to apply them dynamically within an institution so far. * * @param institutionalEndpoint diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/RAPConfiguration.java b/src/main/java/ch/ethz/seb/sebserver/gui/RAPConfiguration.java index 9273d58a..a282b658 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/RAPConfiguration.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/RAPConfiguration.java @@ -1,164 +1,164 @@ -/* - * 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; - -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.ServletContext; -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; -import org.eclipse.rap.rwt.service.ServiceManager; -import org.eclipse.swt.widgets.Composite; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.web.context.WebApplicationContext; -import org.springframework.web.context.support.WebApplicationContextUtils; - -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 { - - 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 { - - // TODO get file path from properties - //application.addStyleSheet(RWT.DEFAULT_THEME_ID, "static/css/sebserver.css"); - 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"); - - final Map properties = new HashMap<>(); - properties.put(WebClient.PAGE_TITLE, "SEB Server"); - properties.put(WebClient.BODY_HTML, "Loading Application"); - properties.put(WebClient.THEME_ID, DEFAULT_THEME_NAME); - // properties.put(WebClient.FAVICON, "icons/favicon.png"); - application.addEntryPoint("/gui", new RAPSpringEntryPointFactory(), properties); - - } 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 static interface EntryPointService { - - void loadLoginPage(final Composite parent); - - void loadMainPage(final Composite parent); - } - - 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) { - 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); - - 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); - this.initialized = true; - } catch (final IllegalArgumentException iae) { - log.warn("Failed to register DownloadService on ServiceManager. Already registered: ", iae); - } - } - } - - private boolean isAuthenticated( - final HttpSession httpSession, - final WebApplicationContext webApplicationContext) { - - final AuthorizationContextHolder authorizationContextHolder = webApplicationContext - .getBean(AuthorizationContextHolder.class); - final SEBServerAuthorizationContext authorizationContext = authorizationContextHolder - .getAuthorizationContext(httpSession); - return authorizationContext.isValid() && authorizationContext.isLoggedIn(); - } - - private WebApplicationContext getWebApplicationContext(final HttpSession httpSession) { - try { - final ServletContext servletContext = httpSession.getServletContext(); - - log.debug("Initialize Spring-Context on Servlet-Context: " + servletContext); - - return WebApplicationContextUtils - .getRequiredWebApplicationContext(servletContext); - - } 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); - } - } - - }; -} +/* + * 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; + +import java.util.HashMap; +import java.util.Map; + +import javax.servlet.ServletContext; +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; +import org.eclipse.rap.rwt.service.ServiceManager; +import org.eclipse.swt.widgets.Composite; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.web.context.WebApplicationContext; +import org.springframework.web.context.support.WebApplicationContextUtils; + +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 { + + 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 { + + // TODO get file path from properties + //application.addStyleSheet(RWT.DEFAULT_THEME_ID, "static/css/sebserver.css"); + 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"); + + final Map properties = new HashMap<>(); + properties.put(WebClient.PAGE_TITLE, "SEB Server"); + properties.put(WebClient.BODY_HTML, "Loading Application"); + properties.put(WebClient.THEME_ID, DEFAULT_THEME_NAME); + // properties.put(WebClient.FAVICON, "icons/favicon.png"); + application.addEntryPoint("/gui", new RAPSpringEntryPointFactory(), properties); + + } 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); + } + + 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) { + 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); + + 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); + this.initialized = true; + } catch (final IllegalArgumentException iae) { + log.warn("Failed to register DownloadService on ServiceManager. Already registered: ", iae); + } + } + } + + private boolean isAuthenticated( + final HttpSession httpSession, + final WebApplicationContext webApplicationContext) { + + final AuthorizationContextHolder authorizationContextHolder = webApplicationContext + .getBean(AuthorizationContextHolder.class); + final SEBServerAuthorizationContext authorizationContext = authorizationContextHolder + .getAuthorizationContext(httpSession); + return authorizationContext.isValid() && authorizationContext.isLoggedIn(); + } + + private WebApplicationContext getWebApplicationContext(final HttpSession httpSession) { + try { + final ServletContext servletContext = httpSession.getServletContext(); + + log.debug("Initialize Spring-Context on Servlet-Context: " + servletContext); + + return WebApplicationContextUtils + .getRequiredWebApplicationContext(servletContext); + + } 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); + } + } + + } +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/RAPSpringConfig.java b/src/main/java/ch/ethz/seb/sebserver/gui/RAPSpringConfig.java index 67db3257..3b24c2d3 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/RAPSpringConfig.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/RAPSpringConfig.java @@ -1,83 +1,81 @@ -/* - * 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; - -import javax.servlet.ServletContext; -import javax.servlet.ServletContextListener; -import javax.servlet.ServletException; - -import org.eclipse.rap.rwt.engine.RWTServlet; -import org.eclipse.rap.rwt.engine.RWTServletContextListener; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.boot.web.servlet.ServletContextInitializer; -import org.springframework.boot.web.servlet.ServletListenerRegistrationBean; -import org.springframework.boot.web.servlet.ServletRegistrationBean; -import org.springframework.context.MessageSource; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.support.ReloadableResourceBundleMessageSource; - -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; - -@Configuration -@GuiProfile -public class RAPSpringConfig { - - private static final Logger log = LoggerFactory.getLogger(RAPSpringConfig.class); - - @Value("${sebserver.gui.entrypoint}") - private String entrypoint; - - @Value("${sebserver.gui.external.messages:messages}") - private String externalMessagesPath; - - @Bean - public ServletContextInitializer initializer() { - return new RAPServletContextInitializer(); - } - - @Bean - public ServletListenerRegistrationBean listenerRegistrationBean() { - final ServletListenerRegistrationBean bean = - new ServletListenerRegistrationBean<>(); - bean.setListener(new RWTServletContextListener()); - return bean; - } - - @Bean - public ServletRegistrationBean servletRegistrationBean() { - return new ServletRegistrationBean<>(new RWTServlet(), this.entrypoint + "/*"); - } - - @Bean - public MessageSource messageSource() { - final ReloadableResourceBundleMessageSource reloadableResourceBundleMessageSource = - new ReloadableResourceBundleMessageSource(); - - log.info(" +++ Register external messages resources from: {}", this.externalMessagesPath); - - reloadableResourceBundleMessageSource.setBasenames( - this.externalMessagesPath, - "classpath:messages"); - - return reloadableResourceBundleMessageSource; - } - - private static class RAPServletContextInitializer implements ServletContextInitializer { - @Override - public void onStartup(final ServletContext servletContext) throws ServletException { - servletContext.setInitParameter( - "org.eclipse.rap.applicationConfiguration", - RAPConfiguration.class.getName()); - } - } - -} +/* + * 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; + +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import org.eclipse.rap.rwt.engine.RWTServlet; +import org.eclipse.rap.rwt.engine.RWTServletContextListener; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.boot.web.servlet.ServletContextInitializer; +import org.springframework.boot.web.servlet.ServletListenerRegistrationBean; +import org.springframework.boot.web.servlet.ServletRegistrationBean; +import org.springframework.context.MessageSource; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.support.ReloadableResourceBundleMessageSource; + +import javax.servlet.ServletContext; +import javax.servlet.ServletContextListener; + +@Configuration +@GuiProfile +public class RAPSpringConfig { + + private static final Logger log = LoggerFactory.getLogger(RAPSpringConfig.class); + + @Value("${sebserver.gui.entrypoint}") + private String entrypoint; + + @Value("${sebserver.gui.external.messages:messages}") + private String externalMessagesPath; + + @Bean + public ServletContextInitializer initializer() { + return new RAPServletContextInitializer(); + } + + @Bean + public ServletListenerRegistrationBean listenerRegistrationBean() { + final ServletListenerRegistrationBean bean = + new ServletListenerRegistrationBean<>(); + bean.setListener(new RWTServletContextListener()); + return bean; + } + + @Bean + public ServletRegistrationBean servletRegistrationBean() { + return new ServletRegistrationBean<>(new RWTServlet(), this.entrypoint + "/*"); + } + + @Bean + public MessageSource messageSource() { + final ReloadableResourceBundleMessageSource reloadableResourceBundleMessageSource = + new ReloadableResourceBundleMessageSource(); + + log.info(" +++ Register external messages resources from: {}", this.externalMessagesPath); + + reloadableResourceBundleMessageSource.setBasenames( + this.externalMessagesPath, + "classpath:messages"); + + return reloadableResourceBundleMessageSource; + } + + private static class RAPServletContextInitializer implements ServletContextInitializer { + @Override + public void onStartup(final ServletContext servletContext) { + servletContext.setInitParameter( + "org.eclipse.rap.applicationConfiguration", + RAPConfiguration.class.getName()); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/ResourceService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/ResourceService.java index 77ab923b..62da1623 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/ResourceService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/ResourceService.java @@ -82,8 +82,8 @@ public class ResourceService { private static final String MISSING_CLIENT_PING_NAME_KEY = "MISSING"; - public static final Comparator> RESOURCE_COMPARATOR = (t1, t2) -> t1._2.compareTo(t2._2); - public static final Comparator> RESOURCE_COMPARATOR_TUPLE_3 = (t1, t2) -> t1._2.compareTo(t2._2); + public static final Comparator> RESOURCE_COMPARATOR = Comparator.comparing(t -> t._2); + public static final Comparator> RESOURCE_COMPARATOR_TUPLE_3 = Comparator.comparing(t -> t._2); public static final EnumSet ENTITY_TYPE_EXCLUDE_MAP = EnumSet.of( EntityType.ADDITIONAL_ATTRIBUTES, diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/FieldValidationError.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/FieldValidationError.java index 9bf0a572..71c4c099 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/FieldValidationError.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/FieldValidationError.java @@ -1,50 +1,50 @@ -/* - * 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.service.page; - -import java.util.Collection; - -import ch.ethz.seb.sebserver.gbl.api.APIMessage; -import ch.ethz.seb.sebserver.gbl.util.Utils; - -public class FieldValidationError { - - public final String messageCode; - public final String domainName; - public final String fieldName; - public final String errorType; - public final Collection attributes; - - public FieldValidationError(final APIMessage apiMessage) { - this( - apiMessage.messageCode, - apiMessage.attributes.toArray(new String[apiMessage.attributes.size()])); - } - - public FieldValidationError( - final String messageCode, - final String[] attributes) { - - this.messageCode = messageCode; - this.attributes = Utils.immutableCollectionOf(attributes); - - this.domainName = (attributes != null && attributes.length > 0) ? attributes[0] : null; - this.fieldName = (attributes != null && attributes.length > 1) ? attributes[1] : null; - this.errorType = (attributes != null && attributes.length > 2) ? attributes[2] : null; - } - - public String[] getAttributes() { - if (this.attributes == null) { - return new String[0]; - } - - return this.attributes.toArray(new String[this.attributes.size()]); - } - -} +/* + * 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.service.page; + +import java.util.Collection; + +import ch.ethz.seb.sebserver.gbl.api.APIMessage; +import ch.ethz.seb.sebserver.gbl.util.Utils; + +public class FieldValidationError { + + public final String messageCode; + public final String domainName; + public final String fieldName; + public final String errorType; + public final Collection attributes; + + public FieldValidationError(final APIMessage apiMessage) { + this( + apiMessage.messageCode, + apiMessage.attributes.toArray(new String[0])); + } + + public FieldValidationError( + final String messageCode, + final String[] attributes) { + + this.messageCode = messageCode; + this.attributes = Utils.immutableCollectionOf(attributes); + + this.domainName = (attributes != null && attributes.length > 0) ? attributes[0] : null; + this.fieldName = (attributes != null && attributes.length > 1) ? attributes[1] : null; + this.errorType = (attributes != null && attributes.length > 2) ? attributes[2] : null; + } + + public String[] getAttributes() { + if (this.attributes == null) { + return new String[0]; + } + + return this.attributes.toArray(new String[0]); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageContext.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageContext.java index 15f3b616..7bf1367c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageContext.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageContext.java @@ -1,292 +1,292 @@ -/* - * Copyright (c) 2019 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.service.page; - -import java.util.function.Consumer; - -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Shell; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.model.EntityKey; -import ch.ethz.seb.sebserver.gui.service.ResourceService; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; - -/** Holds a page-context and defines some convenient functionality for page handling */ -public interface PageContext { - - Logger log = LoggerFactory.getLogger(PageContext.class); - - /** Defines attribute keys that can be used to store attribute values within the page context state */ - public interface AttributeKeys { - - public static final String PAGE_TEMPLATE_COMPOSER_NAME = "ATTR_PAGE_TEMPLATE_COMPOSER_NAME"; - - public static final String READ_ONLY = "READ_ONLY"; - public static final String READ_ONLY_FROM = "READ_ONLY_FROM"; - - public static final String ENTITY_ID = "ENTITY_ID"; - public static final String PARENT_ENTITY_ID = "PARENT_ENTITY_ID"; - public static final String ENTITY_TYPE = "ENTITY_TYPE"; - public static final String PARENT_ENTITY_TYPE = "PARENT_ENTITY_TYPE"; - - public static final String IMPORT_FROM_QUIZ_DATA = "IMPORT_FROM_QUIZ_DATA"; - - public static final String COPY_AS_TEMPLATE = "COPY_AS_TEMPLATE"; - public static final String CREATE_FROM_TEMPLATE = "CREATE_FROM_TEMPLATE"; - - } - - /** The resource-bundle key of the generic load entity error message. */ - public static final String GENERIC_LOAD_ERROR_TEXT_KEY = "sebserver.error.get.entity"; - public static final String GENERIC_REMOVE_ERROR_TEXT_KEY = "sebserver.error.remove.entity"; - public static final String GENERIC_SAVE_ERROR_TEXT_KEY = "sebserver.error.save.entity"; - public static final String GENERIC_ACTIVATE_ERROR_TEXT_KEY = "sebserver.error.activate.entity"; - public static final String GENERIC_IMPORT_ERROR_TEXT_KEY = "sebserver.error.import"; - public static final LocTextKey SUCCESS_MSG_TITLE = - new LocTextKey("sebserver.page.message"); - public static final LocTextKey UNEXPECTED_ERROR_KEY = - new LocTextKey("sebserver.error.action.unexpected.message"); - - /** Get the I18nSupport service - * - * @return the I18nSupport service */ - I18nSupport getI18nSupport(); - - /** Use this to get the ComposerService used by this PageContext - * - * @return the ComposerService used by this PageContext */ - ComposerService composerService(); - - /** Get the RWT Shell that is bound within this PageContext - * - * @return the RWT Shell that is bound within this PageContext */ - Shell getShell(); - - /** Get the page root Component. - * - * @return the page root Component. */ - Composite getRoot(); - - /** Get the Component that is currently set as parent during page tree compose - * - * @return the parent Component */ - Composite getParent(); - - /** Get a copy of this PageContext. - * - * @return */ - PageContext copy(); - - /** Create a copy of this PageContext with a new parent Composite. - * The implementation should take care of the imutability of PageContext and return a copy with the new parent - * by leave this PageContext as is. - * - * @param parent the new parent Composite - * @return a copy of this PageContext with a new parent Composite. */ - PageContext copyOf(Composite parent); - - /** Create a copy of this PageContext with and additionally page context attributes. - * The additionally page context attributes will get merged with them already defined - * The implementation should take care of the imutability of PageContext and return a copy with the merge - * by leave this and the given PageContext as is. - * - * @param attributes additionally page context attributes. - * @return a copy of this PageContext with with and additionally page context attributes. */ - PageContext copyOfAttributes(PageContext otherContext); - - /** Adds the specified attribute to the existing page context attributes. - * The implementation should take care of the imutability of PageContext and return a copy - * by leave this PageContext as is. - * - * @param key the key of the attribute - * @param value the value of the attribute - * @return this PageContext instance (builder pattern) */ - PageContext withAttribute(String key, String value); - - /** Gets a copy of this PageContext with cleared attribute map. - * - * @return a copy of this PageContext with cleared attribute map. */ - PageContext clearAttributes(); - - /** Get the attribute value that is mapped to the given name or null of no mapping exists - * - * @param name the attribute name - * @return the attribute value that is mapped to the given name or null if no mapping exists */ - String getAttribute(String name); - - /** Get the attribute value that is mapped to the given name or a default value if no mapping exists - * - * @param name the attribute name - * @param def the default value - * @return the attribute value that is mapped to the given name or null of no mapping exists */ - String getAttribute(String name, String def); - - /** Indicates if the attribute with the key READ_ONLY is set to true within this PageContext - * - * @return true if the attribute with the key READ_ONLY is set to true */ - boolean isReadonly(); - - /** Gets an EntityKey for the base Entity that is associated within this PageContext by using - * the attribute keys ENTITY_ID and ENTITY_TYPE to fetch the attribute values for an EntityKey - * - * @return the EntityKey of the base Entity that is associated within this PageContext */ - EntityKey getEntityKey(); - - /** Gets an EntityKey for the parent Entity that is associated within this PageContext by using - * the attribute keys PARENT_ENTITY_ID and PARENT_ENTITY_TYPE to fetch the attribute values for an EntityKey - * - * @return the EntityKey of the parent Entity that is associated within this PageContext */ - EntityKey getParentEntityKey(); - - /** Adds a given EntityKey as base Entity key to a new PageContext that is returned as a copy of this PageContext. - * - * @param entityKey the EntityKey to add as base Entity key - * @return the new PageContext with the EntityKey added */ - PageContext withEntityKey(EntityKey entityKey); - - /** Adds a given EntityKey as parent Entity key to a new PageContext that is returned as a copy of this PageContext. - * - * @param entityKey the EntityKey to add as parent Entity key - * @return the new PageContext with the EntityKey added */ - PageContext withParentEntityKey(EntityKey entityKey); - - /** Create a copy of this PageContext and resets both entity keys attributes, the base and the parent EntityKey - * - * @return copy of this PageContext with reseted EntityKey attributes (base and parent) */ - PageContext clearEntityKeys(); - - /** Indicates if an attribute with the specified name exists within this PageContext - * - * @param name the name of the attribute - * @return true if the attribute with the specified name exists within this PageContext */ - boolean hasAttribute(String name); - - /** Returns a new PageContext with the removed attribute by name - * - * @param name the name of the attribute to remove - * @return a copy of this PageContext with the removed attribute */ - PageContext removeAttribute(String name); - - /** Apply a confirm dialog with a specified confirm message and a callback code - * block that will be executed on users OK selection. - * - * @param confirmMessage the localized confirm message key - * @param onOK callback code block that will be called on users selection */ - void applyConfirmDialog(LocTextKey confirmMessage, final Consumer callback); - - /** This can be used to forward to a defined page. - * - * @param pageDefinition the defined page */ - void forwardToPage(PageDefinition pageDefinition); - - /** Forward to main page */ - void forwardToMainPage(); - - /** Forward to login page */ - void forwardToLoginPage(); - - /** Notify an error dialog to the user with specified error message and - * optional exception instance - * - * @param errorMessage the error message to display - * @param error the error as Exception */ - void notifyError(LocTextKey errorMessage, Exception error); - - /** Notify a generic load error to the user by pop-up - * - * @param entityType the type of the entity - * @param error the original error */ - default void notifyLoadError(final EntityType entityType, final Exception error) { - notifyError( - new LocTextKey( - GENERIC_LOAD_ERROR_TEXT_KEY, - getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), - error); - } - - /** Notify a generic remove error to the user by pop-up - * - * @param entityType the type of the entity - * @param error the original error */ - default void notifyRemoveError(final EntityType entityType, final Exception error) { - notifyError( - new LocTextKey( - GENERIC_REMOVE_ERROR_TEXT_KEY, - getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), - error); - } - - /** Notify a generic save error to the user by pop-up - * - * @param entityType the type of the entity - * @param error the original error */ - default void notifySaveError(final EntityType entityType, final Exception error) { - notifyError( - new LocTextKey( - GENERIC_SAVE_ERROR_TEXT_KEY, - getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), - error); - } - - /** Notify a generic activation error to the user by pop-up - * - * @param entityType the type of the entity - * @param error the original error */ - default void notifyActivationError(final EntityType entityType, final Exception error) { - notifyError( - new LocTextKey( - GENERIC_ACTIVATE_ERROR_TEXT_KEY, - getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), - error); - } - - /** Notify a generic import error to the user by pop-up - * - * @param entityType the type of the entity - * @param error the original error */ - default void notifyImportError(final EntityType entityType, final Exception error) { - notifyError( - new LocTextKey( - GENERIC_IMPORT_ERROR_TEXT_KEY, - getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), - error); - } - - /** Notify a generic unexpected error to the user by pop-up - * - * @param error the original error */ - default void notifyUnexpectedError(final Exception error) { - notifyError(UNEXPECTED_ERROR_KEY, error); - } - - /** Publish and shows a message to the user with the given localized title and - * localized message. The message text can also be HTML text as far as RWT supports it. - * - * @param title the localized text key of the title message - * @param message the localized text key of the message */ - void publishPageMessage(LocTextKey title, LocTextKey message); - - /** Publish an information message to the user with the given localized message. - * The message text can also be HTML text as far as RWT supports it - * - * @param message the localized text key of the message */ - default void publishInfo(final LocTextKey message) { - publishPageMessage(new LocTextKey("sebserver.page.message"), message); - } - - /** Publish and shows a formatted PageMessageException to the user. - * - * @param pme the PageMessageException */ - void publishPageMessage(PageMessageException pme); - -} +/* + * Copyright (c) 2019 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.service.page; + +import java.util.function.Consumer; + +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Shell; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.model.EntityKey; +import ch.ethz.seb.sebserver.gui.service.ResourceService; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; + +/** Holds a page-context and defines some convenient functionality for page handling */ +public interface PageContext { + + Logger log = LoggerFactory.getLogger(PageContext.class); + + /** Defines attribute keys that can be used to store attribute values within the page context state */ + interface AttributeKeys { + + String PAGE_TEMPLATE_COMPOSER_NAME = "ATTR_PAGE_TEMPLATE_COMPOSER_NAME"; + + String READ_ONLY = "READ_ONLY"; + String READ_ONLY_FROM = "READ_ONLY_FROM"; + + String ENTITY_ID = "ENTITY_ID"; + String PARENT_ENTITY_ID = "PARENT_ENTITY_ID"; + String ENTITY_TYPE = "ENTITY_TYPE"; + String PARENT_ENTITY_TYPE = "PARENT_ENTITY_TYPE"; + + String IMPORT_FROM_QUIZ_DATA = "IMPORT_FROM_QUIZ_DATA"; + + String COPY_AS_TEMPLATE = "COPY_AS_TEMPLATE"; + String CREATE_FROM_TEMPLATE = "CREATE_FROM_TEMPLATE"; + + } + + /** The resource-bundle key of the generic load entity error message. */ + String GENERIC_LOAD_ERROR_TEXT_KEY = "sebserver.error.get.entity"; + String GENERIC_REMOVE_ERROR_TEXT_KEY = "sebserver.error.remove.entity"; + String GENERIC_SAVE_ERROR_TEXT_KEY = "sebserver.error.save.entity"; + String GENERIC_ACTIVATE_ERROR_TEXT_KEY = "sebserver.error.activate.entity"; + String GENERIC_IMPORT_ERROR_TEXT_KEY = "sebserver.error.import"; + LocTextKey SUCCESS_MSG_TITLE = + new LocTextKey("sebserver.page.message"); + LocTextKey UNEXPECTED_ERROR_KEY = + new LocTextKey("sebserver.error.action.unexpected.message"); + + /** Get the I18nSupport service + * + * @return the I18nSupport service */ + I18nSupport getI18nSupport(); + + /** Use this to get the ComposerService used by this PageContext + * + * @return the ComposerService used by this PageContext */ + ComposerService composerService(); + + /** Get the RWT Shell that is bound within this PageContext + * + * @return the RWT Shell that is bound within this PageContext */ + Shell getShell(); + + /** Get the page root Component. + * + * @return the page root Component. */ + Composite getRoot(); + + /** Get the Component that is currently set as parent during page tree compose + * + * @return the parent Component */ + Composite getParent(); + + /** Get a copy of this PageContext. + * + * @return a deep copy of this PageContext */ + PageContext copy(); + + /** Create a copy of this PageContext with a new parent Composite. + * The implementation should take care of the immutability of PageContext and return a copy with the new parent + * by leave this PageContext as is. + * + * @param parent the new parent Composite + * @return a copy of this PageContext with a new parent Composite. */ + PageContext copyOf(Composite parent); + + /** Create a copy of this PageContext with and additionally page context attributes. + * The additionally page context attributes will get merged with them already defined + * The implementation should take care of the immutability of PageContext and return a copy with the merge + * by leave this and the given PageContext as is. + * + * @param otherContext the other PageContext to copy the attributes from + * @return a copy of this PageContext with with and additionally page context attributes. */ + PageContext copyOfAttributes(PageContext otherContext); + + /** Adds the specified attribute to the existing page context attributes. + * The implementation should take care of the immutability of PageContext and return a copy + * by leave this PageContext as is. + * + * @param key the key of the attribute + * @param value the value of the attribute + * @return this PageContext instance (builder pattern) */ + PageContext withAttribute(String key, String value); + + /** Gets a copy of this PageContext with cleared attribute map. + * + * @return a copy of this PageContext with cleared attribute map. */ + PageContext clearAttributes(); + + /** Get the attribute value that is mapped to the given name or null of no mapping exists + * + * @param name the attribute name + * @return the attribute value that is mapped to the given name or null if no mapping exists */ + String getAttribute(String name); + + /** Get the attribute value that is mapped to the given name or a default value if no mapping exists + * + * @param name the attribute name + * @param def the default value + * @return the attribute value that is mapped to the given name or null of no mapping exists */ + String getAttribute(String name, String def); + + /** Indicates if the attribute with the key READ_ONLY is set to true within this PageContext + * + * @return true if the attribute with the key READ_ONLY is set to true */ + boolean isReadonly(); + + /** Gets an EntityKey for the base Entity that is associated within this PageContext by using + * the attribute keys ENTITY_ID and ENTITY_TYPE to fetch the attribute values for an EntityKey + * + * @return the EntityKey of the base Entity that is associated within this PageContext */ + EntityKey getEntityKey(); + + /** Gets an EntityKey for the parent Entity that is associated within this PageContext by using + * the attribute keys PARENT_ENTITY_ID and PARENT_ENTITY_TYPE to fetch the attribute values for an EntityKey + * + * @return the EntityKey of the parent Entity that is associated within this PageContext */ + EntityKey getParentEntityKey(); + + /** Adds a given EntityKey as base Entity key to a new PageContext that is returned as a copy of this PageContext. + * + * @param entityKey the EntityKey to add as base Entity key + * @return the new PageContext with the EntityKey added */ + PageContext withEntityKey(EntityKey entityKey); + + /** Adds a given EntityKey as parent Entity key to a new PageContext that is returned as a copy of this PageContext. + * + * @param entityKey the EntityKey to add as parent Entity key + * @return the new PageContext with the EntityKey added */ + PageContext withParentEntityKey(EntityKey entityKey); + + /** Create a copy of this PageContext and resets both entity keys attributes, the base and the parent EntityKey + * + * @return copy of this PageContext with reset EntityKey attributes (base and parent) */ + PageContext clearEntityKeys(); + + /** Indicates if an attribute with the specified name exists within this PageContext + * + * @param name the name of the attribute + * @return true if the attribute with the specified name exists within this PageContext */ + boolean hasAttribute(String name); + + /** Returns a new PageContext with the removed attribute by name + * + * @param name the name of the attribute to remove + * @return a copy of this PageContext with the removed attribute */ + PageContext removeAttribute(String name); + + /** Apply a confirm dialog with a specified confirm message and a callback code + * block that will be executed on users OK selection. + * + * @param confirmMessage the localized confirm message key + * @param callback callback code block that will be called on users selection */ + void applyConfirmDialog(LocTextKey confirmMessage, final Consumer callback); + + /** This can be used to forward to a defined page. + * + * @param pageDefinition the defined page */ + void forwardToPage(PageDefinition pageDefinition); + + /** Forward to main page */ + void forwardToMainPage(); + + /** Forward to login page */ + void forwardToLoginPage(); + + /** Notify an error dialog to the user with specified error message and + * optional exception instance + * + * @param errorMessage the error message to display + * @param error the error as Exception */ + void notifyError(LocTextKey errorMessage, Exception error); + + /** Notify a generic load error to the user by pop-up + * + * @param entityType the type of the entity + * @param error the original error */ + default void notifyLoadError(final EntityType entityType, final Exception error) { + notifyError( + new LocTextKey( + GENERIC_LOAD_ERROR_TEXT_KEY, + getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), + error); + } + + /** Notify a generic remove error to the user by pop-up + * + * @param entityType the type of the entity + * @param error the original error */ + default void notifyRemoveError(final EntityType entityType, final Exception error) { + notifyError( + new LocTextKey( + GENERIC_REMOVE_ERROR_TEXT_KEY, + getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), + error); + } + + /** Notify a generic save error to the user by pop-up + * + * @param entityType the type of the entity + * @param error the original error */ + default void notifySaveError(final EntityType entityType, final Exception error) { + notifyError( + new LocTextKey( + GENERIC_SAVE_ERROR_TEXT_KEY, + getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), + error); + } + + /** Notify a generic activation error to the user by pop-up + * + * @param entityType the type of the entity + * @param error the original error */ + default void notifyActivationError(final EntityType entityType, final Exception error) { + notifyError( + new LocTextKey( + GENERIC_ACTIVATE_ERROR_TEXT_KEY, + getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), + error); + } + + /** Notify a generic import error to the user by pop-up + * + * @param entityType the type of the entity + * @param error the original error */ + default void notifyImportError(final EntityType entityType, final Exception error) { + notifyError( + new LocTextKey( + GENERIC_IMPORT_ERROR_TEXT_KEY, + getI18nSupport().getText(ResourceService.getEntityTypeNameKey(entityType))), + error); + } + + /** Notify a generic unexpected error to the user by pop-up + * + * @param error the original error */ + default void notifyUnexpectedError(final Exception error) { + notifyError(UNEXPECTED_ERROR_KEY, error); + } + + /** Publish and shows a message to the user with the given localized title and + * localized message. The message text can also be HTML text as far as RWT supports it. + * + * @param title the localized text key of the title message + * @param message the localized text key of the message */ + void publishPageMessage(LocTextKey title, LocTextKey message); + + /** Publish an information message to the user with the given localized message. + * The message text can also be HTML text as far as RWT supports it + * + * @param message the localized text key of the message */ + default void publishInfo(final LocTextKey message) { + publishPageMessage(new LocTextKey("sebserver.page.message"), message); + } + + /** Publish and shows a formatted PageMessageException to the user. + * + * @param pme the PageMessageException */ + void publishPageMessage(PageMessageException pme); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageDefinition.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageDefinition.java index a5bfa567..4629a336 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageDefinition.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageDefinition.java @@ -1,16 +1,20 @@ -/* - * Copyright (c) 2019 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.service.page; - -public interface PageDefinition { - - Class composer(); - - PageContext applyPageContext(PageContext pageContext); -} +/* + * Copyright (c) 2019 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.service.page; + +/** Defines a global SEB Server page */ +public interface PageDefinition { + + /** Get the type class of the TemplateComposer that composes the page. + * + * @return the type class of the TemplateComposer that composes the page. */ + Class composer(); + + PageContext applyPageContext(PageContext pageContext); +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageService.java index e21a7a08..f981ebfc 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageService.java @@ -363,7 +363,7 @@ public interface PageService { return content; } - /** Used to update the crolledComposite when some if its content has dynamically changed + /** Used to update the scrolledComposite when some if its content has dynamically changed * its dimensions. * * @param composite The Component that changed its dimensions */ diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageStateDefinition.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageStateDefinition.java index 638106d0..fb6bea2b 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageStateDefinition.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/PageStateDefinition.java @@ -1,30 +1,30 @@ -/* - * Copyright (c) 2019 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.service.page; - -public interface PageStateDefinition { - - enum Type { - UNDEFINED, - LIST_VIEW, - FORM_VIEW, - FORM_EDIT, - FORM_IN_TIME_EDIT - } - - String name(); - - Type type(); - - public Class contentPaneComposer(); - - public Class actionPaneComposer(); - - Activity activityAnchor(); -} +/* + * Copyright (c) 2019 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.service.page; + +public interface PageStateDefinition { + + enum Type { + UNDEFINED, + LIST_VIEW, + FORM_VIEW, + FORM_EDIT, + FORM_IN_TIME_EDIT + } + + String name(); + + Type type(); + + Class contentPaneComposer(); + + Class actionPaneComposer(); + + Activity activityAnchor(); +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/TemplateComposer.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/TemplateComposer.java index 537dce28..e1f21d2b 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/TemplateComposer.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/TemplateComposer.java @@ -1,19 +1,27 @@ -/* - * 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.service.page; - -public interface TemplateComposer { - - default boolean validate(final PageContext pageContext) { - return true; - } - - void compose(PageContext pageContext); - -} +/* + * 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.service.page; + +/** interface defining a RAP page template composer */ +public interface TemplateComposer { + + /** Validate given PageContext for completeness to compose a specific TemplateComposer implementation + * Default returns always true. + * @param pageContext The PageContext instance to check + * @return true if the PageContext contains all mandatory data to compose this page template */ + default boolean validate(final PageContext pageContext) { + return true; + } + + /** Compose a specific page template for the given PageContext + * + * @param pageContext The PageContext instance */ + void compose(PageContext pageContext); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/DefaultPageLayout.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/DefaultPageLayout.java index 44714149..dabe4842 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/DefaultPageLayout.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/DefaultPageLayout.java @@ -1,366 +1,362 @@ -/* - * 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.service.page.impl; - -import java.io.ByteArrayInputStream; -import java.nio.charset.StandardCharsets; - -import org.apache.commons.codec.binary.Base64InputStream; -import org.apache.commons.lang3.BooleanUtils; -import org.apache.commons.lang3.StringUtils; -import org.eclipse.rap.rwt.RWT; -import org.eclipse.rap.rwt.client.service.UrlLauncher; -import org.eclipse.swt.SWT; -import org.eclipse.swt.graphics.Image; -import org.eclipse.swt.graphics.ImageData; -import org.eclipse.swt.graphics.Rectangle; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.layout.RowLayout; -import org.eclipse.swt.widgets.Button; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Display; -import org.eclipse.swt.widgets.Label; -import org.eclipse.swt.widgets.MessageBox; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.core.env.Environment; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.i18n.PolyglotPageService; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.service.page.PageContext.AttributeKeys; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.service.page.TemplateComposer; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.AuthorizationContextHolder; -import ch.ethz.seb.sebserver.gui.widget.Message; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; - -@Lazy -@Component -public class DefaultPageLayout implements TemplateComposer { - - private static final Logger log = LoggerFactory.getLogger(DefaultPageLayout.class); - - private static final LocTextKey ABOUT_TEXT_KEY = new LocTextKey("sebserver.overall.about"); - private static final LocTextKey IMPRINT_TEXT_KEY = new LocTextKey("sebserver.overall.imprint"); - private static final LocTextKey HELP_TEXT_KEY = new LocTextKey("sebserver.overall.help"); - - private static final LocTextKey ABOUT_MARKUP_TEXT_KEY = new LocTextKey("sebserver.overall.about.markup"); - private static final LocTextKey IMPRINT_MARKUP_TEXT_KEY = new LocTextKey("sebserver.overall.imprint.markup"); - private static final LocTextKey HELP_LINK_TEXT_KEY = new LocTextKey("sebserver.overall.help.link"); - - public static final int LOGO_IMAGE_MAX_WIDTH = 400; - public static final int LOGO_IMAGE_MAX_HEIGHT = 80; - - private final WidgetFactory widgetFactory; - private final PolyglotPageService polyglotPageService; - private final AuthorizationContextHolder authorizationContextHolder; - private final PageService pageService; - private final String sebServerVersion; - private final boolean multilingual; - - public DefaultPageLayout( - final PageService pageService, - final Environment environment) { - - this.widgetFactory = pageService.getWidgetFactory(); - this.polyglotPageService = pageService.getPolyglotPageService(); - this.authorizationContextHolder = pageService.getAuthorizationContextHolder(); - this.pageService = pageService; - this.sebServerVersion = environment.getProperty("sebserver.version", Constants.EMPTY_NOTE); - this.multilingual = BooleanUtils.toBoolean(environment.getProperty("sebserver.gui.multilingual", "false")); - } - - @Override - public boolean validate(final PageContext pageContext) { - return pageContext.hasAttribute(AttributeKeys.PAGE_TEMPLATE_COMPOSER_NAME); - } - - @Override - public void compose(final PageContext pageContext) { - - final GridLayout skeletonLayout = new GridLayout(); - skeletonLayout.marginBottom = 0; - skeletonLayout.marginLeft = 0; - skeletonLayout.marginRight = 0; - skeletonLayout.marginTop = 0; - skeletonLayout.marginHeight = 0; - skeletonLayout.marginWidth = 0; - skeletonLayout.verticalSpacing = 0; - skeletonLayout.horizontalSpacing = 0; - pageContext.getParent().setLayout(skeletonLayout); - - composeHeader(pageContext); - composeLogoBar(pageContext); - composeContent(pageContext); - composeFooter(pageContext); - - this.polyglotPageService.setDefaultPageLocale(pageContext.getRoot()); - } - - private void composeHeader(final PageContext pageContext) { - final Composite header = new Composite(pageContext.getParent(), SWT.NONE); - final GridLayout gridLayout = new GridLayout(); - gridLayout.marginRight = 50; - gridLayout.marginLeft = 50; - header.setLayout(gridLayout); - final GridData headerCell = new GridData(SWT.FILL, SWT.TOP, true, false); - headerCell.minimumHeight = 40; - headerCell.heightHint = 40; - header.setLayoutData(headerCell); - header.setData(RWT.CUSTOM_VARIANT, "header"); - - final Composite headerRight = new Composite(header, SWT.NONE); - headerRight.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); - final GridLayout headerRightGrid = new GridLayout(2, false); - headerRightGrid.marginHeight = 0; - headerRightGrid.marginWidth = 0; - headerRightGrid.horizontalSpacing = 20; - headerRight.setLayout(headerRightGrid); - headerRight.setData(RWT.CUSTOM_VARIANT, "header"); - - if (this.authorizationContextHolder.getAuthorizationContext().isLoggedIn()) { - final Label username = new Label(headerRight, SWT.NONE); - username.setData(RWT.CUSTOM_VARIANT, "header"); - username.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); - username.setText(this.authorizationContextHolder - .getAuthorizationContext() - .getLoggedInUser() - .get(t -> this.pageService.logoutOnError(t, pageContext)).username); - - final Button logout = this.widgetFactory.buttonLocalized(headerRight, "sebserver.logout"); - logout.setLayoutData(new GridData(SWT.RIGHT, SWT.FILL, true, true)); - logout.setData(RWT.CUSTOM_VARIANT, "header"); - logout.addListener(SWT.Selection, event -> { - this.pageService.logout(pageContext); - // show successful logout message - final MessageBox logoutSuccess = new Message( - pageContext.getShell(), - this.polyglotPageService.getI18nSupport().getText("sebserver.logout"), - this.polyglotPageService.getI18nSupport().getText("sebserver.logout.success.message"), - SWT.ICON_INFORMATION, - pageContext.getI18nSupport()); - logoutSuccess.open(null); - - // TODO Try to invalidate RWT's user session. - // This seems to be more difficult then expected and just invalidate the HttpSession dosn't work - // Try to send a redirect JSON to the client: https://bugs.eclipse.org/bugs/show_bug.cgi?id=388249 - }); - } - } - - private void composeLogoBar(final PageContext pageContext) { - final Composite logoBar = new Composite(pageContext.getParent(), SWT.NONE); - final GridData logoBarCell = new GridData(SWT.FILL, SWT.TOP, false, false); - logoBarCell.minimumHeight = 80; - logoBarCell.heightHint = 80; - logoBar.setLayoutData(logoBarCell); - logoBar.setData(RWT.CUSTOM_VARIANT, "logo"); - final GridLayout logoBarLayout = new GridLayout(2, false); - logoBarLayout.horizontalSpacing = 0; - logoBarLayout.marginHeight = 0; - logoBar.setLayout(logoBarLayout); - - final Composite logo = new Composite(logoBar, SWT.NONE); - final GridData logoCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); - logoCell.minimumHeight = LOGO_IMAGE_MAX_HEIGHT; - logoCell.heightHint = LOGO_IMAGE_MAX_HEIGHT; - logoCell.minimumWidth = LOGO_IMAGE_MAX_WIDTH; - logoCell.horizontalIndent = 50; - logo.setLayoutData(logoCell); - - // try to get institutional logo first. If no success, use default logo - loadInstitutionalLogo(pageContext, logo); - - final Composite langSupport = new Composite(logoBar, SWT.NONE); - final GridData langSupportCell = new GridData(SWT.RIGHT, SWT.CENTER, false, false); - langSupportCell.heightHint = 20; - logoCell.horizontalIndent = 50; - langSupport.setLayoutData(langSupportCell); - langSupport.setData(RWT.CUSTOM_VARIANT, "logo"); - final RowLayout rowLayout = new RowLayout(SWT.HORIZONTAL); - rowLayout.spacing = 7; - rowLayout.marginRight = 70; - langSupport.setLayout(rowLayout); - - if (this.multilingual) { - this.polyglotPageService.createLanguageSelector(pageContext.copyOf(langSupport)); - } - } - - private void composeContent(final PageContext pageContext) { - final Composite contentBackground = new Composite(pageContext.getParent(), SWT.NONE); - contentBackground.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); - contentBackground.setData(RWT.CUSTOM_VARIANT, "bgContent"); - final GridLayout innerGrid = new GridLayout(); - innerGrid.marginLeft = 50; - innerGrid.marginRight = 50; - innerGrid.marginHeight = 0; - innerGrid.marginWidth = 0; - - contentBackground.setLayout(innerGrid); - - final Composite content = new Composite(contentBackground, SWT.NONE); - content.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); - content.setData(RWT.CUSTOM_VARIANT, "content"); - final GridLayout contentGrid = new GridLayout(); - contentGrid.marginHeight = 0; - contentGrid.marginWidth = 0; - content.setLayout(contentGrid); - - final Composite contentInner = new Composite(content, SWT.NONE); - contentInner.setLayoutData(new GridData(SWT.CENTER, SWT.CENTER, true, true)); - final GridLayout gridLayout = new GridLayout(); - gridLayout.marginHeight = 0; - gridLayout.marginWidth = 0; - contentInner.setLayout(gridLayout); - - final String contentComposerName = pageContext.getAttribute( - AttributeKeys.PAGE_TEMPLATE_COMPOSER_NAME); - pageContext.composerService().compose( - contentComposerName, - pageContext.copyOf(contentInner)); - } - - private void composeFooter(final PageContext pageContext) { - final Composite footerBar = new Composite(pageContext.getParent(), SWT.NONE); - final GridData footerCell = new GridData(SWT.FILL, SWT.BOTTOM, false, false); - footerCell.minimumHeight = 30; - footerCell.heightHint = 30; - footerBar.setLayoutData(footerCell); - footerBar.setData(RWT.CUSTOM_VARIANT, "bgFooter"); - final GridLayout innerBarGrid = new GridLayout(); - innerBarGrid.marginHeight = 0; - innerBarGrid.marginWidth = 0; - innerBarGrid.marginLeft = 50; - innerBarGrid.marginRight = 50; - footerBar.setLayout(innerBarGrid); - - final Composite footer = new Composite(footerBar, SWT.NONE); - final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); - footer.setLayoutData(gridData); - final GridLayout footerGrid = new GridLayout(2, false); - footerGrid.marginHeight = 0; - footerGrid.marginWidth = 0; - footerGrid.horizontalSpacing = 0; - footer.setLayout(footerGrid); - footer.setData(RWT.CUSTOM_VARIANT, "footer"); - - final Composite footerLeft = new Composite(footer, SWT.NONE); - footerLeft.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, true)); - footerLeft.setData(RWT.CUSTOM_VARIANT, "footer"); - RowLayout rowLayout = new RowLayout(SWT.HORIZONTAL); - rowLayout.marginLeft = 20; - rowLayout.spacing = 20; - footerLeft.setLayout(rowLayout); - - final Composite footerRight = new Composite(footer, SWT.NONE); - footerRight.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); - footerRight.setData(RWT.CUSTOM_VARIANT, "footer"); - rowLayout = new RowLayout(SWT.HORIZONTAL); - rowLayout.marginRight = 20; - footerRight.setLayout(rowLayout); - - final I18nSupport i18nSupport = this.widgetFactory.getI18nSupport(); - if (StringUtils.isNoneBlank(i18nSupport.getText(IMPRINT_TEXT_KEY, ""))) { - final Label imprint = this.widgetFactory.labelLocalized( - footerLeft, - CustomVariant.FOOTER, - IMPRINT_TEXT_KEY); - - imprint.addListener(SWT.MouseUp, event -> { - try { - pageContext.publishPageMessage(IMPRINT_TEXT_KEY, IMPRINT_MARKUP_TEXT_KEY); - } catch (final Exception e) { - log.error("Invalid markup for 'Imprint'", e); - } - }); - } - if (StringUtils.isNoneBlank(i18nSupport.getText(ABOUT_TEXT_KEY, ""))) { - final Label about = this.widgetFactory.labelLocalized( - footerLeft, - CustomVariant.FOOTER, - ABOUT_TEXT_KEY); - - about.addListener(SWT.MouseUp, event -> { - try { - pageContext.publishPageMessage(ABOUT_TEXT_KEY, ABOUT_MARKUP_TEXT_KEY); - } catch (final Exception e) { - log.error("Invalid markup for 'About'", e); - } - }); - } - if (StringUtils.isNoneBlank(i18nSupport.getText(HELP_TEXT_KEY, ""))) { - final Label help = this.widgetFactory.labelLocalized( - footerLeft, - CustomVariant.FOOTER, - HELP_TEXT_KEY); - - help.addListener(SWT.MouseUp, event -> { - try { - final String link = i18nSupport.getText(HELP_LINK_TEXT_KEY, ""); - if (StringUtils.isNoneBlank(link)) { - final UrlLauncher urlLauncher = RWT.getClient().getService(UrlLauncher.class); - urlLauncher.openURL(link); - } - } catch (final Exception e) { - log.error("Invalid Help link", e); - } - }); - - } - this.widgetFactory.labelLocalized( - footerRight, - CustomVariant.FOOTER, - new LocTextKey("sebserver.overall.version", this.sebServerVersion)); - } - - private void loadInstitutionalLogo(final PageContext pageContext, final Composite logo) { - logo.setData(RWT.CUSTOM_VARIANT, "bgLogo"); - try { - - final String imageBase64 = (String) RWT.getUISession() - .getHttpSession() - .getAttribute(API.PARAM_LOGO_IMAGE); - - if (StringUtils.isBlank(imageBase64)) { - return; - } - - final Base64InputStream input = new Base64InputStream( - new ByteArrayInputStream(imageBase64.getBytes(StandardCharsets.UTF_8)), - false); - - final Display display = pageContext.getShell().getDisplay(); - final Image image = new Image(display, input); - final Rectangle imageBounds = image.getBounds(); - final int width = (imageBounds.width > LOGO_IMAGE_MAX_WIDTH) - ? LOGO_IMAGE_MAX_WIDTH - : imageBounds.width; - final int height = (imageBounds.height > LOGO_IMAGE_MAX_HEIGHT) - ? LOGO_IMAGE_MAX_HEIGHT - : imageBounds.height; - final ImageData imageData = image.getImageData().scaledTo(width, height); - - logo.setData(RWT.CUSTOM_VARIANT, "bgLogoNoImage"); - logo.setBackgroundImage(new Image(display, imageData)); - - } catch (final Exception e) { - log.warn("Get institutional logo failed: {}", e.getMessage()); - } - } - -} +/* + * 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.service.page.impl; + +import java.io.ByteArrayInputStream; +import java.nio.charset.StandardCharsets; + +import org.apache.commons.codec.binary.Base64InputStream; +import org.apache.commons.lang3.BooleanUtils; +import org.apache.commons.lang3.StringUtils; +import org.eclipse.rap.rwt.RWT; +import org.eclipse.rap.rwt.client.service.UrlLauncher; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.graphics.ImageData; +import org.eclipse.swt.graphics.Rectangle; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.layout.RowLayout; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Display; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.MessageBox; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.core.env.Environment; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.i18n.PolyglotPageService; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.service.page.PageContext.AttributeKeys; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.service.page.TemplateComposer; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.AuthorizationContextHolder; +import ch.ethz.seb.sebserver.gui.widget.Message; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; + +@Lazy +@Component +public class DefaultPageLayout implements TemplateComposer { + + private static final Logger log = LoggerFactory.getLogger(DefaultPageLayout.class); + + private static final LocTextKey ABOUT_TEXT_KEY = new LocTextKey("sebserver.overall.about"); + private static final LocTextKey IMPRINT_TEXT_KEY = new LocTextKey("sebserver.overall.imprint"); + private static final LocTextKey HELP_TEXT_KEY = new LocTextKey("sebserver.overall.help"); + + private static final LocTextKey ABOUT_MARKUP_TEXT_KEY = new LocTextKey("sebserver.overall.about.markup"); + private static final LocTextKey IMPRINT_MARKUP_TEXT_KEY = new LocTextKey("sebserver.overall.imprint.markup"); + private static final LocTextKey HELP_LINK_TEXT_KEY = new LocTextKey("sebserver.overall.help.link"); + + public static final int LOGO_IMAGE_MAX_WIDTH = 400; + public static final int LOGO_IMAGE_MAX_HEIGHT = 80; + + private final WidgetFactory widgetFactory; + private final PolyglotPageService polyglotPageService; + private final AuthorizationContextHolder authorizationContextHolder; + private final PageService pageService; + private final String sebServerVersion; + private final boolean multilingual; + + public DefaultPageLayout( + final PageService pageService, + final Environment environment) { + + this.widgetFactory = pageService.getWidgetFactory(); + this.polyglotPageService = pageService.getPolyglotPageService(); + this.authorizationContextHolder = pageService.getAuthorizationContextHolder(); + this.pageService = pageService; + this.sebServerVersion = environment.getProperty("sebserver.version", Constants.EMPTY_NOTE); + this.multilingual = BooleanUtils.toBoolean(environment.getProperty("sebserver.gui.multilingual", "false")); + } + + @Override + public boolean validate(final PageContext pageContext) { + return pageContext.hasAttribute(AttributeKeys.PAGE_TEMPLATE_COMPOSER_NAME); + } + + @Override + public void compose(final PageContext pageContext) { + + final GridLayout skeletonLayout = new GridLayout(); + skeletonLayout.marginBottom = 0; + skeletonLayout.marginLeft = 0; + skeletonLayout.marginRight = 0; + skeletonLayout.marginTop = 0; + skeletonLayout.marginHeight = 0; + skeletonLayout.marginWidth = 0; + skeletonLayout.verticalSpacing = 0; + skeletonLayout.horizontalSpacing = 0; + pageContext.getParent().setLayout(skeletonLayout); + + composeHeader(pageContext); + composeLogoBar(pageContext); + composeContent(pageContext); + composeFooter(pageContext); + + this.polyglotPageService.setDefaultPageLocale(pageContext.getRoot()); + } + + private void composeHeader(final PageContext pageContext) { + final Composite header = new Composite(pageContext.getParent(), SWT.NONE); + final GridLayout gridLayout = new GridLayout(); + gridLayout.marginRight = 50; + gridLayout.marginLeft = 50; + header.setLayout(gridLayout); + final GridData headerCell = new GridData(SWT.FILL, SWT.TOP, true, false); + headerCell.minimumHeight = 40; + headerCell.heightHint = 40; + header.setLayoutData(headerCell); + header.setData(RWT.CUSTOM_VARIANT, "header"); + + final Composite headerRight = new Composite(header, SWT.NONE); + headerRight.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); + final GridLayout headerRightGrid = new GridLayout(2, false); + headerRightGrid.marginHeight = 0; + headerRightGrid.marginWidth = 0; + headerRightGrid.horizontalSpacing = 20; + headerRight.setLayout(headerRightGrid); + headerRight.setData(RWT.CUSTOM_VARIANT, "header"); + + if (this.authorizationContextHolder.getAuthorizationContext().isLoggedIn()) { + final Label username = new Label(headerRight, SWT.NONE); + username.setData(RWT.CUSTOM_VARIANT, "header"); + username.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); + username.setText(this.authorizationContextHolder + .getAuthorizationContext() + .getLoggedInUser() + .get(t -> this.pageService.logoutOnError(t, pageContext)).username); + + final Button logout = this.widgetFactory.buttonLocalized(headerRight, "sebserver.logout"); + logout.setLayoutData(new GridData(SWT.RIGHT, SWT.FILL, true, true)); + logout.setData(RWT.CUSTOM_VARIANT, "header"); + logout.addListener(SWT.Selection, event -> { + this.pageService.logout(pageContext); + // show successful logout message + final MessageBox logoutSuccess = new Message( + pageContext.getShell(), + this.polyglotPageService.getI18nSupport().getText("sebserver.logout"), + this.polyglotPageService.getI18nSupport().getText("sebserver.logout.success.message"), + SWT.ICON_INFORMATION, + pageContext.getI18nSupport()); + logoutSuccess.open(null); + + // TODO Try to invalidate RWT's user session. + // This seems to be more difficult then expected and just invalidate the HttpSession doesn't work + // Try to send a redirect JSON to the client: https://bugs.eclipse.org/bugs/show_bug.cgi?id=388249 + }); + } + } + + private void composeLogoBar(final PageContext pageContext) { + final Composite logoBar = new Composite(pageContext.getParent(), SWT.NONE); + final GridData logoBarCell = new GridData(SWT.FILL, SWT.TOP, false, false); + logoBarCell.minimumHeight = 80; + logoBarCell.heightHint = 80; + logoBar.setLayoutData(logoBarCell); + logoBar.setData(RWT.CUSTOM_VARIANT, "logo"); + final GridLayout logoBarLayout = new GridLayout(2, false); + logoBarLayout.horizontalSpacing = 0; + logoBarLayout.marginHeight = 0; + logoBar.setLayout(logoBarLayout); + + final Composite logo = new Composite(logoBar, SWT.NONE); + final GridData logoCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); + logoCell.minimumHeight = LOGO_IMAGE_MAX_HEIGHT; + logoCell.heightHint = LOGO_IMAGE_MAX_HEIGHT; + logoCell.minimumWidth = LOGO_IMAGE_MAX_WIDTH; + logoCell.horizontalIndent = 50; + logo.setLayoutData(logoCell); + + // try to get institutional logo first. If no success, use default logo + loadInstitutionalLogo(pageContext, logo); + + final Composite langSupport = new Composite(logoBar, SWT.NONE); + final GridData langSupportCell = new GridData(SWT.RIGHT, SWT.CENTER, false, false); + langSupportCell.heightHint = 20; + logoCell.horizontalIndent = 50; + langSupport.setLayoutData(langSupportCell); + langSupport.setData(RWT.CUSTOM_VARIANT, "logo"); + final RowLayout rowLayout = new RowLayout(SWT.HORIZONTAL); + rowLayout.spacing = 7; + rowLayout.marginRight = 70; + langSupport.setLayout(rowLayout); + + if (this.multilingual) { + this.polyglotPageService.createLanguageSelector(pageContext.copyOf(langSupport)); + } + } + + private void composeContent(final PageContext pageContext) { + final Composite contentBackground = new Composite(pageContext.getParent(), SWT.NONE); + contentBackground.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); + contentBackground.setData(RWT.CUSTOM_VARIANT, "bgContent"); + final GridLayout innerGrid = new GridLayout(); + innerGrid.marginLeft = 50; + innerGrid.marginRight = 50; + innerGrid.marginHeight = 0; + innerGrid.marginWidth = 0; + + contentBackground.setLayout(innerGrid); + + final Composite content = new Composite(contentBackground, SWT.NONE); + content.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); + content.setData(RWT.CUSTOM_VARIANT, "content"); + final GridLayout contentGrid = new GridLayout(); + contentGrid.marginHeight = 0; + contentGrid.marginWidth = 0; + content.setLayout(contentGrid); + + final Composite contentInner = new Composite(content, SWT.NONE); + contentInner.setLayoutData(new GridData(SWT.CENTER, SWT.CENTER, true, true)); + final GridLayout gridLayout = new GridLayout(); + gridLayout.marginHeight = 0; + gridLayout.marginWidth = 0; + contentInner.setLayout(gridLayout); + + final String contentComposerName = pageContext.getAttribute( + AttributeKeys.PAGE_TEMPLATE_COMPOSER_NAME); + pageContext.composerService().compose( + contentComposerName, + pageContext.copyOf(contentInner)); + } + + private void composeFooter(final PageContext pageContext) { + final Composite footerBar = new Composite(pageContext.getParent(), SWT.NONE); + final GridData footerCell = new GridData(SWT.FILL, SWT.BOTTOM, false, false); + footerCell.minimumHeight = 30; + footerCell.heightHint = 30; + footerBar.setLayoutData(footerCell); + footerBar.setData(RWT.CUSTOM_VARIANT, "bgFooter"); + final GridLayout innerBarGrid = new GridLayout(); + innerBarGrid.marginHeight = 0; + innerBarGrid.marginWidth = 0; + innerBarGrid.marginLeft = 50; + innerBarGrid.marginRight = 50; + footerBar.setLayout(innerBarGrid); + + final Composite footer = new Composite(footerBar, SWT.NONE); + final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); + footer.setLayoutData(gridData); + final GridLayout footerGrid = new GridLayout(2, false); + footerGrid.marginHeight = 0; + footerGrid.marginWidth = 0; + footerGrid.horizontalSpacing = 0; + footer.setLayout(footerGrid); + footer.setData(RWT.CUSTOM_VARIANT, "footer"); + + final Composite footerLeft = new Composite(footer, SWT.NONE); + footerLeft.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, true)); + footerLeft.setData(RWT.CUSTOM_VARIANT, "footer"); + RowLayout rowLayout = new RowLayout(SWT.HORIZONTAL); + rowLayout.marginLeft = 20; + rowLayout.spacing = 20; + footerLeft.setLayout(rowLayout); + + final Composite footerRight = new Composite(footer, SWT.NONE); + footerRight.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, true, true)); + footerRight.setData(RWT.CUSTOM_VARIANT, "footer"); + rowLayout = new RowLayout(SWT.HORIZONTAL); + rowLayout.marginRight = 20; + footerRight.setLayout(rowLayout); + + final I18nSupport i18nSupport = this.widgetFactory.getI18nSupport(); + if (StringUtils.isNoneBlank(i18nSupport.getText(IMPRINT_TEXT_KEY, ""))) { + final Label imprint = this.widgetFactory.labelLocalized( + footerLeft, + CustomVariant.FOOTER, + IMPRINT_TEXT_KEY); + + imprint.addListener(SWT.MouseUp, event -> { + try { + pageContext.publishPageMessage(IMPRINT_TEXT_KEY, IMPRINT_MARKUP_TEXT_KEY); + } catch (final Exception e) { + log.error("Invalid markup for 'Imprint'", e); + } + }); + } + if (StringUtils.isNoneBlank(i18nSupport.getText(ABOUT_TEXT_KEY, ""))) { + final Label about = this.widgetFactory.labelLocalized( + footerLeft, + CustomVariant.FOOTER, + ABOUT_TEXT_KEY); + + about.addListener(SWT.MouseUp, event -> { + try { + pageContext.publishPageMessage(ABOUT_TEXT_KEY, ABOUT_MARKUP_TEXT_KEY); + } catch (final Exception e) { + log.error("Invalid markup for 'About'", e); + } + }); + } + if (StringUtils.isNoneBlank(i18nSupport.getText(HELP_TEXT_KEY, ""))) { + final Label help = this.widgetFactory.labelLocalized( + footerLeft, + CustomVariant.FOOTER, + HELP_TEXT_KEY); + + help.addListener(SWT.MouseUp, event -> { + try { + final String link = i18nSupport.getText(HELP_LINK_TEXT_KEY, ""); + if (StringUtils.isNoneBlank(link)) { + final UrlLauncher urlLauncher = RWT.getClient().getService(UrlLauncher.class); + urlLauncher.openURL(link); + } + } catch (final Exception e) { + log.error("Invalid Help link", e); + } + }); + + } + this.widgetFactory.labelLocalized( + footerRight, + CustomVariant.FOOTER, + new LocTextKey("sebserver.overall.version", this.sebServerVersion)); + } + + private void loadInstitutionalLogo(final PageContext pageContext, final Composite logo) { + logo.setData(RWT.CUSTOM_VARIANT, "bgLogo"); + try { + + final String imageBase64 = (String) RWT.getUISession() + .getHttpSession() + .getAttribute(API.PARAM_LOGO_IMAGE); + + if (StringUtils.isBlank(imageBase64)) { + return; + } + + final Base64InputStream input = new Base64InputStream( + new ByteArrayInputStream(imageBase64.getBytes(StandardCharsets.UTF_8)), + false); + + final Display display = pageContext.getShell().getDisplay(); + final Image image = new Image(display, input); + final Rectangle imageBounds = image.getBounds(); + final int width = Math.min(imageBounds.width, LOGO_IMAGE_MAX_WIDTH); + final int height = Math.min(imageBounds.height, LOGO_IMAGE_MAX_HEIGHT); + final ImageData imageData = image.getImageData().scaledTo(width, height); + + logo.setData(RWT.CUSTOM_VARIANT, "bgLogoNoImage"); + logo.setBackgroundImage(new Image(display, imageData)); + + } catch (final Exception e) { + log.warn("Get institutional logo failed: {}", e.getMessage()); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/ModalInputDialog.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/ModalInputDialog.java index 50ed02c1..cc4a41b9 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/ModalInputDialog.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/ModalInputDialog.java @@ -1,225 +1,220 @@ -/* - * Copyright (c) 2019 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.service.page.impl; - -import java.util.function.Consumer; -import java.util.function.Predicate; -import java.util.function.Supplier; - -import org.eclipse.rap.rwt.RWT; -import org.eclipse.swt.SWT; -import org.eclipse.swt.graphics.Rectangle; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Button; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Dialog; -import org.eclipse.swt.widgets.Shell; - -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.page.ModalInputDialogComposer; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; - -public class ModalInputDialog extends Dialog { - - private static final long serialVersionUID = -3448614119078234374L; - - public static final int DEFAULT_DIALOG_WIDTH = 400; - public static final int DEFAULT_DIALOG_HEIGHT = 600; - public static final int DEFAULT_DIALOG_BUTTON_WIDTH = 100; - public static final int LARGE_DIALOG_WIDTH = 600; - public static final int VERY_LARGE_DIALOG_WIDTH = 800; - - private static final LocTextKey CANCEL_TEXT_KEY = - new LocTextKey("sebserver.overall.action.cancel"); - private static final LocTextKey OK_TEXT_KEY = - new LocTextKey("sebserver.overall.action.ok"); - private static final LocTextKey CLOSE_TEXT_KEY = - new LocTextKey("sebserver.overall.action.close"); - - private final WidgetFactory widgetFactory; - private int dialogWidth = DEFAULT_DIALOG_WIDTH; - private int dialogHeight = DEFAULT_DIALOG_HEIGHT; - private int buttonWidth = DEFAULT_DIALOG_BUTTON_WIDTH; - - public ModalInputDialog( - final Shell parent, - final WidgetFactory widgetFactory) { - - super(parent, SWT.BORDER | SWT.TITLE | SWT.APPLICATION_MODAL | SWT.CLOSE); - this.widgetFactory = widgetFactory; - } - - public ModalInputDialog setDialogWidth(final int dialogWidth) { - this.dialogWidth = dialogWidth; - return this; - } - - public ModalInputDialog setLargeDialogWidth() { - this.dialogWidth = LARGE_DIALOG_WIDTH; - return this; - } - - public ModalInputDialog setVeryLargeDialogWidth() { - this.dialogWidth = VERY_LARGE_DIALOG_WIDTH; - return this; - } - - public ModalInputDialog setDialogHeight(final int dialogHeight) { - this.dialogHeight = dialogHeight; - return this; - } - - public ModalInputDialog setButtonWidth(final int buttonWidth) { - this.buttonWidth = buttonWidth; - return this; - } - - public void open( - final LocTextKey title, - final ModalInputDialogComposer contentComposer) { - - open( - title, - (Predicate) t -> true, - () -> { - }, contentComposer); - } - - public void open( - final LocTextKey title, - final Consumer callback, - final Runnable cancelCallback, - final ModalInputDialogComposer contentComposer) { - - final Predicate predicate = result -> { - callback.accept(result); - return true; - }; - - open(title, predicate, cancelCallback, contentComposer); - } - - public void open( - final LocTextKey title, - final Predicate callback, - final Runnable cancelCallback, - final ModalInputDialogComposer contentComposer) { - - // Create the selection dialog window - final Shell shell = new Shell(getParent(), getStyle()); - shell.setText(getText()); - shell.setData(RWT.CUSTOM_VARIANT, CustomVariant.MESSAGE.key); - shell.setText(this.widgetFactory.getI18nSupport().getText(title)); - shell.setLayout(new GridLayout(2, true)); - final GridData gridData2 = new GridData(SWT.FILL, SWT.TOP, false, false); - shell.setLayoutData(gridData2); - - final Composite main = new Composite(shell, SWT.NONE); - main.setLayout(new GridLayout()); - final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); - gridData.horizontalSpan = 2; - gridData.widthHint = this.dialogWidth; - main.setLayoutData(gridData); - - final Supplier valueSuppier = contentComposer.compose(main); - gridData.heightHint = calcDialogHeight(main); - - final Button ok = this.widgetFactory.buttonLocalized(shell, OK_TEXT_KEY); - GridData data = new GridData(GridData.HORIZONTAL_ALIGN_END); - data.widthHint = this.buttonWidth; - ok.setLayoutData(data); - ok.addListener(SWT.Selection, event -> { - if (valueSuppier != null) { - final T result = valueSuppier.get(); - if (callback.test(result)) { - shell.close(); - } - } else { - shell.close(); - } - }); - - shell.setDefaultButton(ok); - - final Button cancel = this.widgetFactory.buttonLocalized(shell, CANCEL_TEXT_KEY); - data = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); - data.widthHint = this.buttonWidth; - cancel.setLayoutData(data); - cancel.addListener(SWT.Selection, event -> { - if (cancelCallback != null) { - cancelCallback.run(); - } - shell.close(); - }); - - finishUp(shell); - } - - public void open( - final LocTextKey title, - final PageContext pageContext, - final Consumer contentComposer) { - - // Create the info dialog window - final Shell shell = new Shell(getParent(), getStyle()); - shell.setText(getText()); - shell.setData(RWT.CUSTOM_VARIANT, CustomVariant.MESSAGE.key); - shell.setText(this.widgetFactory.getI18nSupport().getText(title)); - shell.setLayout(new GridLayout()); - final GridData gridData2 = new GridData(SWT.FILL, SWT.TOP, true, true); - - shell.setLayoutData(gridData2); - - final Composite main = new Composite(shell, SWT.NONE); - main.setLayout(new GridLayout()); - final GridData gridData = new GridData(SWT.FILL, SWT.TOP, true, true); - gridData.widthHint = this.dialogWidth; - main.setLayoutData(gridData); - - contentComposer.accept(pageContext.copyOf(main)); - gridData.heightHint = calcDialogHeight(main); - - final Button close = this.widgetFactory.buttonLocalized(shell, CLOSE_TEXT_KEY); - final GridData data = new GridData(GridData.HORIZONTAL_ALIGN_CENTER); - data.widthHint = this.buttonWidth; - close.setLayoutData(data); - close.addListener(SWT.Selection, event -> { - shell.close(); - }); - - finishUp(shell); - } - - private void finishUp(final Shell shell) { - shell.pack(); - final Rectangle bounds = shell.getBounds(); - final Rectangle bounds2 = super.getParent().getDisplay().getBounds(); - bounds.x = (bounds2.width - bounds.width) / 2; - bounds.y = (bounds2.height - bounds.height) / 2; - shell.setBounds(bounds); - - shell.open(); - } - - private int calcDialogHeight(final Composite main) { - final int actualHeight = main.computeSize(SWT.DEFAULT, SWT.DEFAULT).y; - final int displayHeight = main.getDisplay().getClientArea().height; - final int availableHeight = (displayHeight < actualHeight + 100) - ? displayHeight - 100 - : actualHeight; - final int height = (availableHeight > this.dialogHeight) - ? this.dialogHeight - : availableHeight; - return height; - } - -} +/* + * Copyright (c) 2019 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.service.page.impl; + +import java.util.function.Consumer; +import java.util.function.Predicate; +import java.util.function.Supplier; + +import org.eclipse.rap.rwt.RWT; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Rectangle; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Dialog; +import org.eclipse.swt.widgets.Shell; + +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.page.ModalInputDialogComposer; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; + +public class ModalInputDialog extends Dialog { + + private static final long serialVersionUID = -3448614119078234374L; + + public static final int DEFAULT_DIALOG_WIDTH = 400; + public static final int DEFAULT_DIALOG_HEIGHT = 600; + public static final int DEFAULT_DIALOG_BUTTON_WIDTH = 100; + public static final int LARGE_DIALOG_WIDTH = 600; + public static final int VERY_LARGE_DIALOG_WIDTH = 800; + + private static final LocTextKey CANCEL_TEXT_KEY = + new LocTextKey("sebserver.overall.action.cancel"); + private static final LocTextKey OK_TEXT_KEY = + new LocTextKey("sebserver.overall.action.ok"); + private static final LocTextKey CLOSE_TEXT_KEY = + new LocTextKey("sebserver.overall.action.close"); + + private final WidgetFactory widgetFactory; + private int dialogWidth = DEFAULT_DIALOG_WIDTH; + private int dialogHeight = DEFAULT_DIALOG_HEIGHT; + private int buttonWidth = DEFAULT_DIALOG_BUTTON_WIDTH; + + public ModalInputDialog( + final Shell parent, + final WidgetFactory widgetFactory) { + + super(parent, SWT.BORDER | SWT.TITLE | SWT.APPLICATION_MODAL | SWT.CLOSE); + this.widgetFactory = widgetFactory; + } + + public ModalInputDialog setDialogWidth(final int dialogWidth) { + this.dialogWidth = dialogWidth; + return this; + } + + public ModalInputDialog setLargeDialogWidth() { + this.dialogWidth = LARGE_DIALOG_WIDTH; + return this; + } + + public ModalInputDialog setVeryLargeDialogWidth() { + this.dialogWidth = VERY_LARGE_DIALOG_WIDTH; + return this; + } + + public ModalInputDialog setDialogHeight(final int dialogHeight) { + this.dialogHeight = dialogHeight; + return this; + } + + public ModalInputDialog setButtonWidth(final int buttonWidth) { + this.buttonWidth = buttonWidth; + return this; + } + + public void open( + final LocTextKey title, + final ModalInputDialogComposer contentComposer) { + + open( + title, + t -> true, + () -> { + }, contentComposer); + } + + public void open( + final LocTextKey title, + final Consumer callback, + final Runnable cancelCallback, + final ModalInputDialogComposer contentComposer) { + + final Predicate predicate = result -> { + callback.accept(result); + return true; + }; + + open(title, predicate, cancelCallback, contentComposer); + } + + public void open( + final LocTextKey title, + final Predicate callback, + final Runnable cancelCallback, + final ModalInputDialogComposer contentComposer) { + + // Create the selection dialog window + final Shell shell = new Shell(getParent(), getStyle()); + shell.setText(getText()); + shell.setData(RWT.CUSTOM_VARIANT, CustomVariant.MESSAGE.key); + shell.setText(this.widgetFactory.getI18nSupport().getText(title)); + shell.setLayout(new GridLayout(2, true)); + final GridData gridData2 = new GridData(SWT.FILL, SWT.TOP, false, false); + shell.setLayoutData(gridData2); + + final Composite main = new Composite(shell, SWT.NONE); + main.setLayout(new GridLayout()); + final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); + gridData.horizontalSpan = 2; + gridData.widthHint = this.dialogWidth; + main.setLayoutData(gridData); + + final Supplier valueSupplier = contentComposer.compose(main); + gridData.heightHint = calcDialogHeight(main); + + final Button ok = this.widgetFactory.buttonLocalized(shell, OK_TEXT_KEY); + GridData data = new GridData(GridData.HORIZONTAL_ALIGN_END); + data.widthHint = this.buttonWidth; + ok.setLayoutData(data); + ok.addListener(SWT.Selection, event -> { + if (valueSupplier != null) { + final T result = valueSupplier.get(); + if (callback.test(result)) { + shell.close(); + } + } else { + shell.close(); + } + }); + + shell.setDefaultButton(ok); + + final Button cancel = this.widgetFactory.buttonLocalized(shell, CANCEL_TEXT_KEY); + data = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); + data.widthHint = this.buttonWidth; + cancel.setLayoutData(data); + cancel.addListener(SWT.Selection, event -> { + if (cancelCallback != null) { + cancelCallback.run(); + } + shell.close(); + }); + + finishUp(shell); + } + + public void open( + final LocTextKey title, + final PageContext pageContext, + final Consumer contentComposer) { + + // Create the info dialog window + final Shell shell = new Shell(getParent(), getStyle()); + shell.setText(getText()); + shell.setData(RWT.CUSTOM_VARIANT, CustomVariant.MESSAGE.key); + shell.setText(this.widgetFactory.getI18nSupport().getText(title)); + shell.setLayout(new GridLayout()); + final GridData gridData2 = new GridData(SWT.FILL, SWT.TOP, true, true); + + shell.setLayoutData(gridData2); + + final Composite main = new Composite(shell, SWT.NONE); + main.setLayout(new GridLayout()); + final GridData gridData = new GridData(SWT.FILL, SWT.TOP, true, true); + gridData.widthHint = this.dialogWidth; + main.setLayoutData(gridData); + + contentComposer.accept(pageContext.copyOf(main)); + gridData.heightHint = calcDialogHeight(main); + + final Button close = this.widgetFactory.buttonLocalized(shell, CLOSE_TEXT_KEY); + final GridData data = new GridData(GridData.HORIZONTAL_ALIGN_CENTER); + data.widthHint = this.buttonWidth; + close.setLayoutData(data); + close.addListener(SWT.Selection, event -> shell.close()); + + finishUp(shell); + } + + private void finishUp(final Shell shell) { + shell.pack(); + final Rectangle bounds = shell.getBounds(); + final Rectangle bounds2 = super.getParent().getDisplay().getBounds(); + bounds.x = (bounds2.width - bounds.width) / 2; + bounds.y = (bounds2.height - bounds.height) / 2; + shell.setBounds(bounds); + + shell.open(); + } + + private int calcDialogHeight(final Composite main) { + final int actualHeight = main.computeSize(SWT.DEFAULT, SWT.DEFAULT).y; + final int displayHeight = main.getDisplay().getClientArea().height; + final int availableHeight = (displayHeight < actualHeight + 100) + ? displayHeight - 100 + : actualHeight; + return Math.min(availableHeight, this.dialogHeight); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageContextImpl.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageContextImpl.java index bfbf1dd1..e481d290 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageContextImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageContextImpl.java @@ -1,346 +1,340 @@ -/* - * Copyright (c) 2019 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.service.page.impl; - -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.function.Consumer; - -import org.apache.commons.lang3.BooleanUtils; -import org.eclipse.rap.rwt.widgets.DialogCallback; -import org.eclipse.swt.SWT; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.MessageBox; -import org.eclipse.swt.widgets.Shell; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.api.APIMessage; -import ch.ethz.seb.sebserver.gbl.api.APIMessageError; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.model.EntityKey; -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.page.ComposerService; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.service.page.PageDefinition; -import ch.ethz.seb.sebserver.gui.service.page.PageMessageException; -import ch.ethz.seb.sebserver.gui.widget.Message; - -public class PageContextImpl implements PageContext { - - private static final Logger log = LoggerFactory.getLogger(PageContextImpl.class); - - private final I18nSupport i18nSupport; - private final ComposerService composerService; - private final Composite root; - private final Composite parent; - private final Map attributes; - - PageContextImpl( - final I18nSupport i18nSupport, - final ComposerService composerService, - final Composite root, - final Composite parent, - final Map attributes) { - - this.i18nSupport = i18nSupport; - this.composerService = composerService; - this.root = root; - this.parent = parent; - this.attributes = Utils.immutableMapOf(attributes); - } - - @Override - public I18nSupport getI18nSupport() { - return this.i18nSupport; - } - - @Override - public Shell getShell() { - if (this.root == null) { - return null; - } - - return this.root.getShell(); - } - - @Override - public ComposerService composerService() { - return this.composerService; - } - - @Override - public Composite getRoot() { - return this.root; - } - - @Override - public Composite getParent() { - return this.parent; - } - - @Override - public PageContext copy() { - return copyOf(this.parent); - } - - @Override - public PageContext copyOf(final Composite parent) { - return new PageContextImpl( - this.i18nSupport, - this.composerService, - this.root, - parent, - new HashMap<>(this.attributes)); - } - - @Override - public PageContext copyOfAttributes(final PageContext otherContext) { - final Map attrs = new HashMap<>(); - attrs.putAll(this.attributes); - attrs.putAll(((PageContextImpl) otherContext).attributes); - return new PageContextImpl( - this.i18nSupport, - this.composerService, - this.root, - this.parent, - attrs); - } - - @Override - public PageContext withAttribute(final String key, final String value) { - final Map attrs = new HashMap<>(); - attrs.putAll(this.attributes); - attrs.put(key, value); - return new PageContextImpl( - this.i18nSupport, - this.composerService, - this.root, - this.parent, - attrs); - } - - @Override - public String getAttribute(final String name) { - return this.attributes.get(name); - } - - @Override - public String getAttribute(final String name, final String def) { - if (this.attributes.containsKey(name)) { - return this.attributes.get(name); - } else { - return def; - } - } - - @Override - public boolean isReadonly() { - return BooleanUtils.toBoolean(getAttribute(AttributeKeys.READ_ONLY, "true")); - } - - @Override - public EntityKey getEntityKey() { - if (hasAttribute(AttributeKeys.ENTITY_ID) && hasAttribute(AttributeKeys.ENTITY_TYPE)) { - return new EntityKey( - getAttribute(AttributeKeys.ENTITY_ID), - EntityType.valueOf(getAttribute(AttributeKeys.ENTITY_TYPE))); - } - - return null; - } - - @Override - public EntityKey getParentEntityKey() { - if (hasAttribute(AttributeKeys.PARENT_ENTITY_ID) && hasAttribute(AttributeKeys.PARENT_ENTITY_TYPE)) { - return new EntityKey( - getAttribute(AttributeKeys.PARENT_ENTITY_ID), - EntityType.valueOf(getAttribute(AttributeKeys.PARENT_ENTITY_TYPE))); - } - - return null; - } - - @Override - public PageContext withEntityKey(final EntityKey entityKey) { - if (entityKey == null) { - return removeAttribute(AttributeKeys.ENTITY_ID) - .removeAttribute(AttributeKeys.ENTITY_TYPE); - } - return withAttribute(AttributeKeys.ENTITY_ID, entityKey.modelId) - .withAttribute(AttributeKeys.ENTITY_TYPE, entityKey.entityType.name()); - } - - @Override - public PageContext withParentEntityKey(final EntityKey entityKey) { - if (entityKey == null) { - return removeAttribute(AttributeKeys.PARENT_ENTITY_ID) - .removeAttribute(AttributeKeys.PARENT_ENTITY_TYPE); - } - return withAttribute(AttributeKeys.PARENT_ENTITY_ID, entityKey.modelId) - .withAttribute(AttributeKeys.PARENT_ENTITY_TYPE, entityKey.entityType.name()); - } - - @Override - public PageContext clearEntityKeys() { - return withEntityKey(null) - .withParentEntityKey(null); - } - - @Override - public boolean hasAttribute(final String name) { - return this.attributes.containsKey(name); - } - - @Override - public PageContext removeAttribute(final String name) { - final Map attrs = new HashMap<>(); - attrs.putAll(this.attributes); - attrs.remove(name); - return new PageContextImpl( - this.i18nSupport, - this.composerService, - this.root, - this.parent, - attrs); - } - - @Override - public PageContext clearAttributes() { - return new PageContextImpl( - this.i18nSupport, - this.composerService, - this.root, - this.parent, - null); - } - - @Override - public void applyConfirmDialog(final LocTextKey confirmMessage, final Consumer callback) { - final Message messageBox = new Message( - this.root.getShell(), - this.i18nSupport.getText("sebserver.dialog.confirm.title"), - this.i18nSupport.getText(confirmMessage), - SWT.OK | SWT.CANCEL, - this.i18nSupport); - messageBox.setMarkupEnabled(true); - messageBox.open(new ConfirmDialogCallback(callback)); - } - - @Override - public void forwardToPage(final PageDefinition pageDefinition) { - this.composerService.compose( - pageDefinition.composer(), - pageDefinition.applyPageContext(copyOf(this.root))); - } - - @Override - public void forwardToMainPage() { - forwardToPage(this.composerService.mainPage()); - } - - @Override - public void forwardToLoginPage() { - this.clearAttributes() - .forwardToPage(this.composerService.loginPage()); - } - - @Override - public void publishPageMessage(final LocTextKey title, final LocTextKey message) { - final MessageBox messageBox = new Message( - getShell(), - (title != null) - ? this.i18nSupport.getText(title) - : "", - this.i18nSupport.getText(message), - SWT.NONE, - this.i18nSupport); - - messageBox.setMarkupEnabled(true); - messageBox.open(null); - } - - @Override - public void publishPageMessage(final PageMessageException pme) { - final MessageBox messageBox = new Message( - getShell(), - this.i18nSupport.getText("sebserver.page.message"), - this.i18nSupport.getText(pme.getMessageKey()), - SWT.NONE, - this.i18nSupport); - messageBox.setMarkupEnabled(true); - messageBox.open(null); - } - - @Override - public void notifyError(final LocTextKey message, final Exception error) { - - log.error("Unexpected GUI error notified: {}", error.getMessage()); - - final String errorMessage = message != null - ? this.i18nSupport.getText(message) - : error.getMessage(); - - if (error instanceof APIMessageError) { - final List errorMessages = ((APIMessageError) error).getErrorMessages(); - final MessageBox messageBox = new Message( - getShell(), - this.i18nSupport.getText("sebserver.error.unexpected"), - APIMessage.toHTML(errorMessage, errorMessages), - SWT.ERROR, - this.i18nSupport); - messageBox.setMarkupEnabled(true); - messageBox.open(null); - return; - } - - final MessageBox messageBox = new Message( - getShell(), - this.i18nSupport.getText("sebserver.error.unexpected"), - Utils.formatHTMLLines(errorMessage + "

Cause: " + error.getMessage()), - SWT.ERROR, - this.i18nSupport); - messageBox.open(null); - } - - @Override - public String toString() { - return "PageContextImpl [root=" + this.root + ", parent=" + this.parent + ", attributes=" + this.attributes - + "]"; - } - - private static final class ConfirmDialogCallback implements DialogCallback { - private static final long serialVersionUID = 1491270214433492441L; - private final Consumer onOK; - - private ConfirmDialogCallback(final Consumer onOK) { - this.onOK = onOK; - } - - @Override - public void dialogClosed(final int returnCode) { - if (returnCode == SWT.OK) { - try { - this.onOK.accept(true); - } catch (final Exception e) { - log.error( - "Unexpected on confirm callback execution. This should not happen, plase secure the given onOK Runnable", - e); - this.onOK.accept(false); - } - } else { - this.onOK.accept(false); - } - } - } - -} +/* + * Copyright (c) 2019 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.service.page.impl; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.Consumer; + +import org.apache.commons.lang3.BooleanUtils; +import org.eclipse.rap.rwt.widgets.DialogCallback; +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.MessageBox; +import org.eclipse.swt.widgets.Shell; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import ch.ethz.seb.sebserver.gbl.api.APIMessage; +import ch.ethz.seb.sebserver.gbl.api.APIMessageError; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.model.EntityKey; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.page.ComposerService; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.service.page.PageDefinition; +import ch.ethz.seb.sebserver.gui.service.page.PageMessageException; +import ch.ethz.seb.sebserver.gui.widget.Message; + +public class PageContextImpl implements PageContext { + + private static final Logger log = LoggerFactory.getLogger(PageContextImpl.class); + + private final I18nSupport i18nSupport; + private final ComposerService composerService; + private final Composite root; + private final Composite parent; + private final Map attributes; + + PageContextImpl( + final I18nSupport i18nSupport, + final ComposerService composerService, + final Composite root, + final Composite parent, + final Map attributes) { + + this.i18nSupport = i18nSupport; + this.composerService = composerService; + this.root = root; + this.parent = parent; + this.attributes = Utils.immutableMapOf(attributes); + } + + @Override + public I18nSupport getI18nSupport() { + return this.i18nSupport; + } + + @Override + public Shell getShell() { + if (this.root == null) { + return null; + } + + return this.root.getShell(); + } + + @Override + public ComposerService composerService() { + return this.composerService; + } + + @Override + public Composite getRoot() { + return this.root; + } + + @Override + public Composite getParent() { + return this.parent; + } + + @Override + public PageContext copy() { + return copyOf(this.parent); + } + + @Override + public PageContext copyOf(final Composite parent) { + return new PageContextImpl( + this.i18nSupport, + this.composerService, + this.root, + parent, + new HashMap<>(this.attributes)); + } + + @Override + public PageContext copyOfAttributes(final PageContext otherContext) { + final Map attrs = new HashMap<>(); + attrs.putAll(this.attributes); + attrs.putAll(((PageContextImpl) otherContext).attributes); + return new PageContextImpl( + this.i18nSupport, + this.composerService, + this.root, + this.parent, + attrs); + } + + @Override + public PageContext withAttribute(final String key, final String value) { + final Map attrs = new HashMap<>(this.attributes); + attrs.put(key, value); + return new PageContextImpl( + this.i18nSupport, + this.composerService, + this.root, + this.parent, + attrs); + } + + @Override + public String getAttribute(final String name) { + return this.attributes.get(name); + } + + @Override + public String getAttribute(final String name, final String def) { + return this.attributes.getOrDefault(name, def); + } + + @Override + public boolean isReadonly() { + return BooleanUtils.toBoolean(getAttribute(AttributeKeys.READ_ONLY, "true")); + } + + @Override + public EntityKey getEntityKey() { + if (hasAttribute(AttributeKeys.ENTITY_ID) && hasAttribute(AttributeKeys.ENTITY_TYPE)) { + return new EntityKey( + getAttribute(AttributeKeys.ENTITY_ID), + EntityType.valueOf(getAttribute(AttributeKeys.ENTITY_TYPE))); + } + + return null; + } + + @Override + public EntityKey getParentEntityKey() { + if (hasAttribute(AttributeKeys.PARENT_ENTITY_ID) && hasAttribute(AttributeKeys.PARENT_ENTITY_TYPE)) { + return new EntityKey( + getAttribute(AttributeKeys.PARENT_ENTITY_ID), + EntityType.valueOf(getAttribute(AttributeKeys.PARENT_ENTITY_TYPE))); + } + + return null; + } + + @Override + public PageContext withEntityKey(final EntityKey entityKey) { + if (entityKey == null) { + return removeAttribute(AttributeKeys.ENTITY_ID) + .removeAttribute(AttributeKeys.ENTITY_TYPE); + } + return withAttribute(AttributeKeys.ENTITY_ID, entityKey.modelId) + .withAttribute(AttributeKeys.ENTITY_TYPE, entityKey.entityType.name()); + } + + @Override + public PageContext withParentEntityKey(final EntityKey entityKey) { + if (entityKey == null) { + return removeAttribute(AttributeKeys.PARENT_ENTITY_ID) + .removeAttribute(AttributeKeys.PARENT_ENTITY_TYPE); + } + return withAttribute(AttributeKeys.PARENT_ENTITY_ID, entityKey.modelId) + .withAttribute(AttributeKeys.PARENT_ENTITY_TYPE, entityKey.entityType.name()); + } + + @Override + public PageContext clearEntityKeys() { + return withEntityKey(null) + .withParentEntityKey(null); + } + + @Override + public boolean hasAttribute(final String name) { + return this.attributes.containsKey(name); + } + + @Override + public PageContext removeAttribute(final String name) { + final Map attrs = new HashMap<>(this.attributes); + attrs.remove(name); + return new PageContextImpl( + this.i18nSupport, + this.composerService, + this.root, + this.parent, + attrs); + } + + @Override + public PageContext clearAttributes() { + return new PageContextImpl( + this.i18nSupport, + this.composerService, + this.root, + this.parent, + null); + } + + @Override + public void applyConfirmDialog(final LocTextKey confirmMessage, final Consumer callback) { + final Message messageBox = new Message( + this.root.getShell(), + this.i18nSupport.getText("sebserver.dialog.confirm.title"), + this.i18nSupport.getText(confirmMessage), + SWT.OK | SWT.CANCEL, + this.i18nSupport); + messageBox.setMarkupEnabled(true); + messageBox.open(new ConfirmDialogCallback(callback)); + } + + @Override + public void forwardToPage(final PageDefinition pageDefinition) { + this.composerService.compose( + pageDefinition.composer(), + pageDefinition.applyPageContext(copyOf(this.root))); + } + + @Override + public void forwardToMainPage() { + forwardToPage(this.composerService.mainPage()); + } + + @Override + public void forwardToLoginPage() { + this.clearAttributes() + .forwardToPage(this.composerService.loginPage()); + } + + @Override + public void publishPageMessage(final LocTextKey title, final LocTextKey message) { + final MessageBox messageBox = new Message( + getShell(), + (title != null) + ? this.i18nSupport.getText(title) + : "", + this.i18nSupport.getText(message), + SWT.NONE, + this.i18nSupport); + + messageBox.setMarkupEnabled(true); + messageBox.open(null); + } + + @Override + public void publishPageMessage(final PageMessageException pme) { + final MessageBox messageBox = new Message( + getShell(), + this.i18nSupport.getText("sebserver.page.message"), + this.i18nSupport.getText(pme.getMessageKey()), + SWT.NONE, + this.i18nSupport); + messageBox.setMarkupEnabled(true); + messageBox.open(null); + } + + @Override + public void notifyError(final LocTextKey message, final Exception error) { + + log.error("Unexpected GUI error notified: {}", error.getMessage()); + + final String errorMessage = message != null + ? this.i18nSupport.getText(message) + : error.getMessage(); + + if (error instanceof APIMessageError) { + final List errorMessages = ((APIMessageError) error).getErrorMessages(); + final MessageBox messageBox = new Message( + getShell(), + this.i18nSupport.getText("sebserver.error.unexpected"), + APIMessage.toHTML(errorMessage, errorMessages), + SWT.ERROR, + this.i18nSupport); + messageBox.setMarkupEnabled(true); + messageBox.open(null); + return; + } + + final MessageBox messageBox = new Message( + getShell(), + this.i18nSupport.getText("sebserver.error.unexpected"), + Utils.formatHTMLLines(errorMessage + "

Cause: " + error.getMessage()), + SWT.ERROR, + this.i18nSupport); + messageBox.open(null); + } + + @Override + public String toString() { + return "PageContextImpl [root=" + this.root + ", parent=" + this.parent + ", attributes=" + this.attributes + + "]"; + } + + private static final class ConfirmDialogCallback implements DialogCallback { + private static final long serialVersionUID = 1491270214433492441L; + private final Consumer onOK; + + private ConfirmDialogCallback(final Consumer onOK) { + this.onOK = onOK; + } + + @Override + public void dialogClosed(final int returnCode) { + if (returnCode == SWT.OK) { + try { + this.onOK.accept(true); + } catch (final Exception e) { + log.error( + "Unexpected on confirm callback execution. This should not happen, please secure the given onOK Runnable", + e); + this.onOK.accept(false); + } + } else { + this.onOK.accept(false); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageServiceImpl.java index e276f578..73ebd521 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/page/impl/PageServiceImpl.java @@ -227,7 +227,7 @@ public class PageServiceImpl implements PageService { final int dependencies = (int) entities.stream() .flatMap(entity -> { final RestCall>.RestCallBuilder builder = - restService.>getBuilder( + restService.getBuilder( entity.entityType(), CallType.GET_DEPENDENCIES); @@ -366,7 +366,7 @@ public class PageServiceImpl implements PageService { final boolean logoutSuccessful = this.currentUser.logout(); if (!logoutSuccessful) { - log.warn("Failed to logout. See logfiles for more information"); + log.warn("Failed to logout. See log-files for more information"); } } catch (final Exception e) { diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/push/ServerPushService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/push/ServerPushService.java index 1da98370..fb4b5e6a 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/push/ServerPushService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/push/ServerPushService.java @@ -1,104 +1,104 @@ -/* - * 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.service.push; - -import java.util.function.Consumer; - -import org.eclipse.rap.rwt.service.ServerPushSession; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Service; - -/** Puts RAP's server-push functionality in a well defined service by using a context - * as state holder and the possibility to split the server-push process into two - * separated processes, a business-process to get and update business data and the - * an update-process to update the UI after according to updated data */ -@Lazy -@Service -public class ServerPushService { - - private static final Logger log = LoggerFactory.getLogger(ServerPushService.class); - - public void runServerPush( - final ServerPushContext context, - final long intervalPause, - final Consumer update) { - - this.runServerPush(context, intervalPause, null, update); - } - - public void runServerPush( - final ServerPushContext context, - final long intervalPause, - final Consumer business, - final Consumer update) { - - final ServerPushSession pushSession = new ServerPushSession(); - - pushSession.start(); - final Thread bgThread = new Thread(() -> { - while (!context.isDisposed() && context.runAgain()) { - - try { - Thread.sleep(intervalPause); - } catch (final Exception e) { - if (log.isDebugEnabled()) { - log.debug("unexpected error while sleep: ", e); - } - } - - if (business != null) { - try { - log.trace("Call business on Server Push Session on: {}", Thread.currentThread().getName()); - business.accept(context); - } catch (final Exception e) { - log.error("Unexpected error while do business for server push service", e); - if (context.runAgain()) { - continue; - } else { - return; - } - } - } - - if (!context.isDisposed()) { - - log.trace("Call update on Server Push Session on: {}", Thread.currentThread().getName()); - - context.getDisplay().asyncExec(() -> { - try { - update.accept(context); - } catch (final Exception e) { - log.warn( - "Failed to update on Server Push Session {}. It seems that the UISession is not available anymore. " - + "This may source from a connection interruption.", - Thread.currentThread().getName(), e.getMessage()); - } - }); - } - } - - log.info("Stop Server Push Session on: {}", Thread.currentThread().getName()); - try { - pushSession.stop(); - } catch (final Exception e) { - log.warn( - "Failed to stop Server Push Session on: {}. It seems that the UISession is not available anymore. This may source from a connection interruption", - Thread.currentThread().getName(), e); - } - - }); - - log.info("Start new Server Push Session on: {}", bgThread.getName()); - - bgThread.setDaemon(true); - bgThread.start(); - } -} +/* + * 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.service.push; + +import java.util.function.Consumer; + +import org.eclipse.rap.rwt.service.ServerPushSession; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; + +/** Puts RAP's server-push functionality in a well defined service by using a context + * as state holder and the possibility to split the server-push process into two + * separated processes, a business-process to get and update business data and the + * an update-process to update the UI after according to updated data */ +@Lazy +@Service +public class ServerPushService { + + private static final Logger log = LoggerFactory.getLogger(ServerPushService.class); + + public void runServerPush( + final ServerPushContext context, + final long intervalPause, + final Consumer update) { + + this.runServerPush(context, intervalPause, null, update); + } + + public void runServerPush( + final ServerPushContext context, + final long intervalPause, + final Consumer business, + final Consumer update) { + + final ServerPushSession pushSession = new ServerPushSession(); + + pushSession.start(); + final Thread bgThread = new Thread(() -> { + while (!context.isDisposed() && context.runAgain()) { + + try { + Thread.sleep(intervalPause); + } catch (final Exception e) { + if (log.isDebugEnabled()) { + log.debug("unexpected error while sleep: ", e); + } + } + + if (business != null) { + try { + log.trace("Call business on Server Push Session on: {}", Thread.currentThread().getName()); + business.accept(context); + } catch (final Exception e) { + log.error("Unexpected error while do business for server push service", e); + if (context.runAgain()) { + continue; + } else { + return; + } + } + } + + if (!context.isDisposed()) { + + log.trace("Call update on Server Push Session on: {}", Thread.currentThread().getName()); + + context.getDisplay().asyncExec(() -> { + try { + update.accept(context); + } catch (final Exception e) { + log.warn( + "Failed to update on Server Push Session {}. It seems that the UISession is not available anymore. " + + "This may source from a connection interruption. cause: {}", + Thread.currentThread().getName(), e.getMessage()); + } + }); + } + } + + log.info("Stop Server Push Session on: {}", Thread.currentThread().getName()); + try { + pushSession.stop(); + } catch (final Exception e) { + log.warn( + "Failed to stop Server Push Session on: {}. It seems that the UISession is not available anymore. This may source from a connection interruption", + Thread.currentThread().getName(), e); + } + + }); + + log.info("Start new Server Push Session on: {}", bgThread.getName()); + + bgThread.setDaemon(true); + bgThread.start(); + } +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java index 53bdeddb..5ba0c74f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java @@ -1,119 +1,116 @@ -/* - * Copyright (c) 2019 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.service.remote.download; - -import java.io.IOException; -import java.util.Collection; -import java.util.Map; -import java.util.function.Function; -import java.util.stream.Collectors; - -import javax.servlet.ServletException; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import org.apache.commons.lang3.StringUtils; -import org.eclipse.rap.rwt.RWT; -import org.eclipse.rap.rwt.service.ServiceHandler; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Service; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; - -@Lazy -@Service -@GuiProfile -public class DownloadService implements ServiceHandler { - - private static final Logger log = LoggerFactory.getLogger(DownloadService.class); - - public static final String DOWNLOAD_SERVICE_NAME = "DOWNLOAD_SERVICE"; - public static final String HANDLER_NAME_PARAMETER = "download-handler-name"; - public static final String DOWNLOAD_FILE_NAME = "download-file-name"; - - private final Map handler; - - protected DownloadService(final Collection handler) { - this.handler = handler - .stream() - .collect(Collectors.toMap( - h -> h.getClass().getSimpleName(), - Function.identity())); - } - - @Override - public void service( - final HttpServletRequest request, - final HttpServletResponse response) throws IOException, ServletException { - - log.debug("Received download service request: {}", request.getRequestURI()); - - final String handlerName = request.getParameter(HANDLER_NAME_PARAMETER); - if (StringUtils.isBlank(handlerName)) { - log.error("Missing request parameter {}. Ignoring download service request", - HANDLER_NAME_PARAMETER); - return; - } - - if (!this.handler.containsKey(handlerName)) { - log.error("Missing DownloadServiceHandler with name {}. Ignoring download service request", - handlerName); - return; - } - - this.handler - .get(handlerName) - .processDownload(request, response); - } - - public String createDownloadURL( - final String modelId, - final Class handlerClass, - final String downloadFileName) { - - return createDownloadURL(modelId, null, handlerClass, downloadFileName); - } - - public String createDownloadURL( - final String modelId, - final String parentModelId, - final Class handlerClass, - final String downloadFileName) { - - final StringBuilder url = new StringBuilder() - .append(RWT.getServiceManager() - .getServiceHandlerUrl(DownloadService.DOWNLOAD_SERVICE_NAME)) - .append(Constants.FORM_URL_ENCODED_SEPARATOR) - .append(API.PARAM_MODEL_ID) - .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) - .append(modelId) - .append(Constants.FORM_URL_ENCODED_SEPARATOR) - .append(DownloadService.HANDLER_NAME_PARAMETER) - .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) - .append(handlerClass.getSimpleName()) - .append(Constants.FORM_URL_ENCODED_SEPARATOR) - .append(DownloadService.DOWNLOAD_FILE_NAME) - .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) - .append(downloadFileName); - - if (StringUtils.isNotBlank(parentModelId)) { - url.append(Constants.FORM_URL_ENCODED_SEPARATOR) - .append(API.PARAM_PARENT_MODEL_ID) - .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) - .append(parentModelId); - } - - return url.toString(); - } - -} +/* + * Copyright (c) 2019 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.service.remote.download; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import org.apache.commons.lang3.StringUtils; +import org.eclipse.rap.rwt.RWT; +import org.eclipse.rap.rwt.service.ServiceHandler; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.util.Collection; +import java.util.Map; +import java.util.function.Function; +import java.util.stream.Collectors; + +/** Implements a eclipse RAP ServiceHandler to handle downloads */ +@Lazy +@Service +@GuiProfile +public class DownloadService implements ServiceHandler { + + private static final Logger log = LoggerFactory.getLogger(DownloadService.class); + + public static final String DOWNLOAD_SERVICE_NAME = "DOWNLOAD_SERVICE"; + public static final String HANDLER_NAME_PARAMETER = "download-handler-name"; + public static final String DOWNLOAD_FILE_NAME = "download-file-name"; + + private final Map handler; + + protected DownloadService(final Collection handler) { + this.handler = handler + .stream() + .collect(Collectors.toMap( + h -> h.getClass().getSimpleName(), + Function.identity())); + } + + @Override + public void service( + final HttpServletRequest request, + final HttpServletResponse response) { + + log.debug("Received download service request: {}", request.getRequestURI()); + + final String handlerName = request.getParameter(HANDLER_NAME_PARAMETER); + if (StringUtils.isBlank(handlerName)) { + log.error("Missing request parameter {}. Ignoring download service request", + HANDLER_NAME_PARAMETER); + return; + } + + if (!this.handler.containsKey(handlerName)) { + log.error("Missing DownloadServiceHandler with name {}. Ignoring download service request", + handlerName); + return; + } + + this.handler + .get(handlerName) + .processDownload(request, response); + } + + public String createDownloadURL( + final String modelId, + final Class handlerClass, + final String downloadFileName) { + + return createDownloadURL(modelId, null, handlerClass, downloadFileName); + } + + public String createDownloadURL( + final String modelId, + final String parentModelId, + final Class handlerClass, + final String downloadFileName) { + + final StringBuilder url = new StringBuilder() + .append(RWT.getServiceManager() + .getServiceHandlerUrl(DownloadService.DOWNLOAD_SERVICE_NAME)) + .append(Constants.FORM_URL_ENCODED_SEPARATOR) + .append(API.PARAM_MODEL_ID) + .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) + .append(modelId) + .append(Constants.FORM_URL_ENCODED_SEPARATOR) + .append(DownloadService.HANDLER_NAME_PARAMETER) + .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) + .append(handlerClass.getSimpleName()) + .append(Constants.FORM_URL_ENCODED_SEPARATOR) + .append(DownloadService.DOWNLOAD_FILE_NAME) + .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) + .append(downloadFileName); + + if (StringUtils.isNotBlank(parentModelId)) { + url.append(Constants.FORM_URL_ENCODED_SEPARATOR) + .append(API.PARAM_PARENT_MODEL_ID) + .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) + .append(parentModelId); + } + + return url.toString(); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadServiceHandler.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadServiceHandler.java index 2a6c84d7..c036c75a 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadServiceHandler.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadServiceHandler.java @@ -1,18 +1,23 @@ -/* - * Copyright (c) 2019 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.service.remote.download; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -public interface DownloadServiceHandler { - - void processDownload(final HttpServletRequest request, final HttpServletResponse response); - -} +/* + * Copyright (c) 2019 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.service.remote.download; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +/** Interface defining a service to handle downloads */ +public interface DownloadServiceHandler { + + /** Process a requested download + * + * @param request The download HttpServletRequest + * @param response the response to send the download to */ + void processDownload(final HttpServletRequest request, final HttpServletResponse response); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebClientConfigDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebClientConfigDownload.java index 06940051..0652ce35 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebClientConfigDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebClientConfigDownload.java @@ -1,69 +1,69 @@ -/* - * Copyright (c) 2019 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.service.remote.download; - -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; - -import org.apache.tomcat.util.http.fileupload.IOUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.clientconfig.ExportClientConfig; - -@Lazy -@Component -@GuiProfile -public class SebClientConfigDownload extends AbstractDownloadServiceHandler { - - private static final Logger log = LoggerFactory.getLogger(SebClientConfigDownload.class); - - private final RestService restService; - public final String downloadFileName; - - protected SebClientConfigDownload( - final RestService restService, - @Value("${sebserver.gui.seb.client.config.download.filename}") final String downloadFileName) { - - this.restService = restService; - this.downloadFileName = downloadFileName; - } - - @Override - protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { - - final InputStream input = this.restService.getBuilder(ExportClientConfig.class) - .withURIVariable(API.PARAM_MODEL_ID, modelId) - .call() - .getOrThrow(); - - try { - IOUtils.copyLarge(input, downloadOut); - } catch (final IOException e) { - log.error( - "Unexpected error while streaming incomming config data from web-service to output-stream of download response: ", - e); - } finally { - try { - downloadOut.flush(); - downloadOut.close(); - } catch (final IOException e) { - log.error("Unexpected error while trying to close download output-stream"); - } - } - } - -} +/* + * Copyright (c) 2019 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.service.remote.download; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +import org.apache.tomcat.util.http.fileupload.IOUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.clientconfig.ExportClientConfig; + +@Lazy +@Component +@GuiProfile +public class SebClientConfigDownload extends AbstractDownloadServiceHandler { + + private static final Logger log = LoggerFactory.getLogger(SebClientConfigDownload.class); + + private final RestService restService; + public final String downloadFileName; + + protected SebClientConfigDownload( + final RestService restService, + @Value("${sebserver.gui.seb.client.config.download.filename}") final String downloadFileName) { + + this.restService = restService; + this.downloadFileName = downloadFileName; + } + + @Override + protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { + + final InputStream input = this.restService.getBuilder(ExportClientConfig.class) + .withURIVariable(API.PARAM_MODEL_ID, modelId) + .call() + .getOrThrow(); + + try { + IOUtils.copyLarge(input, downloadOut); + } catch (final IOException e) { + log.error( + "Unexpected error while streaming incoming config data from web-service to output-stream of download response: ", + e); + } finally { + try { + downloadOut.flush(); + downloadOut.close(); + } catch (final IOException e) { + log.error("Unexpected error while trying to close download output-stream"); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigDownload.java index 37f088a8..151744b6 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigDownload.java @@ -1,64 +1,64 @@ -/* - * Copyright (c) 2019 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.service.remote.download; - -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; - -import org.apache.tomcat.util.http.fileupload.IOUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.ExportExamConfig; - -@Lazy -@Component -@GuiProfile -public class SebExamConfigDownload extends AbstractDownloadServiceHandler { - - private static final Logger log = LoggerFactory.getLogger(SebExamConfigDownload.class); - - private final RestService restService; - - protected SebExamConfigDownload(final RestService restService) { - this.restService = restService; - } - - @Override - protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { - - final InputStream input = this.restService.getBuilder(ExportExamConfig.class) - .withURIVariable(API.PARAM_MODEL_ID, modelId) - .withURIVariable(API.PARAM_PARENT_MODEL_ID, parentModelId) - .call() - .getOrThrow(); - - try { - IOUtils.copyLarge(input, downloadOut); - } catch (final IOException e) { - log.error( - "Unexpected error while streaming incomming config data from web-service to output-stream of download response: ", - e); - } finally { - try { - downloadOut.flush(); - downloadOut.close(); - } catch (final IOException e) { - log.error("Unexpected error while trying to close download output-stream"); - } - } - } - -} +/* + * Copyright (c) 2019 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.service.remote.download; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +import org.apache.tomcat.util.http.fileupload.IOUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.ExportExamConfig; + +@Lazy +@Component +@GuiProfile +public class SebExamConfigDownload extends AbstractDownloadServiceHandler { + + private static final Logger log = LoggerFactory.getLogger(SebExamConfigDownload.class); + + private final RestService restService; + + protected SebExamConfigDownload(final RestService restService) { + this.restService = restService; + } + + @Override + protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { + + final InputStream input = this.restService.getBuilder(ExportExamConfig.class) + .withURIVariable(API.PARAM_MODEL_ID, modelId) + .withURIVariable(API.PARAM_PARENT_MODEL_ID, parentModelId) + .call() + .getOrThrow(); + + try { + IOUtils.copyLarge(input, downloadOut); + } catch (final IOException e) { + log.error( + "Unexpected error while streaming incoming config data from web-service to output-stream of download response: ", + e); + } finally { + try { + downloadOut.flush(); + downloadOut.close(); + } catch (final IOException e) { + log.error("Unexpected error while trying to close download output-stream"); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigPlaintextDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigPlaintextDownload.java index 8f25bf30..6bbcff1c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigPlaintextDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SebExamConfigPlaintextDownload.java @@ -1,63 +1,63 @@ -/* - * Copyright (c) 2019 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.service.remote.download; - -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; - -import org.apache.tomcat.util.http.fileupload.IOUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig.ExportPlainXML; - -@Lazy -@Component -@GuiProfile -public class SebExamConfigPlaintextDownload extends AbstractDownloadServiceHandler { - - private static final Logger log = LoggerFactory.getLogger(SebExamConfigPlaintextDownload.class); - - private final RestService restService; - - protected SebExamConfigPlaintextDownload(final RestService restService) { - this.restService = restService; - } - - @Override - protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { - - final InputStream input = this.restService.getBuilder(ExportPlainXML.class) - .withURIVariable(API.PARAM_MODEL_ID, modelId) - .call() - .getOrThrow(); - - try { - IOUtils.copyLarge(input, downloadOut); - } catch (final IOException e) { - log.error( - "Unexpected error while streaming incomming config data from web-service to output-stream of download response: ", - e); - } finally { - try { - downloadOut.flush(); - downloadOut.close(); - } catch (final IOException e) { - log.error("Unexpected error while trying to close download output-stream"); - } - } - } - -} +/* + * Copyright (c) 2019 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.service.remote.download; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +import org.apache.tomcat.util.http.fileupload.IOUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig.ExportPlainXML; + +@Lazy +@Component +@GuiProfile +public class SebExamConfigPlaintextDownload extends AbstractDownloadServiceHandler { + + private static final Logger log = LoggerFactory.getLogger(SebExamConfigPlaintextDownload.class); + + private final RestService restService; + + protected SebExamConfigPlaintextDownload(final RestService restService) { + this.restService = restService; + } + + @Override + protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) { + + final InputStream input = this.restService.getBuilder(ExportPlainXML.class) + .withURIVariable(API.PARAM_MODEL_ID, modelId) + .call() + .getOrThrow(); + + try { + IOUtils.copyLarge(input, downloadOut); + } catch (final IOException e) { + log.error( + "Unexpected error while streaming incoming config data from web-service to output-stream of download response: ", + e); + } finally { + try { + downloadOut.flush(); + downloadOut.close(); + } catch (final IOException e) { + log.error("Unexpected error while trying to close download output-stream"); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/AbstractExportCall.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/AbstractExportCall.java index 84f341d0..632529ac 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/AbstractExportCall.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/AbstractExportCall.java @@ -1,49 +1,45 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api; - -import java.io.InputStream; - -import org.apache.commons.io.IOUtils; -import org.springframework.http.HttpMethod; -import org.springframework.http.MediaType; -import org.springframework.http.client.ClientHttpRequest; - -import ch.ethz.seb.sebserver.gbl.util.Result; - -public abstract class AbstractExportCall extends RestCall { - - protected AbstractExportCall( - final TypeKey typeKey, - final HttpMethod httpMethod, - final MediaType contentType, - final String path) { - - super(typeKey, httpMethod, contentType, path); - } - - @Override - protected Result exchange(final RestCallBuilder builder) { - - return Result.tryCatch(() -> { - - return builder - .getRestTemplate() - .execute( - builder.buildURI(), - this.httpMethod, - (final ClientHttpRequest requestCallback) -> { - }, - response -> IOUtils.toBufferedInputStream(response.getBody()), - builder.getURIVariables()); - - }); - } - -} +/* + * Copyright (c) 2019 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.service.remote.webservice.api; + +import java.io.InputStream; + +import org.apache.commons.io.IOUtils; +import org.springframework.http.HttpMethod; +import org.springframework.http.MediaType; +import org.springframework.http.client.ClientHttpRequest; + +import ch.ethz.seb.sebserver.gbl.util.Result; + +public abstract class AbstractExportCall extends RestCall { + + protected AbstractExportCall( + final TypeKey typeKey, + final HttpMethod httpMethod, + final MediaType contentType, + final String path) { + + super(typeKey, httpMethod, contentType, path); + } + + @Override + protected Result exchange(final RestCallBuilder builder) { + + return Result.tryCatch(() -> builder + .getRestTemplate() + .execute( + builder.buildURI(), + this.httpMethod, + (final ClientHttpRequest requestCallback) -> { + }, + response -> IOUtils.toBufferedInputStream(response.getBody()), + builder.getURIVariables())); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/FormBinding.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/FormBinding.java index b6e654f4..f6ee3fd6 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/FormBinding.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/FormBinding.java @@ -1,17 +1,25 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api; - -public interface FormBinding { - - String getFormAsJson(); - - String getFormUrlEncoded(); - -} +/* + * Copyright (c) 2019 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.service.remote.webservice.api; + +/** Defines a form binding to get form parameter and values either in JSON format + * or as form-url-encoded string */ +public interface FormBinding { + + /** Get the form parameter and values in JSON format + * + * @return the form parameter and values in JSON format */ + String getFormAsJson(); + + /** Get the form parameter and values in form-url-encoded string format. + * + * @return the form parameter and values in form-url-encoded string format.*/ + String getFormUrlEncoded(); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java index 9895de7e..bf30909b 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java @@ -1,422 +1,420 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api; - -import java.io.IOException; -import java.io.InputStream; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.function.Function; - -import org.apache.commons.lang3.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.core.io.InputStreamResource; -import org.springframework.http.HttpEntity; -import org.springframework.http.HttpHeaders; -import org.springframework.http.HttpMethod; -import org.springframework.http.HttpStatus; -import org.springframework.http.MediaType; -import org.springframework.http.ResponseEntity; -import org.springframework.util.LinkedMultiValueMap; -import org.springframework.util.MultiValueMap; -import org.springframework.web.client.RestClientResponseException; -import org.springframework.web.client.RestTemplate; -import org.springframework.web.util.UriComponentsBuilder; - -import com.fasterxml.jackson.core.JsonParseException; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.JsonMappingException; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.APIMessage; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.api.JSONMapper; -import ch.ethz.seb.sebserver.gbl.model.Entity; -import ch.ethz.seb.sebserver.gbl.model.Page; -import ch.ethz.seb.sebserver.gbl.model.PageSortOrder; -import ch.ethz.seb.sebserver.gbl.util.Result; -import ch.ethz.seb.sebserver.gbl.util.Utils; - -public abstract class RestCall { - - private static final Logger log = LoggerFactory.getLogger(RestCall.class); - - public enum CallType { - UNDEFINED, - GET_SINGLE, - GET_PAGE, - GET_NAMES, - GET_DEPENDENCIES, - GET_LIST, - NEW, - REGISTER, - SAVE, - DELETE, - ACTIVATION_ACTIVATE, - ACTIVATION_DEACTIVATE - } - - protected RestService restService; - protected JSONMapper jsonMapper; - protected TypeKey typeKey; - protected final HttpMethod httpMethod; - protected final MediaType contentType; - protected final String path; - - protected RestCall( - final TypeKey typeKey, - final HttpMethod httpMethod, - final MediaType contentType, - final String path) { - - this.typeKey = typeKey; - this.httpMethod = httpMethod; - this.contentType = contentType; - this.path = path; - - } - - protected RestCall init( - final RestService restService, - final JSONMapper jsonMapper) { - - this.restService = restService; - this.jsonMapper = jsonMapper; - return this; - } - - public EntityType getEntityType() { - if (this.typeKey != null) { - return this.typeKey.entityType; - } - - return null; - } - - protected Result exchange(final RestCallBuilder builder) { - - log.debug("Call webservice API on {} for {}", this.path, builder); - - try { - final ResponseEntity responseEntity = builder.restTemplate - .exchange( - builder.buildURI(), - this.httpMethod, - builder.buildRequestEntity(), - String.class, - builder.uriVariables); - - if (responseEntity.getStatusCode() == HttpStatus.OK) { - - if (log.isTraceEnabled()) { - log.trace("response body --> {}" + responseEntity.getBody()); - } - - if (!responseEntity.hasBody()) { - return Result.ofEmpty(); - } - - return Result.of(RestCall.this.jsonMapper.readValue( - responseEntity.getBody(), - RestCall.this.typeKey.typeRef)); - - } else { - return handleRestCallError(responseEntity); - } - } catch (final RestClientResponseException responseError) { - - final RestCallError restCallError = new RestCallError("Unexpected error while rest call", responseError); - try { - - final String responseBody = responseError.getResponseBodyAsString(); - restCallError.errors.addAll(RestCall.this.jsonMapper.readValue( - responseBody, - new TypeReference>() { - })); - - } catch (final IOException e) { - restCallError.errors.add(APIMessage.ErrorMessage.UNEXPECTED.of( - responseError, - "NO RESPONSE AVAILABLE" + " cause: " + e.getMessage(), - String.valueOf(builder))); - } - - return Result.ofError(restCallError); - } catch (final Exception e) { - final RestCallError restCallError = new RestCallError("Unexpected error while rest call", e); - restCallError.errors.add(APIMessage.ErrorMessage.UNEXPECTED.of( - e, - "NO RESPONSE AVAILABLE", - String.valueOf(builder))); - return Result.ofError(e); - } - } - - public RestCallBuilder newBuilder() { - return new RestCallBuilder( - this.restService.getWebserviceAPIRestTemplate(), - this.restService.getWebserviceURIBuilder()); - } - - public RestCall.RestCallBuilder newBuilder(final RestCall.RestCallBuilder builder) { - return new RestCallBuilder(builder); - } - - private Result handleRestCallError(final ResponseEntity responseEntity) - throws IOException, JsonParseException, JsonMappingException { - - final RestCallError restCallError = - new RestCallError("Response Entity: " + responseEntity.toString()); - - try { - restCallError.errors.addAll(RestCall.this.jsonMapper.readValue( - responseEntity.getBody(), - new TypeReference>() { - })); - } catch (final JsonParseException jpe) { - if (responseEntity.getStatusCode() == HttpStatus.UNAUTHORIZED) { - restCallError.errors.add(APIMessage.ErrorMessage.UNAUTHORIZED.of(responseEntity.getBody())); - } else { - restCallError.errors.add(APIMessage.ErrorMessage.GENERIC.of(responseEntity.getBody())); - } - } - - log.debug( - "Webservice answered with well defined error- or validation-failure-response: ", - restCallError); - - return Result.ofError(restCallError); - } - - public class RestCallBuilder { - - private RestTemplate restTemplate; - private UriComponentsBuilder uriComponentsBuilder; - private final HttpHeaders httpHeaders; - private String body = null; - private InputStream streamingBody = null; - - private final MultiValueMap queryParams; - private final Map uriVariables; - - protected RestCallBuilder(final RestTemplate restTemplate, final UriComponentsBuilder uriComponentsBuilder) { - this.restTemplate = restTemplate; - this.uriComponentsBuilder = uriComponentsBuilder; - this.httpHeaders = new HttpHeaders(); - this.queryParams = new LinkedMultiValueMap<>(); - this.uriVariables = new HashMap<>(); - this.httpHeaders.set( - HttpHeaders.CONTENT_TYPE, - RestCall.this.contentType.toString()); - } - - public RestCallBuilder(final RestCall.RestCallBuilder builder) { - this.restTemplate = builder.restTemplate; - this.uriComponentsBuilder = builder.uriComponentsBuilder; - this.httpHeaders = builder.httpHeaders; - this.body = builder.body; - this.queryParams = new LinkedMultiValueMap<>(builder.queryParams); - this.uriVariables = new HashMap<>(builder.uriVariables); - } - - public RestTemplate getRestTemplate() { - return this.restTemplate; - } - - public RestCallBuilder withRestTemplate(final RestTemplate restTemplate) { - this.restTemplate = restTemplate; - return this; - } - - public RestCallBuilder withUriComponentsBuilder(final UriComponentsBuilder uriComponentsBuilder) { - this.uriComponentsBuilder = uriComponentsBuilder; - return this; - } - - public RestCallBuilder withHeaders(final HttpHeaders headers) { - this.httpHeaders.addAll(headers); - return this; - } - - public RestCallBuilder withHeader(final String name, final String value) { - this.httpHeaders.set(name, value); - return this; - } - - public RestCallBuilder withHeaders(final MultiValueMap params) { - this.httpHeaders.addAll(params); - return this; - } - - public RestCallBuilder apply(final Function f) { - return f.apply(this); - } - - public RestCallBuilder withBody(final Object body) { - if (body == null) { - this.body = null; - return this; - } - - if (body instanceof String) { - this.body = String.valueOf(body); - return this; - } - - if (body instanceof InputStream) { - this.streamingBody = (InputStream) body; - return this; - } - - try { - this.body = RestCall.this.jsonMapper.writeValueAsString(body); - } catch (final JsonProcessingException e) { - log.error("Error while trying to parse body json object: " + body); - } - - return this; - } - - public RestCallBuilder withURIVariable(final String name, final String value) { - this.uriVariables.put(name, value); - return this; - } - - public RestCallBuilder withQueryParam(final String name, final String value) { - this.queryParams.put(name, Arrays.asList(value)); - return this; - } - - public RestCallBuilder withQueryParams(final MultiValueMap params) { - if (params != null) { - this.queryParams.putAll(params); - } - return this; - } - - public RestCallBuilder withFormParam(final String name, final String value) { - final String encodedVal = Utils.encodeFormURL_UTF_8(value); - if (StringUtils.isBlank(this.body)) { - this.body = name + Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR + encodedVal; - } else { - this.body = this.body + Constants.FORM_URL_ENCODED_SEPARATOR + name + - Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR + encodedVal; - } - - return this; - } - - public RestCallBuilder withPaging(final int pageNumber, final int pageSize) { - this.queryParams.put(Page.ATTR_PAGE_NUMBER, Arrays.asList(String.valueOf(pageNumber))); - this.queryParams.put(Page.ATTR_PAGE_SIZE, Arrays.asList(String.valueOf(pageSize))); - return this; - } - - public RestCallBuilder withSorting(final String column, final PageSortOrder order) { - if (column != null) { - this.queryParams.put(Page.ATTR_SORT, Arrays.asList(order.encode(column))); - } - return this; - } - - public RestCallBuilder withFormBinding(final FormBinding formBinding) { - if (RestCall.this.httpMethod == HttpMethod.PUT) { - return withBody(formBinding.getFormAsJson()); - } else { - this.body = formBinding.getFormUrlEncoded(); - return this; - } - } - - public RestCallBuilder onlyActive(final boolean active) { - this.queryParams.put(Entity.FILTER_ATTR_ACTIVE, Arrays.asList(String.valueOf(active))); - return this; - } - - public final Result call() { - return RestCall.this.exchange(this); - } - - public String buildURI() { - return this.uriComponentsBuilder - .cloneBuilder() - .path(RestCall.this.path) - .queryParams(this.queryParams) - .toUriString(); - } - - public HttpEntity buildRequestEntity() { - if (this.streamingBody != null) { - return new HttpEntity<>(new InputStreamResource(this.streamingBody), this.httpHeaders); - } else if (this.body != null) { - return new HttpEntity<>(this.body, this.httpHeaders); - } else { - return new HttpEntity<>(this.httpHeaders); - } - } - - public Map getURIVariables() { - return Utils.immutableMapOf(this.uriVariables); - } - - @Override - public String toString() { - return "RestCallBuilder [httpHeaders=" + this.httpHeaders + ", body=" + this.body + ", queryParams=" - + this.queryParams - + ", uriVariables=" + this.uriVariables + "]"; - } - - } - - public static final class TypeKey { - final CallType callType; - final EntityType entityType; - private final TypeReference typeRef; - - public TypeKey( - final CallType callType, - final EntityType entityType, - final TypeReference typeRef) { - - this.callType = callType; - this.entityType = entityType; - this.typeRef = typeRef; - } - - @Override - public int hashCode() { - final int prime = 31; - int result = 1; - result = prime * result + ((this.callType == null) ? 0 : this.callType.hashCode()); - result = prime * result + ((this.entityType == null) ? 0 : this.entityType.hashCode()); - return result; - } - - @Override - public boolean equals(final Object obj) { - if (this == obj) - return true; - if (obj == null) - return false; - if (getClass() != obj.getClass()) - return false; - final TypeKey other = (TypeKey) obj; - if (this.callType != other.callType) - return false; - if (this.entityType != other.entityType) - return false; - return true; - } - } - -} +/* + * Copyright (c) 2019 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.service.remote.webservice.api; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.APIMessage; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.api.JSONMapper; +import ch.ethz.seb.sebserver.gbl.model.Entity; +import ch.ethz.seb.sebserver.gbl.model.Page; +import ch.ethz.seb.sebserver.gbl.model.PageSortOrder; +import ch.ethz.seb.sebserver.gbl.util.Result; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import com.fasterxml.jackson.core.JsonParseException; +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.core.io.InputStreamResource; +import org.springframework.http.HttpEntity; +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpMethod; +import org.springframework.http.HttpStatus; +import org.springframework.http.MediaType; +import org.springframework.http.ResponseEntity; +import org.springframework.util.LinkedMultiValueMap; +import org.springframework.util.MultiValueMap; +import org.springframework.web.client.RestClientResponseException; +import org.springframework.web.client.RestTemplate; +import org.springframework.web.util.UriComponentsBuilder; + +import java.io.IOException; +import java.io.InputStream; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.Function; + +public abstract class RestCall { + + private static final Logger log = LoggerFactory.getLogger(RestCall.class); + + public enum CallType { + UNDEFINED, + GET_SINGLE, + GET_PAGE, + GET_NAMES, + GET_DEPENDENCIES, + GET_LIST, + NEW, + REGISTER, + SAVE, + DELETE, + ACTIVATION_ACTIVATE, + ACTIVATION_DEACTIVATE + } + + protected RestService restService; + protected JSONMapper jsonMapper; + protected TypeKey typeKey; + protected final HttpMethod httpMethod; + protected final MediaType contentType; + protected final String path; + + protected RestCall( + final TypeKey typeKey, + final HttpMethod httpMethod, + final MediaType contentType, + final String path) { + + this.typeKey = typeKey; + this.httpMethod = httpMethod; + this.contentType = contentType; + this.path = path; + + } + + protected RestCall init( + final RestService restService, + final JSONMapper jsonMapper) { + + this.restService = restService; + this.jsonMapper = jsonMapper; + return this; + } + + public EntityType getEntityType() { + if (this.typeKey != null) { + return this.typeKey.entityType; + } + + return null; + } + + protected Result exchange(final RestCallBuilder builder) { + + log.debug("Call webservice API on {} for {}", this.path, builder); + + try { + final ResponseEntity responseEntity = builder.restTemplate + .exchange( + builder.buildURI(), + this.httpMethod, + builder.buildRequestEntity(), + String.class, + builder.uriVariables); + + if (responseEntity.getStatusCode() == HttpStatus.OK) { + + if (log.isTraceEnabled()) { + log.trace("response body --> {}" + responseEntity.getBody()); + } + + if (!responseEntity.hasBody()) { + return Result.ofEmpty(); + } + + return Result.of(RestCall.this.jsonMapper.readValue( + responseEntity.getBody(), + RestCall.this.typeKey.typeRef)); + + } else { + return handleRestCallError(responseEntity); + } + } catch (final RestClientResponseException responseError) { + + final RestCallError restCallError = new RestCallError("Unexpected error while rest call", responseError); + try { + + final String responseBody = responseError.getResponseBodyAsString(); + restCallError.errors.addAll(RestCall.this.jsonMapper.readValue( + responseBody, + new TypeReference>() { + })); + + } catch (final IOException e) { + restCallError.errors.add(APIMessage.ErrorMessage.UNEXPECTED.of( + responseError, + "NO RESPONSE AVAILABLE" + " cause: " + e.getMessage(), + String.valueOf(builder))); + } + + return Result.ofError(restCallError); + } catch (final Exception e) { + final RestCallError restCallError = new RestCallError("Unexpected error while rest call", e); + restCallError.errors.add(APIMessage.ErrorMessage.UNEXPECTED.of( + e, + "NO RESPONSE AVAILABLE", + String.valueOf(builder))); + return Result.ofError(e); + } + } + + public RestCallBuilder newBuilder() { + return new RestCallBuilder( + this.restService.getWebserviceAPIRestTemplate(), + this.restService.getWebserviceURIBuilder()); + } + + public RestCall.RestCallBuilder newBuilder(final RestCall.RestCallBuilder builder) { + return new RestCallBuilder(builder); + } + + private Result handleRestCallError(final ResponseEntity responseEntity) + throws IOException { + + final RestCallError restCallError = + new RestCallError("Response Entity: " + responseEntity.toString()); + + try { + restCallError.errors.addAll(RestCall.this.jsonMapper.readValue( + responseEntity.getBody(), + new TypeReference>() { + })); + } catch (final JsonParseException jpe) { + if (responseEntity.getStatusCode() == HttpStatus.UNAUTHORIZED) { + restCallError.errors.add(APIMessage.ErrorMessage.UNAUTHORIZED.of(responseEntity.getBody())); + } else { + restCallError.errors.add(APIMessage.ErrorMessage.GENERIC.of(responseEntity.getBody())); + } + } + + log.debug( + "Webservice answered with well defined error- or validation-failure-response: ", + restCallError); + + return Result.ofError(restCallError); + } + + public class RestCallBuilder { + + private RestTemplate restTemplate; + private UriComponentsBuilder uriComponentsBuilder; + private final HttpHeaders httpHeaders; + private String body = null; + private InputStream streamingBody = null; + + private final MultiValueMap queryParams; + private final Map uriVariables; + + protected RestCallBuilder(final RestTemplate restTemplate, final UriComponentsBuilder uriComponentsBuilder) { + this.restTemplate = restTemplate; + this.uriComponentsBuilder = uriComponentsBuilder; + this.httpHeaders = new HttpHeaders(); + this.queryParams = new LinkedMultiValueMap<>(); + this.uriVariables = new HashMap<>(); + this.httpHeaders.set( + HttpHeaders.CONTENT_TYPE, + RestCall.this.contentType.toString()); + } + + public RestCallBuilder(final RestCall.RestCallBuilder builder) { + this.restTemplate = builder.restTemplate; + this.uriComponentsBuilder = builder.uriComponentsBuilder; + this.httpHeaders = builder.httpHeaders; + this.body = builder.body; + this.streamingBody = builder.streamingBody; + this.queryParams = new LinkedMultiValueMap<>(builder.queryParams); + this.uriVariables = new HashMap<>(builder.uriVariables); + } + + public RestTemplate getRestTemplate() { + return this.restTemplate; + } + + public RestCallBuilder withRestTemplate(final RestTemplate restTemplate) { + this.restTemplate = restTemplate; + return this; + } + + public RestCallBuilder withUriComponentsBuilder(final UriComponentsBuilder uriComponentsBuilder) { + this.uriComponentsBuilder = uriComponentsBuilder; + return this; + } + + public RestCallBuilder withHeaders(final HttpHeaders headers) { + this.httpHeaders.addAll(headers); + return this; + } + + public RestCallBuilder withHeader(final String name, final String value) { + this.httpHeaders.set(name, value); + return this; + } + + public RestCallBuilder withHeaders(final MultiValueMap params) { + this.httpHeaders.addAll(params); + return this; + } + + public RestCallBuilder apply(final Function f) { + return f.apply(this); + } + + public RestCallBuilder withBody(final Object body) { + if (body == null) { + this.body = null; + return this; + } + + if (body instanceof String) { + this.body = String.valueOf(body); + return this; + } + + if (body instanceof InputStream) { + this.streamingBody = (InputStream) body; + return this; + } + + try { + this.body = RestCall.this.jsonMapper.writeValueAsString(body); + } catch (final JsonProcessingException e) { + log.error("Error while trying to parse body json object: " + body); + } + + return this; + } + + public RestCallBuilder withURIVariable(final String name, final String value) { + this.uriVariables.put(name, value); + return this; + } + + public RestCallBuilder withQueryParam(final String name, final String value) { + this.queryParams.put(name, Arrays.asList(value)); + return this; + } + + public RestCallBuilder withQueryParams(final MultiValueMap params) { + if (params != null) { + this.queryParams.putAll(params); + } + return this; + } + + public RestCallBuilder withFormParam(final String name, final String value) { + final String encodedVal = Utils.encodeFormURL_UTF_8(value); + if (StringUtils.isBlank(this.body)) { + this.body = name + Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR + encodedVal; + } else { + this.body = this.body + Constants.FORM_URL_ENCODED_SEPARATOR + name + + Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR + encodedVal; + } + + return this; + } + + public RestCallBuilder withPaging(final int pageNumber, final int pageSize) { + this.queryParams.put(Page.ATTR_PAGE_NUMBER, Arrays.asList(String.valueOf(pageNumber))); + this.queryParams.put(Page.ATTR_PAGE_SIZE, Arrays.asList(String.valueOf(pageSize))); + return this; + } + + public RestCallBuilder withSorting(final String column, final PageSortOrder order) { + if (column != null) { + this.queryParams.put(Page.ATTR_SORT, Arrays.asList(order.encode(column))); + } + return this; + } + + public RestCallBuilder withFormBinding(final FormBinding formBinding) { + if (RestCall.this.httpMethod == HttpMethod.PUT) { + return withBody(formBinding.getFormAsJson()); + } else { + this.body = formBinding.getFormUrlEncoded(); + return this; + } + } + + public RestCallBuilder onlyActive(final boolean active) { + this.queryParams.put(Entity.FILTER_ATTR_ACTIVE, Arrays.asList(String.valueOf(active))); + return this; + } + + public final Result call() { + return RestCall.this.exchange(this); + } + + public String buildURI() { + return this.uriComponentsBuilder + .cloneBuilder() + .path(RestCall.this.path) + .queryParams(this.queryParams) + .toUriString(); + } + + public HttpEntity buildRequestEntity() { + if (this.streamingBody != null) { + return new HttpEntity<>(new InputStreamResource(this.streamingBody), this.httpHeaders); + } else if (this.body != null) { + return new HttpEntity<>(this.body, this.httpHeaders); + } else { + return new HttpEntity<>(this.httpHeaders); + } + } + + public Map getURIVariables() { + return Utils.immutableMapOf(this.uriVariables); + } + + @Override + public String toString() { + return "RestCallBuilder [httpHeaders=" + this.httpHeaders + ", body=" + this.body + ", queryParams=" + + this.queryParams + + ", uriVariables=" + this.uriVariables + "]"; + } + + } + + public static final class TypeKey { + final CallType callType; + final EntityType entityType; + private final TypeReference typeRef; + + public TypeKey( + final CallType callType, + final EntityType entityType, + final TypeReference typeRef) { + + this.callType = callType; + this.entityType = entityType; + this.typeRef = typeRef; + } + + @Override + public int hashCode() { + final int prime = 31; + int result = 1; + result = prime * result + ((this.callType == null) ? 0 : this.callType.hashCode()); + result = prime * result + ((this.entityType == null) ? 0 : this.entityType.hashCode()); + return result; + } + + @Override + public boolean equals(final Object obj) { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + final TypeKey other = (TypeKey) obj; + if (this.callType != other.callType) + return false; + if (this.entityType != other.entityType) + return false; + return true; + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCallError.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCallError.java index 0838def6..ed71d178 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCallError.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCallError.java @@ -1,61 +1,59 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; - -import ch.ethz.seb.sebserver.gbl.api.APIMessage; -import ch.ethz.seb.sebserver.gbl.api.APIMessageError; -import ch.ethz.seb.sebserver.gbl.util.Utils; - -public class RestCallError extends RuntimeException implements APIMessageError { - - private static final long serialVersionUID = -5201349295667957490L; - - final List errors; - - public RestCallError(final String message, final Throwable cause) { - super(message, cause); - this.errors = new ArrayList<>(); - } - - public RestCallError(final String message, final Collection apiErrors) { - super(message); - this.errors = Utils.immutableListOf(apiErrors); - } - - public RestCallError(final String message) { - super(message); - this.errors = new ArrayList<>(); - } - - @Override - public List getErrorMessages() { - return this.errors; - } - - public boolean hasErrorMessages() { - return !this.errors.isEmpty(); - } - - public boolean isFieldValidationError() { - return this.errors - .stream() - .filter(error -> APIMessage.ErrorMessage.FIELD_VALIDATION.isOf(error)) - .findFirst() - .isPresent(); - } - - @Override - public String toString() { - return "RestCallError [errors=" + this.errors + "]"; - } -} +/* + * Copyright (c) 2019 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.service.remote.webservice.api; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +import ch.ethz.seb.sebserver.gbl.api.APIMessage; +import ch.ethz.seb.sebserver.gbl.api.APIMessageError; +import ch.ethz.seb.sebserver.gbl.util.Utils; + +public class RestCallError extends RuntimeException implements APIMessageError { + + private static final long serialVersionUID = -5201349295667957490L; + + final List errors; + + public RestCallError(final String message, final Throwable cause) { + super(message, cause); + this.errors = new ArrayList<>(); + } + + public RestCallError(final String message, final Collection apiErrors) { + super(message); + this.errors = Utils.immutableListOf(apiErrors); + } + + public RestCallError(final String message) { + super(message); + this.errors = new ArrayList<>(); + } + + @Override + public List getErrorMessages() { + return this.errors; + } + + public boolean hasErrorMessages() { + return !this.errors.isEmpty(); + } + + public boolean isFieldValidationError() { + return this.errors + .stream() + .anyMatch(APIMessage.ErrorMessage.FIELD_VALIDATION::isOf); + } + + @Override + public String toString() { + return "RestCallError [errors=" + this.errors + "]"; + } +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestService.java index a6d63ec8..693a267c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestService.java @@ -1,74 +1,74 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api; - -import org.springframework.web.client.RestTemplate; -import org.springframework.web.util.UriComponentsBuilder; - -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall.CallType; - -/** Interface to SEB Server webservice API thought RestCall's - * or thought Spring's RestTemplate on lower level. - * - * A RestCall's can be used to call a specified SEB Server webservice API endpoint with given request parameter. - * This service collects all the available RestCalls and map them by Class type or EntityType and CallType. - * - * For Example if one want to get a certain User-Account by API request on SEB Server webservice API: - * - *
- *  UserInfo userInfo = RestService.getBuilder(GetUserAccount.class)
- *      .withURIVariable(API.PARAM_MODEL_ID, user-account-id)           adds an URI path variable
- *      .withQueryParam(API.PARAM_INSTITUTION_ID, institutionId)        adds a URI query parameter
- *      .call()                                                         executes the API request call
- *      .get(pageContext::notifyError)                                  gets the result or notify an error to the user if happened
- * 
- */ -public interface RestService { - - /** Get Spring's RestTemplate that is used within this service. - * - * @return Spring's RestTemplate that is used within this service. */ - RestTemplate getWebserviceAPIRestTemplate(); - - /** Get Spring's UriComponentsBuilder that is used within this service. - * - * @return Spring's UriComponentsBuilder that is used within this service. */ - UriComponentsBuilder getWebserviceURIBuilder(); - - /** Get a certain RestCall by Class type. - * - * @param type the Class type of the RestCall - * @return RestCall instance */ - RestCall getRestCall(Class> type); - - /** Get a certain RestCall by EntityType and CallType. - * NOTE not all RestCall can be get within this method. Only the ones that have a defined CallType - * - * @param entityType The EntityType of the RestCall - * @param callType The CallType of the RestCall (not UNDEFINDED) - * @return RestCall instance */ - RestCall getRestCall(EntityType entityType, CallType callType); - - /** Get a certain RestCallBuilder by RestCall Class type. - * - * @param type the Class type of the RestCall - * @return RestCallBuilder instance to build a dedicated call and execute it */ - RestCall.RestCallBuilder getBuilder(Class> type); - - /** Get a certain RestCallBuilder by EntityType and CallType. - * - * @param entityType The EntityType of the RestCall to get a builder for - * @param callType The CallType of the RestCall to get a builder for (not UNDEFINDED) - * @return RestCallBuilder instance to build a dedicated call and execute it */ - RestCall.RestCallBuilder getBuilder( - EntityType entityType, - CallType callType); - +/* + * Copyright (c) 2019 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.service.remote.webservice.api; + +import org.springframework.web.client.RestTemplate; +import org.springframework.web.util.UriComponentsBuilder; + +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall.CallType; + +/** Interface to SEB Server webservice API thought RestCall's + * or thought Spring's RestTemplate on lower level. + * + * A RestCall's can be used to call a specified SEB Server webservice API endpoint with given request parameter. + * This service collects all the available RestCalls and map them by Class type or EntityType and CallType. + * + * For Example if one want to get a certain User-Account by API request on SEB Server webservice API: + * + *
+ *  UserInfo userInfo = RestService.getBuilder(GetUserAccount.class)
+ *      .withURIVariable(API.PARAM_MODEL_ID, user-account-id)           adds an URI path variable
+ *      .withQueryParam(API.PARAM_INSTITUTION_ID, institutionId)        adds a URI query parameter
+ *      .call()                                                         executes the API request call
+ *      .get(pageContext::notifyError)                                  gets the result or notify an error to the user if happened
+ * 
+ */ +public interface RestService { + + /** Get Spring's RestTemplate that is used within this service. + * + * @return Spring's RestTemplate that is used within this service. */ + RestTemplate getWebserviceAPIRestTemplate(); + + /** Get Spring's UriComponentsBuilder that is used within this service. + * + * @return Spring's UriComponentsBuilder that is used within this service. */ + UriComponentsBuilder getWebserviceURIBuilder(); + + /** Get a certain RestCall by Class type. + * + * @param type the Class type of the RestCall + * @return RestCall instance */ + RestCall getRestCall(Class> type); + + /** Get a certain RestCall by EntityType and CallType. + * NOTE not all RestCall can be get within this method. Only the ones that have a defined CallType + * + * @param entityType The EntityType of the RestCall + * @param callType The CallType of the RestCall (not UNDEFINED) + * @return RestCall instance */ + RestCall getRestCall(EntityType entityType, CallType callType); + + /** Get a certain RestCallBuilder by RestCall Class type. + * + * @param type the Class type of the RestCall + * @return RestCallBuilder instance to build a dedicated call and execute it */ + RestCall.RestCallBuilder getBuilder(Class> type); + + /** Get a certain RestCallBuilder by EntityType and CallType. + * + * @param entityType The EntityType of the RestCall to get a builder for + * @param callType The CallType of the RestCall to get a builder for (not UNDEFINED) + * @return RestCallBuilder instance to build a dedicated call and execute it */ + RestCall.RestCallBuilder getBuilder( + EntityType entityType, + CallType callType); + } \ No newline at end of file diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/useraccount/SaveUserAccount.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/useraccount/SaveUserAccount.java index 6007a241..ee16e0ac 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/useraccount/SaveUserAccount.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/useraccount/SaveUserAccount.java @@ -1,40 +1,41 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.api.useraccount; - -import org.springframework.context.annotation.Lazy; -import org.springframework.http.HttpMethod; -import org.springframework.http.MediaType; -import org.springframework.stereotype.Component; - -import com.fasterxml.jackson.core.type.TypeReference; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; - -@Lazy -@Component -@GuiProfile -public class SaveUserAccount extends RestCall { - - public SaveUserAccount() { - super(new TypeKey<>( - CallType.SAVE, - EntityType.USER, - new TypeReference() { - }), - HttpMethod.PUT, - MediaType.APPLICATION_JSON_UTF8, - API.USER_ACCOUNT_ENDPOINT); - } - -} +/* + * Copyright (c) 2019 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.service.remote.webservice.api.useraccount; + +import org.springframework.context.annotation.Lazy; +import org.springframework.http.HttpMethod; +import org.springframework.http.MediaType; +import org.springframework.stereotype.Component; + +import com.fasterxml.jackson.core.type.TypeReference; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; + + +@Lazy +@Component +@GuiProfile +public class SaveUserAccount extends RestCall { + + public SaveUserAccount() { + super(new TypeKey<>( + CallType.SAVE, + EntityType.USER, + new TypeReference() { + }), + HttpMethod.PUT, + MediaType.APPLICATION_JSON_UTF8, + API.USER_ACCOUNT_ENDPOINT); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/CurrentUser.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/CurrentUser.java index 37bda322..f4d489f6 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/CurrentUser.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/CurrentUser.java @@ -1,348 +1,344 @@ -/* - * 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.service.remote.webservice.auth; - -import java.util.Collection; -import java.util.HashMap; -import java.util.Map; -import java.util.function.Function; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Scope; -import org.springframework.context.annotation.ScopedProxyMode; -import org.springframework.http.HttpEntity; -import org.springframework.http.HttpMethod; -import org.springframework.http.HttpStatus; -import org.springframework.http.ResponseEntity; -import org.springframework.stereotype.Component; -import org.springframework.web.context.WebApplicationContext; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.api.authorization.Privilege; -import ch.ethz.seb.sebserver.gbl.api.authorization.Privilege.RoleTypeKey; -import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; -import ch.ethz.seb.sebserver.gbl.model.GrantEntity; -import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; -import ch.ethz.seb.sebserver.gbl.model.user.UserRole; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; - -@Component -@GuiProfile -@Scope(value = WebApplicationContext.SCOPE_SESSION, proxyMode = ScopedProxyMode.TARGET_CLASS) -public class CurrentUser { - - private static final Logger log = LoggerFactory.getLogger(CurrentUser.class); - - private final AuthorizationContextHolder authorizationContextHolder; - private SEBServerAuthorizationContext authContext = null; - private Map privileges = null; - private final Map attributes; - - public CurrentUser(final AuthorizationContextHolder authorizationContextHolder) { - this.authorizationContextHolder = authorizationContextHolder; - this.attributes = new HashMap<>(); - } - - public void putAttribute(final String name, final String value) { - this.attributes.put(name, value); - } - - public String getAttribute(final String name) { - return this.attributes.get(name); - } - - public AuthorizationContextHolder getAuthorizationContextHolder() { - return this.authorizationContextHolder; - } - - public UserInfo get() { - - if (isAvailable()) { - return this.authContext - .getLoggedInUser() - .getOrThrow(); - } - - return handleIllegalSessionState(); - } - - public UserInfo getOrHandleError(final Function errorHandler) { - if (isAvailable()) { - return this.authContext - .getLoggedInUser() - .get(errorHandler); - } - - return handleIllegalSessionState(); - } - - private UserInfo handleIllegalSessionState() { - log.warn("Current user requested but no user is currently logged in"); - this.logout(); - throw new IllegalUserSessionStateException("User not logged in"); - } - - public GrantCheck grantCheck(final EntityType entityType) { - return new GrantCheck(entityType); - } - - public EntityGrantCheck entityGrantCheck(final GrantEntity grantEntity) { - return new EntityGrantCheck(grantEntity); - } - - public boolean hasBasePrivilege(final PrivilegeType privilegeType, final EntityType entityType) { - return hasPrivilege(privilegeType, entityType, null, null); - } - - public boolean hasInstitutionalPrivilege(final PrivilegeType privilegeType, final EntityType entityType) { - final UserInfo userInfo = get(); - return hasPrivilege(privilegeType, entityType, userInfo.institutionId, null); - } - - public boolean hasPrivilege( - final PrivilegeType privilegeType, - final EntityType entityType, - final Long institutionId, - final String ownerId) { - if (loadPrivileges()) { - try { - final UserInfo userInfo = get(); - return userInfo - .getRoles() - .stream() - .map(roleName -> UserRole.valueOf(roleName)) - .map(role -> new RoleTypeKey(entityType, role)) - .map(key -> this.privileges.get(key)) - .filter(priv -> (priv != null) && priv.hasGrant( - userInfo.uuid, - userInfo.institutionId, - privilegeType, - institutionId, - ownerId)) - .findFirst() - .isPresent(); - } catch (final Exception e) { - log.error("Failed to verify privilege: PrivilegeType {} EntityType {}", - privilegeType, entityType, e); - } - } - - return false; - } - - public boolean hasPrivilege( - final PrivilegeType privilegeType, - final GrantEntity grantEntity) { - - if (loadPrivileges()) { - final EntityType entityType = grantEntity.entityType(); - try { - final UserInfo userInfo = get(); - return userInfo.getRoles() - .stream() - .map(roleName -> UserRole.valueOf(roleName)) - .map(role -> new RoleTypeKey(entityType, role)) - .map(key -> this.privileges.get(key)) - .filter(priv -> (priv != null) && priv.hasGrant( - userInfo.uuid, - userInfo.institutionId, - privilegeType, - grantEntity.getInstitutionId(), - grantEntity.getOwnerId())) - .findFirst() - .isPresent(); - } catch (final Exception e) { - log.error("Failed to verify privilege: PrivilegeType {} EntityType {}", - privilegeType, entityType, e); - } - } - - return false; - } - - public boolean isAvailable() { - updateContext(); - return this.authContext != null && this.authContext.isLoggedIn(); - } - - public void refresh(final UserInfo userInfo) { - this.authContext.refreshUser(userInfo); - } - - public boolean logout() { - if (this.attributes != null) { - this.attributes.clear(); - } - - this.privileges = null; - - if (isAvailable()) { - if (this.authContext.logout()) { - this.authContext = null; - return true; - } else { - return false; - } - } else { - try { - return this.authorizationContextHolder - .getAuthorizationContext() - .logout(); - } catch (final Exception e) { - log.warn("Unexpected error while logout: {}", e.getMessage()); - return false; - } - } - } - - private void updateContext() { - if (this.authContext == null || !this.authContext.isValid()) { - this.authContext = this.authorizationContextHolder.getAuthorizationContext(); - } - } - - private boolean loadPrivileges() { - if (this.privileges != null) { - return true; - } - - updateContext(); - if (this.authContext != null) { - try { - final WebserviceURIService webserviceURIService = - this.authorizationContextHolder.getWebserviceURIService(); - final ResponseEntity> exchange = this.authContext.getRestTemplate() - .exchange( - webserviceURIService.getURIBuilder() - .path(API.PRIVILEGES_ENDPOINT) - .toUriString(), - HttpMethod.GET, - HttpEntity.EMPTY, - Constants.TYPE_REFERENCE_PRIVILEGES); - - if (exchange.getStatusCodeValue() == HttpStatus.OK.value()) { - final Collection privileges = exchange.getBody(); - if (privileges != null) { - this.privileges = privileges - .stream() - .reduce(new HashMap(), - (map, priv) -> { - map.put(priv.roleTypeKey, priv); - return map; - }, - (map1, map2) -> { - map1.putAll(map2); - return map1; - }); - } else { - log.error("Failed to get Privileges from webservice API: {}", exchange); - return false; - } - - return true; - } else { - log.error("Failed to get Privileges from webservice API: {}", exchange); - return false; - } - - } catch (final Exception e) { - log.error("Failed to get Privileges from webservice API: ", e); - return false; - } - } else { - log.error("Failed to get Privileges from webservice API. No AuthorizationContext available"); - return false; - } - } - - /** Wrapper can be used for base and institutional grant checks for a specified EntityType */ - public class GrantCheck { - private final EntityType entityType; - - protected GrantCheck(final EntityType entityType) { - this.entityType = entityType; - } - - /** Checks the base read-only privilege grant - * - * @return true on read-only privilege grant on wrapped EntityType */ - public boolean r() { - return hasBasePrivilege(PrivilegeType.READ, this.entityType); - } - - /** Checks the base modify privilege grant - * - * @return true on modify privilege grant on wrapped EntityType */ - public boolean m() { - return hasBasePrivilege(PrivilegeType.MODIFY, this.entityType); - } - - /** Checks the base write privilege grant - * - * @return true on write privilege grant on wrapped EntityType */ - public boolean w() { - return hasBasePrivilege(PrivilegeType.WRITE, this.entityType); - } - - /** Checks the institutional read-only privilege grant - * - * @return true institutional read-only privilege grant on wrapped EntityType */ - public boolean ir() { - return hasInstitutionalPrivilege(PrivilegeType.READ, this.entityType); - } - - /** Checks the institutional modify privilege grant - * - * @return true institutional modify privilege grant on wrapped EntityType */ - public boolean im() { - return hasInstitutionalPrivilege(PrivilegeType.MODIFY, this.entityType); - } - - /** Checks the institutional write privilege grant - * - * @return true institutional write privilege grant on wrapped EntityType */ - public boolean iw() { - return hasInstitutionalPrivilege(PrivilegeType.WRITE, this.entityType); - } - } - - /** Wrapper can be used for Entity based grant checks */ - public class EntityGrantCheck { - private final GrantEntity grantEntity; - - protected EntityGrantCheck(final GrantEntity grantEntity) { - this.grantEntity = grantEntity; - } - - /** Checks the read-only privilege grant for wrapped grantEntity - * - * @return true on read-only privilege grant for wrapped grantEntity */ - public boolean r() { - return hasPrivilege(PrivilegeType.READ, this.grantEntity); - } - - /** Checks the modify privilege grant for wrapped grantEntity - * - * @return true on modify privilege grant for wrapped grantEntity */ - public boolean m() { - return hasPrivilege(PrivilegeType.MODIFY, this.grantEntity); - } - - /** Checks the write privilege grant for wrapped grantEntity - * - * @return true on write privilege grant for wrapped grantEntity */ - public boolean w() { - return hasPrivilege(PrivilegeType.WRITE, this.grantEntity); - } - } - -} +/* + * 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.service.remote.webservice.auth; + +import java.util.Collection; +import java.util.HashMap; +import java.util.Map; +import java.util.function.Function; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Scope; +import org.springframework.context.annotation.ScopedProxyMode; +import org.springframework.http.HttpEntity; +import org.springframework.http.HttpMethod; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.context.WebApplicationContext; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.api.authorization.Privilege; +import ch.ethz.seb.sebserver.gbl.api.authorization.Privilege.RoleTypeKey; +import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; +import ch.ethz.seb.sebserver.gbl.model.GrantEntity; +import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; +import ch.ethz.seb.sebserver.gbl.model.user.UserRole; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; + +@Component +@GuiProfile +@Scope(value = WebApplicationContext.SCOPE_SESSION, proxyMode = ScopedProxyMode.TARGET_CLASS) +public class CurrentUser { + + private static final Logger log = LoggerFactory.getLogger(CurrentUser.class); + + private final AuthorizationContextHolder authorizationContextHolder; + private SEBServerAuthorizationContext authContext = null; + private Map privileges = null; + private final Map attributes; + + public CurrentUser(final AuthorizationContextHolder authorizationContextHolder) { + this.authorizationContextHolder = authorizationContextHolder; + this.attributes = new HashMap<>(); + } + + public void putAttribute(final String name, final String value) { + this.attributes.put(name, value); + } + + public String getAttribute(final String name) { + return this.attributes.get(name); + } + + public AuthorizationContextHolder getAuthorizationContextHolder() { + return this.authorizationContextHolder; + } + + public UserInfo get() { + + if (isAvailable()) { + return this.authContext + .getLoggedInUser() + .getOrThrow(); + } + + return handleIllegalSessionState(); + } + + public UserInfo getOrHandleError(final Function errorHandler) { + if (isAvailable()) { + return this.authContext + .getLoggedInUser() + .get(errorHandler); + } + + return handleIllegalSessionState(); + } + + private UserInfo handleIllegalSessionState() { + log.warn("Current user requested but no user is currently logged in"); + this.logout(); + throw new IllegalUserSessionStateException("User not logged in"); + } + + public GrantCheck grantCheck(final EntityType entityType) { + return new GrantCheck(entityType); + } + + public EntityGrantCheck entityGrantCheck(final GrantEntity grantEntity) { + return new EntityGrantCheck(grantEntity); + } + + public boolean hasBasePrivilege(final PrivilegeType privilegeType, final EntityType entityType) { + return hasPrivilege(privilegeType, entityType, null, null); + } + + public boolean hasInstitutionalPrivilege(final PrivilegeType privilegeType, final EntityType entityType) { + final UserInfo userInfo = get(); + return hasPrivilege(privilegeType, entityType, userInfo.institutionId, null); + } + + public boolean hasPrivilege( + final PrivilegeType privilegeType, + final EntityType entityType, + final Long institutionId, + final String ownerId) { + if (loadPrivileges()) { + try { + final UserInfo userInfo = get(); + return userInfo + .getRoles() + .stream() + .map(UserRole::valueOf) + .map(role -> new RoleTypeKey(entityType, role)) + .map(key -> this.privileges.get(key)) + .anyMatch(privilege -> (privilege != null) && privilege.hasGrant( + userInfo.uuid, + userInfo.institutionId, + privilegeType, + institutionId, + ownerId)); + } catch (final Exception e) { + log.error("Failed to verify privilege: PrivilegeType {} EntityType {}", + privilegeType, entityType, e); + } + } + + return false; + } + + public boolean hasPrivilege( + final PrivilegeType privilegeType, + final GrantEntity grantEntity) { + + if (loadPrivileges()) { + final EntityType entityType = grantEntity.entityType(); + try { + final UserInfo userInfo = get(); + return userInfo.getRoles() + .stream() + .map(UserRole::valueOf) + .map(role -> new RoleTypeKey(entityType, role)) + .map(key -> this.privileges.get(key)) + .anyMatch(privilege -> (privilege != null) && privilege.hasGrant( + userInfo.uuid, + userInfo.institutionId, + privilegeType, + grantEntity.getInstitutionId(), + grantEntity.getOwnerId())); + } catch (final Exception e) { + log.error("Failed to verify privilege: PrivilegeType {} EntityType {}", + privilegeType, entityType, e); + } + } + + return false; + } + + public boolean isAvailable() { + updateContext(); + return this.authContext != null && this.authContext.isLoggedIn(); + } + + public void refresh(final UserInfo userInfo) { + this.authContext.refreshUser(userInfo); + } + + public boolean logout() { + if (this.attributes != null) { + this.attributes.clear(); + } + + this.privileges = null; + + if (isAvailable()) { + if (this.authContext.logout()) { + this.authContext = null; + return true; + } else { + return false; + } + } else { + try { + return this.authorizationContextHolder + .getAuthorizationContext() + .logout(); + } catch (final Exception e) { + log.warn("Unexpected error while logout: {}", e.getMessage()); + return false; + } + } + } + + private void updateContext() { + if (this.authContext == null || !this.authContext.isValid()) { + this.authContext = this.authorizationContextHolder.getAuthorizationContext(); + } + } + + private boolean loadPrivileges() { + if (this.privileges != null) { + return true; + } + + updateContext(); + if (this.authContext != null) { + try { + final WebserviceURIService webserviceURIService = + this.authorizationContextHolder.getWebserviceURIService(); + final ResponseEntity> exchange = this.authContext.getRestTemplate() + .exchange( + webserviceURIService.getURIBuilder() + .path(API.PRIVILEGES_ENDPOINT) + .toUriString(), + HttpMethod.GET, + HttpEntity.EMPTY, + Constants.TYPE_REFERENCE_PRIVILEGES); + + if (exchange.getStatusCodeValue() == HttpStatus.OK.value()) { + final Collection privileges = exchange.getBody(); + if (privileges != null) { + this.privileges = privileges + .stream() + .reduce(new HashMap<>(), + (map, privilege) -> { + map.put(privilege.roleTypeKey, privilege); + return map; + }, + (map1, map2) -> { + map1.putAll(map2); + return map1; + }); + } else { + log.error("Failed to get Privileges from webservice API: {}", exchange); + return false; + } + + return true; + } else { + log.error("Failed to get Privileges from webservice API: {}", exchange); + return false; + } + + } catch (final Exception e) { + log.error("Failed to get Privileges from webservice API: ", e); + return false; + } + } else { + log.error("Failed to get Privileges from webservice API. No AuthorizationContext available"); + return false; + } + } + + /** Wrapper can be used for base and institutional grant checks for a specified EntityType */ + public class GrantCheck { + private final EntityType entityType; + + protected GrantCheck(final EntityType entityType) { + this.entityType = entityType; + } + + /** Checks the base read-only privilege grant + * + * @return true on read-only privilege grant on wrapped EntityType */ + public boolean r() { + return hasBasePrivilege(PrivilegeType.READ, this.entityType); + } + + /** Checks the base modify privilege grant + * + * @return true on modify privilege grant on wrapped EntityType */ + public boolean m() { + return hasBasePrivilege(PrivilegeType.MODIFY, this.entityType); + } + + /** Checks the base write privilege grant + * + * @return true on write privilege grant on wrapped EntityType */ + public boolean w() { + return hasBasePrivilege(PrivilegeType.WRITE, this.entityType); + } + + /** Checks the institutional read-only privilege grant + * + * @return true institutional read-only privilege grant on wrapped EntityType */ + public boolean ir() { + return hasInstitutionalPrivilege(PrivilegeType.READ, this.entityType); + } + + /** Checks the institutional modify privilege grant + * + * @return true institutional modify privilege grant on wrapped EntityType */ + public boolean im() { + return hasInstitutionalPrivilege(PrivilegeType.MODIFY, this.entityType); + } + + /** Checks the institutional write privilege grant + * + * @return true institutional write privilege grant on wrapped EntityType */ + public boolean iw() { + return hasInstitutionalPrivilege(PrivilegeType.WRITE, this.entityType); + } + } + + /** Wrapper can be used for Entity based grant checks */ + public class EntityGrantCheck { + private final GrantEntity grantEntity; + + protected EntityGrantCheck(final GrantEntity grantEntity) { + this.grantEntity = grantEntity; + } + + /** Checks the read-only privilege grant for wrapped grantEntity + * + * @return true on read-only privilege grant for wrapped grantEntity */ + public boolean r() { + return hasPrivilege(PrivilegeType.READ, this.grantEntity); + } + + /** Checks the modify privilege grant for wrapped grantEntity + * + * @return true on modify privilege grant for wrapped grantEntity */ + public boolean m() { + return hasPrivilege(PrivilegeType.MODIFY, this.grantEntity); + } + + /** Checks the write privilege grant for wrapped grantEntity + * + * @return true on write privilege grant for wrapped grantEntity */ + public boolean w() { + return hasPrivilege(PrivilegeType.WRITE, this.grantEntity); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/OAuth2AuthorizationContextHolder.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/OAuth2AuthorizationContextHolder.java index 717e1644..a972a918 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/OAuth2AuthorizationContextHolder.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/OAuth2AuthorizationContextHolder.java @@ -1,333 +1,331 @@ -/* - * 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.service.remote.webservice.auth; - -import java.io.IOException; -import java.net.URI; -import java.nio.charset.Charset; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; - -import javax.servlet.http.HttpSession; - -import org.apache.commons.lang3.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.annotation.Lazy; -import org.springframework.http.HttpMethod; -import org.springframework.http.HttpStatus; -import org.springframework.http.ResponseEntity; -import org.springframework.http.client.ClientHttpRequestFactory; -import org.springframework.http.client.ClientHttpResponse; -import org.springframework.http.converter.StringHttpMessageConverter; -import org.springframework.security.access.AccessDeniedException; -import org.springframework.security.oauth2.client.DefaultOAuth2ClientContext; -import org.springframework.security.oauth2.client.OAuth2RestTemplate; -import org.springframework.security.oauth2.client.http.OAuth2ErrorHandler; -import org.springframework.security.oauth2.client.resource.OAuth2AccessDeniedException; -import org.springframework.security.oauth2.client.resource.OAuth2ProtectedResourceDetails; -import org.springframework.security.oauth2.client.token.DefaultAccessTokenRequest; -import org.springframework.security.oauth2.client.token.grant.password.ResourceOwnerPasswordResourceDetails; -import org.springframework.security.oauth2.common.OAuth2AccessToken; -import org.springframework.stereotype.Component; -import org.springframework.web.client.RequestCallback; -import org.springframework.web.client.ResponseExtractor; -import org.springframework.web.client.RestClientException; -import org.springframework.web.client.RestTemplate; - -import ch.ethz.seb.sebserver.ClientHttpRequestFactoryService; -import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; -import ch.ethz.seb.sebserver.gbl.model.user.UserRole; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gbl.util.Result; -import ch.ethz.seb.sebserver.gbl.util.Utils; - -@Lazy -@Component -@GuiProfile -public class OAuth2AuthorizationContextHolder implements AuthorizationContextHolder { - - private static final Logger log = LoggerFactory.getLogger(OAuth2AuthorizationContextHolder.class); - - private static final String CONTEXT_HOLDER_ATTRIBUTE = "CONTEXT_HOLDER_ATTRIBUTE"; - - private final String guiClientId; - private final String guiClientSecret; - private final WebserviceURIService webserviceURIService; - private final ClientHttpRequestFactoryService clientHttpRequestFactoryService; - - @Autowired - public OAuth2AuthorizationContextHolder( - @Value("${sebserver.webservice.api.admin.clientId}") final String guiClientId, - @Value("${sebserver.webservice.api.admin.clientSecret}") final String guiClientSecret, - final WebserviceURIService webserviceURIService, - final ClientHttpRequestFactoryService clientHttpRequestFactoryService) { - - this.guiClientId = guiClientId; - this.guiClientSecret = guiClientSecret; - this.webserviceURIService = webserviceURIService; - this.clientHttpRequestFactoryService = clientHttpRequestFactoryService; - } - - @Override - public WebserviceURIService getWebserviceURIService() { - return this.webserviceURIService; - } - - @Override - public SEBServerAuthorizationContext getAuthorizationContext(final HttpSession session) { - log.debug("Trying to get OAuth2AuthorizationContext from HttpSession: {}", session.getId()); - - OAuth2AuthorizationContext context = - (OAuth2AuthorizationContext) session.getAttribute(CONTEXT_HOLDER_ATTRIBUTE); - - if (context == null || !context.valid) { - log.debug( - "OAuth2AuthorizationContext for HttpSession: {} is not present or is invalid. " - + "Create new OAuth2AuthorizationContext for this session", - session.getId()); - - final ClientHttpRequestFactory clientHttpRequestFactory = this.clientHttpRequestFactoryService - .getClientHttpRequestFactory() - .getOrThrow(); - - context = new OAuth2AuthorizationContext( - this.guiClientId, - this.guiClientSecret, - this.webserviceURIService, - clientHttpRequestFactory); - - session.setAttribute(CONTEXT_HOLDER_ATTRIBUTE, context); - } - - return context; - } - - private static final class DisposableOAuth2RestTemplate extends OAuth2RestTemplate { - - private boolean enabled = true; - - public DisposableOAuth2RestTemplate(final OAuth2ProtectedResourceDetails resource) { - super( - resource, - new DefaultOAuth2ClientContext(new DefaultAccessTokenRequest())); - } - - @Override - protected T doExecute( - final URI url, - final HttpMethod method, - final RequestCallback requestCallback, - final ResponseExtractor responseExtractor) throws RestClientException { - - if (this.enabled) { - return super.doExecute(url, method, requestCallback, responseExtractor); - } else { - throw new IllegalStateException( - "Error: Forbidden execution call on disabled DisposableOAuth2RestTemplate"); - } - } - } - - private static final class OAuth2AuthorizationContext implements SEBServerAuthorizationContext { - - private static final String GRANT_TYPE = "password"; - private static final List SCOPES = Collections.unmodifiableList( - Arrays.asList("read", "write")); - - private boolean valid = true; - - private final ResourceOwnerPasswordResourceDetails resource; - private final DisposableOAuth2RestTemplate restTemplate; - private final String revokeTokenURI; - private final String currentUserURI; - - private Result loggedInUser = null; - - OAuth2AuthorizationContext( - final String guiClientId, - final String guiClientSecret, - final WebserviceURIService webserviceURIService, - final ClientHttpRequestFactory clientHttpRequestFactory) { - - this.resource = new ResourceOwnerPasswordResourceDetails(); - this.resource.setAccessTokenUri(webserviceURIService.getOAuthTokenURI()); - this.resource.setClientId(guiClientId); - this.resource.setClientSecret(guiClientSecret); - this.resource.setGrantType(GRANT_TYPE); - this.resource.setScope(SCOPES); - - this.restTemplate = new DisposableOAuth2RestTemplate(this.resource); - this.restTemplate.setRequestFactory(clientHttpRequestFactory); - this.restTemplate.setErrorHandler(new ErrorHandler(this.resource)); - this.restTemplate - .getMessageConverters() - .add(0, new StringHttpMessageConverter(Charset.forName("UTF-8"))); - - this.revokeTokenURI = webserviceURIService.getOAuthRevokeTokenURI(); - this.currentUserURI = webserviceURIService.getCurrentUserRequestURI(); - } - - @Override - public boolean isValid() { - return this.valid; - } - - @Override - public boolean isLoggedIn() { - final OAuth2AccessToken accessToken = this.restTemplate.getOAuth2ClientContext().getAccessToken(); - if (accessToken == null || StringUtils.isEmpty(accessToken.toString())) { - return false; - } - - try { - final ResponseEntity forEntity = - this.restTemplate.getForEntity(this.currentUserURI, String.class); - if (forEntity.getStatusCode() != HttpStatus.OK) { - return false; - } - } catch (final Exception e) { - log.error("Failed to verify logged in user: {}", e.getMessage()); - return false; - } - - return true; - } - - @Override - public boolean login(final String username, final CharSequence password) { - if (!this.valid || this.isLoggedIn()) { - return false; - } - - this.resource.setUsername(username); - this.resource.setPassword(Utils.toString(password)); - - log.debug("Trying to login for user: {}", username); - - try { - this.restTemplate.getAccessToken(); - log.debug("Got token for user: {}", username); - this.loggedInUser = getLoggedInUser(); - return true; - } catch (final OAuth2AccessDeniedException | AccessDeniedException e) { - log.info("Access Denied for user: {}", username); - return false; - } - } - - @Override - public boolean logout() { - // set this context invalid to force creation of a new context on next request - this.valid = false; - this.loggedInUser = null; - if (this.restTemplate.getAccessToken() != null) { - // delete the access-token (and refresh-token) on authentication server side - this.restTemplate.delete(this.revokeTokenURI); - // delete the access-token within the RestTemplate - this.restTemplate.getOAuth2ClientContext().setAccessToken(null); - } - // mark the RestTemplate as disposed - this.restTemplate.enabled = false; - return true; - } - - @Override - public RestTemplate getRestTemplate() { - return this.restTemplate; - } - - @Override - public void refreshUser(final UserInfo userInfo) { - // delete the access-token (and refresh-token) on authentication server side - this.restTemplate.delete(this.revokeTokenURI); - // delete the access-token within the RestTemplate - this.restTemplate.getOAuth2ClientContext().setAccessToken(null); - // check if username has changed - if (!userInfo.username.equals(getLoggedInUser().get().username)) { - // Set new username to be able to request new access token - this.resource.setUsername(userInfo.username); - } - - // and request new access token - this.restTemplate.getAccessToken(); - // and reset logged in user by getting actual one from webservice - this.loggedInUser = null; - getLoggedInUser() - .getOrThrow(); - } - - @Override - public Result getLoggedInUser() { - if (this.loggedInUser != null) { - return this.loggedInUser; - } - - log.debug("Request logged in User from SEBserver web-service API"); - - try { - if (isValid() && isLoggedIn()) { - final ResponseEntity response = - this.restTemplate - .getForEntity(this.currentUserURI, UserInfo.class); - if (response.getStatusCode() == HttpStatus.OK) { - this.loggedInUser = Result.of(response.getBody()); - return this.loggedInUser; - } else { - log.error("Unexpected error response: {}", response); - return Result.ofError(new IllegalStateException( - "Http Request responded with status: " + response.getStatusCode())); - } - } else { - return Result.ofError( - new IllegalStateException("Logged in User requested on invalid or not logged in ")); - } - } catch (final AccessDeniedException | OAuth2AccessDeniedException ade) { - log.error("Acccess denied while trying to request logged in User from API", ade); - return Result.ofError(ade); - } catch (final Exception e) { - log.error("Unexpected error while trying to request logged in User from API", e); - return Result.ofError( - new RuntimeException("Unexpected error while trying to request logged in User from API", e)); - } - } - - @Override - public boolean hasRole(final UserRole role) { - if (!isValid() || !isLoggedIn()) { - return false; - } - - return getLoggedInUser() - .getOrThrow().roles - .contains(role.name()); - } - - private static final class ErrorHandler extends OAuth2ErrorHandler { - private ErrorHandler(final OAuth2ProtectedResourceDetails resource) { - super(resource); - } - - @Override - public boolean hasError(final ClientHttpResponse response) throws IOException { - try { - final HttpStatus statusCode = HttpStatus.resolve(response.getRawStatusCode()); - return (statusCode != null && statusCode.series().equals(HttpStatus.Series.SERVER_ERROR)); - } catch (final Exception e) { - log.error("Unexpected: ", e); - return super.hasError(response); - } - } - } - - } -} +/* + * 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.service.remote.webservice.auth; + +import ch.ethz.seb.sebserver.ClientHttpRequestFactoryService; +import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; +import ch.ethz.seb.sebserver.gbl.model.user.UserRole; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gbl.util.Result; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Lazy; +import org.springframework.http.HttpMethod; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.http.client.ClientHttpRequestFactory; +import org.springframework.http.client.ClientHttpResponse; +import org.springframework.http.converter.StringHttpMessageConverter; +import org.springframework.security.access.AccessDeniedException; +import org.springframework.security.oauth2.client.DefaultOAuth2ClientContext; +import org.springframework.security.oauth2.client.OAuth2RestTemplate; +import org.springframework.security.oauth2.client.http.OAuth2ErrorHandler; +import org.springframework.security.oauth2.client.resource.OAuth2AccessDeniedException; +import org.springframework.security.oauth2.client.resource.OAuth2ProtectedResourceDetails; +import org.springframework.security.oauth2.client.token.DefaultAccessTokenRequest; +import org.springframework.security.oauth2.client.token.grant.password.ResourceOwnerPasswordResourceDetails; +import org.springframework.security.oauth2.common.OAuth2AccessToken; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RequestCallback; +import org.springframework.web.client.ResponseExtractor; +import org.springframework.web.client.RestClientException; +import org.springframework.web.client.RestTemplate; + +import javax.servlet.http.HttpSession; +import java.io.IOException; +import java.net.URI; +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + +@Lazy +@Component +@GuiProfile +public class OAuth2AuthorizationContextHolder implements AuthorizationContextHolder { + + private static final Logger log = LoggerFactory.getLogger(OAuth2AuthorizationContextHolder.class); + + private static final String CONTEXT_HOLDER_ATTRIBUTE = "CONTEXT_HOLDER_ATTRIBUTE"; + + private final String guiClientId; + private final String guiClientSecret; + private final WebserviceURIService webserviceURIService; + private final ClientHttpRequestFactoryService clientHttpRequestFactoryService; + + @Autowired + public OAuth2AuthorizationContextHolder( + @Value("${sebserver.webservice.api.admin.clientId}") final String guiClientId, + @Value("${sebserver.webservice.api.admin.clientSecret}") final String guiClientSecret, + final WebserviceURIService webserviceURIService, + final ClientHttpRequestFactoryService clientHttpRequestFactoryService) { + + this.guiClientId = guiClientId; + this.guiClientSecret = guiClientSecret; + this.webserviceURIService = webserviceURIService; + this.clientHttpRequestFactoryService = clientHttpRequestFactoryService; + } + + @Override + public WebserviceURIService getWebserviceURIService() { + return this.webserviceURIService; + } + + @Override + public SEBServerAuthorizationContext getAuthorizationContext(final HttpSession session) { + log.debug("Trying to get OAuth2AuthorizationContext from HttpSession: {}", session.getId()); + + OAuth2AuthorizationContext context = + (OAuth2AuthorizationContext) session.getAttribute(CONTEXT_HOLDER_ATTRIBUTE); + + if (context == null || !context.valid) { + log.debug( + "OAuth2AuthorizationContext for HttpSession: {} is not present or is invalid. " + + "Create new OAuth2AuthorizationContext for this session", + session.getId()); + + final ClientHttpRequestFactory clientHttpRequestFactory = this.clientHttpRequestFactoryService + .getClientHttpRequestFactory() + .getOrThrow(); + + context = new OAuth2AuthorizationContext( + this.guiClientId, + this.guiClientSecret, + this.webserviceURIService, + clientHttpRequestFactory); + + session.setAttribute(CONTEXT_HOLDER_ATTRIBUTE, context); + } + + return context; + } + + private static final class DisposableOAuth2RestTemplate extends OAuth2RestTemplate { + + private boolean enabled = true; + + public DisposableOAuth2RestTemplate(final OAuth2ProtectedResourceDetails resource) { + super( + resource, + new DefaultOAuth2ClientContext(new DefaultAccessTokenRequest())); + } + + @Override + protected T doExecute( + final URI url, + final HttpMethod method, + final RequestCallback requestCallback, + final ResponseExtractor responseExtractor) throws RestClientException { + + if (this.enabled) { + return super.doExecute(url, method, requestCallback, responseExtractor); + } else { + throw new IllegalStateException( + "Error: Forbidden execution call on disabled DisposableOAuth2RestTemplate"); + } + } + } + + private static final class OAuth2AuthorizationContext implements SEBServerAuthorizationContext { + + private static final String GRANT_TYPE = "password"; + private static final List SCOPES = Collections.unmodifiableList( + Arrays.asList("read", "write")); + + private boolean valid = true; + + private final ResourceOwnerPasswordResourceDetails resource; + private final DisposableOAuth2RestTemplate restTemplate; + private final String revokeTokenURI; + private final String currentUserURI; + + private Result loggedInUser = null; + + OAuth2AuthorizationContext( + final String guiClientId, + final String guiClientSecret, + final WebserviceURIService webserviceURIService, + final ClientHttpRequestFactory clientHttpRequestFactory) { + + this.resource = new ResourceOwnerPasswordResourceDetails(); + this.resource.setAccessTokenUri(webserviceURIService.getOAuthTokenURI()); + this.resource.setClientId(guiClientId); + this.resource.setClientSecret(guiClientSecret); + this.resource.setGrantType(GRANT_TYPE); + this.resource.setScope(SCOPES); + + this.restTemplate = new DisposableOAuth2RestTemplate(this.resource); + this.restTemplate.setRequestFactory(clientHttpRequestFactory); + this.restTemplate.setErrorHandler(new ErrorHandler(this.resource)); + this.restTemplate + .getMessageConverters() + .add(0, new StringHttpMessageConverter(StandardCharsets.UTF_8)); + + this.revokeTokenURI = webserviceURIService.getOAuthRevokeTokenURI(); + this.currentUserURI = webserviceURIService.getCurrentUserRequestURI(); + } + + @Override + public boolean isValid() { + return this.valid; + } + + @Override + public boolean isLoggedIn() { + final OAuth2AccessToken accessToken = this.restTemplate.getOAuth2ClientContext().getAccessToken(); + if (accessToken == null || StringUtils.isEmpty(accessToken.toString())) { + return false; + } + + try { + final ResponseEntity forEntity = + this.restTemplate.getForEntity(this.currentUserURI, String.class); + if (forEntity.getStatusCode() != HttpStatus.OK) { + return false; + } + } catch (final Exception e) { + log.error("Failed to verify logged in user: {}", e.getMessage()); + return false; + } + + return true; + } + + @Override + public boolean login(final String username, final CharSequence password) { + if (!this.valid || this.isLoggedIn()) { + return false; + } + + this.resource.setUsername(username); + this.resource.setPassword(Utils.toString(password)); + + log.debug("Trying to login for user: {}", username); + + try { + this.restTemplate.getAccessToken(); + log.debug("Got token for user: {}", username); + this.loggedInUser = getLoggedInUser(); + return true; + } catch (final OAuth2AccessDeniedException | AccessDeniedException e) { + log.info("Access Denied for user: {}", username); + return false; + } + } + + @Override + public boolean logout() { + // set this context invalid to force creation of a new context on next request + this.valid = false; + this.loggedInUser = null; + if (this.restTemplate.getAccessToken() != null) { + // delete the access-token (and refresh-token) on authentication server side + this.restTemplate.delete(this.revokeTokenURI); + // delete the access-token within the RestTemplate + this.restTemplate.getOAuth2ClientContext().setAccessToken(null); + } + // mark the RestTemplate as disposed + this.restTemplate.enabled = false; + return true; + } + + @Override + public RestTemplate getRestTemplate() { + return this.restTemplate; + } + + @Override + public void refreshUser(final UserInfo userInfo) { + // delete the access-token (and refresh-token) on authentication server side + this.restTemplate.delete(this.revokeTokenURI); + // delete the access-token within the RestTemplate + this.restTemplate.getOAuth2ClientContext().setAccessToken(null); + // check if username has changed + if (!userInfo.username.equals(getLoggedInUser().get().username)) { + // Set new username to be able to request new access token + this.resource.setUsername(userInfo.username); + } + + // and request new access token + this.restTemplate.getAccessToken(); + // and reset logged in user by getting actual one from webservice + this.loggedInUser = null; + getLoggedInUser() + .getOrThrow(); + } + + @Override + public Result getLoggedInUser() { + if (this.loggedInUser != null) { + return this.loggedInUser; + } + + log.debug("Request logged in User from SEBserver web-service API"); + + try { + if (isValid() && isLoggedIn()) { + final ResponseEntity response = + this.restTemplate + .getForEntity(this.currentUserURI, UserInfo.class); + if (response.getStatusCode() == HttpStatus.OK) { + this.loggedInUser = Result.of(response.getBody()); + return this.loggedInUser; + } else { + log.error("Unexpected error response: {}", response); + return Result.ofError(new IllegalStateException( + "Http Request responded with status: " + response.getStatusCode())); + } + } else { + return Result.ofError( + new IllegalStateException("Logged in User requested on invalid or not logged in ")); + } + } catch (final AccessDeniedException | OAuth2AccessDeniedException ade) { + log.error("Acccess denied while trying to request logged in User from API", ade); + return Result.ofError(ade); + } catch (final Exception e) { + log.error("Unexpected error while trying to request logged in User from API", e); + return Result.ofError( + new RuntimeException("Unexpected error while trying to request logged in User from API", e)); + } + } + + @Override + public boolean hasRole(final UserRole role) { + if (!isValid() || !isLoggedIn()) { + return false; + } + + return getLoggedInUser() + .getOrThrow().roles + .contains(role.name()); + } + + private static final class ErrorHandler extends OAuth2ErrorHandler { + private ErrorHandler(final OAuth2ProtectedResourceDetails resource) { + super(resource); + } + + @Override + public boolean hasError(final ClientHttpResponse response) throws IOException { + try { + final HttpStatus statusCode = HttpStatus.resolve(response.getRawStatusCode()); + return (statusCode != null && statusCode.series().equals(HttpStatus.Series.SERVER_ERROR)); + } catch (final Exception e) { + log.error("Unexpected: ", e); + return super.hasError(response); + } + } + } + + } +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/SEBServerAuthorizationContext.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/SEBServerAuthorizationContext.java index a4485014..7db177c2 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/SEBServerAuthorizationContext.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/SEBServerAuthorizationContext.java @@ -1,66 +1,66 @@ -/* - * 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.service.remote.webservice.auth; - -import org.springframework.web.client.RestTemplate; - -import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; -import ch.ethz.seb.sebserver.gbl.model.user.UserRole; -import ch.ethz.seb.sebserver.gbl.util.Result; - -/** Defines functionality for the SEB Server webservice authorization context used to - * manage a user session on GUI service. */ -public interface SEBServerAuthorizationContext { - - /** Indicates if this authorization context is still valid - * - * @return true if the SEBServerAuthorizationContext is valid. False of not. */ - boolean isValid(); - - /** Indicated whether a user is logged in within this authorization context or not. - * - * @return whether a user is logged in within this authorization context or not */ - boolean isLoggedIn(); - - /** Requests a login with username and password on SEB Server webservice. - * This uses OAuth 2 and Springs OAuth2RestTemplate to exchange user/client credentials - * with an access and refresh token. - * - * @param username the username for login - * @param password the password for login - * @return */ - boolean login(String username, CharSequence password); - - /** Requests a logout on SEB Server webservice if a user is currently logged in - * This uses OAuth 2 and Springs OAuth2RestTemplate to make a revoke token request for the - * currently logged in user and also invalidates this SEBServerAuthorizationContext - * - * @return true if logout was successful */ - boolean logout(); - - /** Gets a Result of the UserInfo data of currently logged in user or of an error if no user is logged in - * or there was an unexpected error while trying to get the user information. - * - * @return Result of logged in user data or of an error on fail */ - Result getLoggedInUser(); - - void refreshUser(UserInfo userInfo); - - /** Returns true if a current logged in user has the specified role. - * - * @param role the UserRole to check - * @return true if a current logged in user has the specified role */ - boolean hasRole(UserRole role); - - /** Get the underling RestTemplate to connect and communicate with the SEB Server webservice. - * - * @return the underling RestTemplate to connect and communicate with the SEB Server webservice */ - RestTemplate getRestTemplate(); - -} +/* + * 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.service.remote.webservice.auth; + +import org.springframework.web.client.RestTemplate; + +import ch.ethz.seb.sebserver.gbl.model.user.UserInfo; +import ch.ethz.seb.sebserver.gbl.model.user.UserRole; +import ch.ethz.seb.sebserver.gbl.util.Result; + +/** Defines functionality for the SEB Server webservice authorization context used to + * manage a user session on GUI service. */ +public interface SEBServerAuthorizationContext { + + /** Indicates if this authorization context is still valid + * + * @return true if the SEBServerAuthorizationContext is valid. False of not. */ + boolean isValid(); + + /** Indicated whether a user is logged in within this authorization context or not. + * + * @return whether a user is logged in within this authorization context or not */ + boolean isLoggedIn(); + + /** Requests a login with username and password on SEB Server webservice. + * This uses OAuth 2 and Springs OAuth2RestTemplate to exchange user/client credentials + * with an access and refresh token. + * + * @param username the username for login + * @param password the password for login + * @return true if login was successful, false if no */ + boolean login(String username, CharSequence password); + + /** Requests a logout on SEB Server webservice if a user is currently logged in + * This uses OAuth 2 and Springs OAuth2RestTemplate to make a revoke token request for the + * currently logged in user and also invalidates this SEBServerAuthorizationContext + * + * @return true if logout was successful */ + boolean logout(); + + /** Gets a Result of the UserInfo data of currently logged in user or of an error if no user is logged in + * or there was an unexpected error while trying to get the user information. + * + * @return Result of logged in user data or of an error on fail */ + Result getLoggedInUser(); + + void refreshUser(UserInfo userInfo); + + /** Returns true if a current logged in user has the specified role. + * + * @param role the UserRole to check + * @return true if a current logged in user has the specified role */ + boolean hasRole(UserRole role); + + /** Get the underling RestTemplate to connect and communicate with the SEB Server webservice. + * + * @return the underling RestTemplate to connect and communicate with the SEB Server webservice */ + RestTemplate getRestTemplate(); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceConnectionData.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceConnectionData.java index 22a2f631..4245719f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceConnectionData.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceConnectionData.java @@ -1,114 +1,114 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.auth; - -import org.springframework.web.util.UriComponentsBuilder; - -public class WebserviceConnectionData { - - final String id; - final String webserviceProtocol; - final String webserviceServerAdress; - final String webserviceServerPort; - final String webserviceAPIPath; - final String webserviceServerAddress; - - private final UriComponentsBuilder webserviceURIBuilder; - - protected WebserviceConnectionData( - final String id, - final String webserviceProtocol, - final String webserviceServerAdress, - final String webserviceServerPort, - final String webserviceAPIPath) { - - this.id = id; - this.webserviceProtocol = webserviceProtocol; - this.webserviceServerAdress = webserviceServerAdress; - this.webserviceServerPort = webserviceServerPort; - this.webserviceAPIPath = webserviceAPIPath; - - this.webserviceServerAddress = webserviceProtocol + "://" + webserviceServerAdress + ":" + webserviceServerPort; - this.webserviceURIBuilder = UriComponentsBuilder - .fromHttpUrl(webserviceProtocol + "://" + webserviceServerAdress) - .port(webserviceServerPort) - .path(webserviceAPIPath); - } - - public String getId() { - return this.id; - } - - public String getWebserviceProtocol() { - return this.webserviceProtocol; - } - - public String getWebserviceServerAdress() { - return this.webserviceServerAdress; - } - - public String getWebserviceServerPort() { - return this.webserviceServerPort; - } - - public String getWebserviceAPIPath() { - return this.webserviceAPIPath; - } - - public String getWebserviceServerAddress() { - return this.webserviceServerAddress; - } - - public UriComponentsBuilder getWebserviceURIBuilder() { - return this.webserviceURIBuilder.cloneBuilder(); - } - - @Override - public int hashCode() { - final int prime = 31; - int result = 1; - result = prime * result + ((this.id == null) ? 0 : this.id.hashCode()); - return result; - } - - @Override - public boolean equals(final Object obj) { - if (this == obj) - return true; - if (obj == null) - return false; - if (getClass() != obj.getClass()) - return false; - final WebserviceConnectionData other = (WebserviceConnectionData) obj; - if (this.id == null) { - if (other.id != null) - return false; - } else if (!this.id.equals(other.id)) - return false; - return true; - } - - @Override - public String toString() { - final StringBuilder builder = new StringBuilder(); - builder.append("WebserviceConnectionData [id="); - builder.append(this.id); - builder.append(", webserviceProtocol="); - builder.append(this.webserviceProtocol); - builder.append(", webserviceServerAdress="); - builder.append(this.webserviceServerAdress); - builder.append(", webserviceServerPort="); - builder.append(this.webserviceServerPort); - builder.append(", webserviceAPIPath="); - builder.append(this.webserviceAPIPath); - builder.append("]"); - return builder.toString(); - } - -} +/* + * Copyright (c) 2019 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.service.remote.webservice.auth; + +import org.springframework.web.util.UriComponentsBuilder; + +public class WebserviceConnectionData { + + final String id; + final String webserviceProtocol; + final String webserviceServerAddress; + final String webserviceServerPort; + final String webserviceAPIPath; + final String webserviceServerURL; + + private final UriComponentsBuilder webserviceURIBuilder; + + protected WebserviceConnectionData( + final String id, + final String webserviceProtocol, + final String webserviceServerAddress, + final String webserviceServerPort, + final String webserviceAPIPath) { + + this.id = id; + this.webserviceProtocol = webserviceProtocol; + this.webserviceServerAddress = webserviceServerAddress; + this.webserviceServerPort = webserviceServerPort; + this.webserviceAPIPath = webserviceAPIPath; + + this.webserviceServerURL = webserviceProtocol + "://" + webserviceServerAddress + ":" + webserviceServerPort; + this.webserviceURIBuilder = UriComponentsBuilder + .fromHttpUrl(webserviceProtocol + "://" + webserviceServerAddress) + .port(webserviceServerPort) + .path(webserviceAPIPath); + } + + public String getId() { + return this.id; + } + + public String getWebserviceProtocol() { + return this.webserviceProtocol; + } + + public String getWebserviceServerAddress() { + return this.webserviceServerAddress; + } + + public String getWebserviceServerPort() { + return this.webserviceServerPort; + } + + public String getWebserviceAPIPath() { + return this.webserviceAPIPath; + } + + public String getWebserviceServerURL() { + return this.webserviceServerURL; + } + + public UriComponentsBuilder getWebserviceURIBuilder() { + return this.webserviceURIBuilder.cloneBuilder(); + } + + @Override + public int hashCode() { + final int prime = 31; + int result = 1; + result = prime * result + ((this.id == null) ? 0 : this.id.hashCode()); + return result; + } + + @Override + public boolean equals(final Object obj) { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + final WebserviceConnectionData other = (WebserviceConnectionData) obj; + if (this.id == null) { + if (other.id != null) + return false; + } else if (!this.id.equals(other.id)) + return false; + return true; + } + + @Override + public String toString() { + final StringBuilder builder = new StringBuilder(); + builder.append("WebserviceConnectionData [id="); + builder.append(this.id); + builder.append(", webserviceProtocol="); + builder.append(this.webserviceProtocol); + builder.append(", webserviceServerAddress="); + builder.append(this.webserviceServerAddress); + builder.append(", webserviceServerPort="); + builder.append(this.webserviceServerPort); + builder.append(", webserviceAPIPath="); + builder.append(this.webserviceAPIPath); + builder.append("]"); + return builder.toString(); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceURIService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceURIService.java index a3565b49..a5791281 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceURIService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/auth/WebserviceURIService.java @@ -1,69 +1,69 @@ -/* - * Copyright (c) 2019 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.service.remote.webservice.auth; - -import org.springframework.beans.factory.annotation.Value; -import org.springframework.stereotype.Component; -import org.springframework.web.util.UriComponentsBuilder; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; - -@Component -@GuiProfile -public class WebserviceURIService { - - private final String servletContextPath; - private final String webserviceServerAddress; - private final UriComponentsBuilder webserviceURIBuilder; - - public WebserviceURIService( - @Value("${sebserver.gui.webservice.protocol}") final String webserviceProtocol, - @Value("${sebserver.gui.webservice.address}") final String webserviceServerAdress, - @Value("${sebserver.gui.webservice.port}") final String webserviceServerPort, - @Value("${server.servlet.context-path}") final String servletContextPath, - @Value("${sebserver.gui.webservice.apipath}") final String webserviceAPIPath) { - - this.servletContextPath = servletContextPath; - this.webserviceServerAddress = webserviceProtocol + "://" + webserviceServerAdress + ":" + webserviceServerPort; - this.webserviceURIBuilder = UriComponentsBuilder - .fromHttpUrl(webserviceProtocol + "://" + webserviceServerAdress) - .port(webserviceServerPort) - .path(servletContextPath) - .path(webserviceAPIPath); - } - - public String getWebserviceServerAddress() { - return this.webserviceServerAddress; - } - - public UriComponentsBuilder getURIBuilder() { - return this.webserviceURIBuilder.cloneBuilder(); - } - - public String getOAuthTokenURI() { - return UriComponentsBuilder.fromHttpUrl(this.webserviceServerAddress) - .path(this.servletContextPath) - .path(API.OAUTH_TOKEN_ENDPOINT) - .toUriString(); - } - - public String getOAuthRevokeTokenURI() { - return UriComponentsBuilder.fromHttpUrl(this.webserviceServerAddress) - .path(this.servletContextPath) - .path(API.OAUTH_REVOKE_TOKEN_ENDPOINT) - .toUriString(); - } - - public String getCurrentUserRequestURI() { - return getURIBuilder() - .path(API.CURRENT_USER_ENDPOINT) - .toUriString(); - } -} +/* + * Copyright (c) 2019 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.service.remote.webservice.auth; + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.stereotype.Component; +import org.springframework.web.util.UriComponentsBuilder; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; + +@Component +@GuiProfile +public class WebserviceURIService { + + private final String servletContextPath; + private final String webserviceServerAddress; + private final UriComponentsBuilder webserviceURIBuilder; + + public WebserviceURIService( + @Value("${sebserver.gui.webservice.protocol}") final String webserviceProtocol, + @Value("${sebserver.gui.webservice.address}") final String webserviceServerAddress, + @Value("${sebserver.gui.webservice.port}") final String webserviceServerPort, + @Value("${server.servlet.context-path}") final String servletContextPath, + @Value("${sebserver.gui.webservice.apipath}") final String webserviceAPIPath) { + + this.servletContextPath = servletContextPath; + this.webserviceServerAddress = webserviceProtocol + "://" + webserviceServerAddress + ":" + webserviceServerPort; + this.webserviceURIBuilder = UriComponentsBuilder + .fromHttpUrl(webserviceProtocol + "://" + webserviceServerAddress) + .port(webserviceServerPort) + .path(servletContextPath) + .path(webserviceAPIPath); + } + + public String getWebserviceServerAddress() { + return this.webserviceServerAddress; + } + + public UriComponentsBuilder getURIBuilder() { + return this.webserviceURIBuilder.cloneBuilder(); + } + + public String getOAuthTokenURI() { + return UriComponentsBuilder.fromHttpUrl(this.webserviceServerAddress) + .path(this.servletContextPath) + .path(API.OAUTH_TOKEN_ENDPOINT) + .toUriString(); + } + + public String getOAuthRevokeTokenURI() { + return UriComponentsBuilder.fromHttpUrl(this.webserviceServerAddress) + .path(this.servletContextPath) + .path(API.OAUTH_REVOKE_TOKEN_ENDPOINT) + .toUriString(); + } + + public String getCurrentUserRequestURI() { + return getURIBuilder() + .path(API.CURRENT_USER_ENDPOINT) + .toUriString(); + } +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionDetails.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionDetails.java index a76e06a1..66549357 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionDetails.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionDetails.java @@ -49,12 +49,10 @@ public class ClientConnectionDetails { private static final int NUMBER_OF_NONE_INDICATOR_ROWS = 3; - private final PageService pageService; private final ResourceService resourceService; - private final Exam exam; private final EnumMap indicatorMapping; private final RestCall.RestCallBuilder restCallBuilder; - private final FormHandle formhandle; + private final FormHandle formHandle; private final ColorData colorData; private ClientConnectionData connectionData = null; @@ -69,9 +67,7 @@ public class ClientConnectionDetails { final Display display = pageContext.getRoot().getDisplay(); - this.pageService = pageService; this.resourceService = pageService.getResourceService(); - this.exam = exam; this.restCallBuilder = restCallBuilder; this.colorData = new ColorData(display); this.indicatorMapping = IndicatorData.createFormIndicators( @@ -80,12 +76,12 @@ public class ClientConnectionDetails { this.colorData, NUMBER_OF_NONE_INDICATOR_ROWS); - final FormBuilder formBuilder = this.pageService.formBuilder(pageContext) + final FormBuilder formBuilder = pageService.formBuilder(pageContext) .readonly(true) .addField(FormBuilder.text( QuizData.QUIZ_ATTR_NAME, EXAM_NAME_TEXT_KEY, - this.exam.getName())) + exam.getName())) .addField(FormBuilder.text( Domain.CLIENT_CONNECTION.ATTR_EXAM_USER_SESSION_ID, CONNECTION_ID_TEXT_KEY, @@ -112,7 +108,7 @@ public class ClientConnectionDetails { .withDefaultLabel(indData.indicator.name)) .addEmptyCell()); - this.formhandle = formBuilder.build(); + this.formHandle = formBuilder.build(); } public void updateData() { @@ -136,7 +132,7 @@ public class ClientConnectionDetails { return; } - final Form form = this.formhandle.getForm(); + final Form form = this.formHandle.getForm(); form.setFieldValue( Domain.CLIENT_CONNECTION.ATTR_EXAM_USER_SESSION_ID, this.connectionData.clientConnection.userSessionId); diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionTable.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionTable.java index 1806b13d..7ebebea4 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionTable.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/ClientConnectionTable.java @@ -371,7 +371,7 @@ public final class ClientConnectionTable { private void sortTable() { this.tableMapping = this.tableMapping.entrySet() .stream() - .sorted((e1, e2) -> e1.getValue().compareTo(e2.getValue())) + .sorted(Entry.comparingByValue()) .collect(Collectors.toMap( Entry::getKey, Entry::getValue, @@ -398,13 +398,12 @@ public final class ClientConnectionTable { final String attribute = this.resourceService .getCurrentUser() .getAttribute(USER_SESSION_STATUS_FILTER_ATTRIBUTE); + this.statusFilter.clear(); if (attribute != null) { - this.statusFilter.clear(); Arrays.asList(StringUtils.split(attribute, Constants.LIST_SEPARATOR)) .forEach(name -> this.statusFilter.add(ConnectionStatus.valueOf(name))); } else { - this.statusFilter.clear(); this.statusFilter.add(ConnectionStatus.DISABLED); } } catch (final Exception e) { @@ -460,7 +459,7 @@ public final class ClientConnectionTable { } void updateData(final TableItem tableItem) { - tableItem.setText(0, getConnectionIdentifer()); + tableItem.setText(0, getConnectionIdentifier()); tableItem.setText(1, getConnectionAddress()); tableItem.setText(2, getStatusName()); } @@ -533,7 +532,7 @@ public final class ClientConnectionTable { public int compareTo(final UpdatableTableItem other) { return Comparator.comparingInt(UpdatableTableItem::statusWeight) .thenComparingInt(UpdatableTableItem::thresholdsWeight) - .thenComparing(UpdatableTableItem::getConnectionIdentifer) + .thenComparing(UpdatableTableItem::getConnectionIdentifier) .compare(this, other); } @@ -580,7 +579,7 @@ public final class ClientConnectionTable { return Constants.EMPTY_NOTE; } - String getConnectionIdentifer() { + String getConnectionIdentifier() { if (this.connectionData != null && this.connectionData.clientConnection.userSessionId != null) { return this.connectionData.clientConnection.userSessionId; } @@ -608,10 +607,7 @@ public final class ClientConnectionTable { final IndicatorData indicatorData = ClientConnectionTable.this.indicatorMapping.get(indicatorValue.getType()); - if (indicatorData == null) { - log.error("No IndicatorData of type: {} found", indicatorValue.getType()); - } else { - + if (indicatorData != null) { final double value = indicatorValue.getValue(); final int indicatorWeight = IndicatorData.getWeight(indicatorData, value); if (this.indicatorWeights[indicatorData.index] != indicatorWeight) { diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/IndicatorData.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/IndicatorData.java index f64483a6..bfd2225e 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/IndicatorData.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/IndicatorData.java @@ -1,100 +1,99 @@ -/* - * Copyright (c) 2019 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.service.session; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Collections; -import java.util.EnumMap; - -import org.eclipse.swt.graphics.Color; -import org.eclipse.swt.widgets.Display; - -import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; -import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.IndicatorType; -import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.Threshold; -import ch.ethz.seb.sebserver.gbl.util.Utils; - -final class IndicatorData { - - final int index; - final int tableIndex; - final Indicator indicator; - final Color defaultColor; - final Color defaultTextColor; - final ThresholdColor[] thresholdColor; - - protected IndicatorData( - final Indicator indicator, - final int index, - final int tableIndex, - final ColorData colorData, - final Display display) { - - this.indicator = indicator; - this.index = index; - this.tableIndex = tableIndex; - this.defaultColor = new Color(display, Utils.toRGB(indicator.defaultColor), 255); - this.defaultTextColor = Utils.darkColor(this.defaultColor.getRGB()) - ? colorData.darkColor - : colorData.lightColor; - - this.thresholdColor = new ThresholdColor[indicator.thresholds.size()]; - final ArrayList sortedThresholds = new ArrayList<>(indicator.thresholds); - Collections.sort(sortedThresholds, (t1, t2) -> t1.value.compareTo(t2.value)); - for (int i = 0; i < indicator.thresholds.size(); i++) { - this.thresholdColor[i] = new ThresholdColor(sortedThresholds.get(i), display, colorData); - } - } - - static final EnumMap createFormIndicators( - final Collection indicators, - final Display display, - final ColorData colorData, - final int tableIndexOffset) { - - final EnumMap indicatorMapping = new EnumMap<>(IndicatorType.class); - int i = 0; - for (final Indicator indicator : indicators) { - indicatorMapping.put(indicator.type, new IndicatorData( - indicator, - i, - i + tableIndexOffset, - colorData, - display)); - i++; - } - return indicatorMapping; - } - - static final int getWeight(final IndicatorData indicatorData, final double value) { - for (int j = 0; j < indicatorData.thresholdColor.length; j++) { - if (value < indicatorData.thresholdColor[j].value) { - return (j == 0) ? -1 : j - 1; - } - } - - return indicatorData.thresholdColor.length - 1; - } - - static final class ThresholdColor { - final double value; - final Color color; - final Color textColor; - - protected ThresholdColor(final Threshold threshold, final Display display, final ColorData colorData) { - this.value = threshold.value; - this.color = new Color(display, Utils.toRGB(threshold.color), 255); - this.textColor = Utils.darkColor(this.color.getRGB()) - ? colorData.darkColor - : colorData.lightColor; - } - } - +/* + * Copyright (c) 2019 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.service.session; + +import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; +import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.IndicatorType; +import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.Threshold; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.widgets.Display; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Comparator; +import java.util.EnumMap; + +final class IndicatorData { + + final int index; + final int tableIndex; + final Indicator indicator; + final Color defaultColor; + final Color defaultTextColor; + final ThresholdColor[] thresholdColor; + + protected IndicatorData( + final Indicator indicator, + final int index, + final int tableIndex, + final ColorData colorData, + final Display display) { + + this.indicator = indicator; + this.index = index; + this.tableIndex = tableIndex; + this.defaultColor = new Color(display, Utils.toRGB(indicator.defaultColor), 255); + this.defaultTextColor = Utils.darkColor(this.defaultColor.getRGB()) + ? colorData.darkColor + : colorData.lightColor; + + this.thresholdColor = new ThresholdColor[indicator.thresholds.size()]; + final ArrayList sortedThresholds = new ArrayList<>(indicator.thresholds); + sortedThresholds.sort(Comparator.comparing(t -> t.value)); + for (int i = 0; i < indicator.thresholds.size(); i++) { + this.thresholdColor[i] = new ThresholdColor(sortedThresholds.get(i), display, colorData); + } + } + + static EnumMap createFormIndicators( + final Collection indicators, + final Display display, + final ColorData colorData, + final int tableIndexOffset) { + + final EnumMap indicatorMapping = new EnumMap<>(IndicatorType.class); + int i = 0; + for (final Indicator indicator : indicators) { + indicatorMapping.put(indicator.type, new IndicatorData( + indicator, + i, + i + tableIndexOffset, + colorData, + display)); + i++; + } + return indicatorMapping; + } + + static int getWeight(final IndicatorData indicatorData, final double value) { + for (int j = 0; j < indicatorData.thresholdColor.length; j++) { + if (value < indicatorData.thresholdColor[j].value) { + return (j == 0) ? -1 : j - 1; + } + } + + return indicatorData.thresholdColor.length - 1; + } + + static final class ThresholdColor { + final double value; + final Color color; + final Color textColor; + + protected ThresholdColor(final Threshold threshold, final Display display, final ColorData colorData) { + this.value = threshold.value; + this.color = new Color(display, Utils.toRGB(threshold.color), 255); + this.textColor = Utils.darkColor(this.color.getRGB()) + ? colorData.darkColor + : colorData.lightColor; + } + } + } \ No newline at end of file diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/InstructionProcessor.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/InstructionProcessor.java index ff9a4d5e..9128b6ce 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/session/InstructionProcessor.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/session/InstructionProcessor.java @@ -1,165 +1,160 @@ -/* - * Copyright (c) 2019 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.service.session; - -import java.util.Collection; -import java.util.Set; -import java.util.function.Function; -import java.util.function.Predicate; -import java.util.function.Supplier; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -import org.apache.commons.lang3.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Service; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.api.APIMessage; -import ch.ethz.seb.sebserver.gbl.api.JSONMapper; -import ch.ethz.seb.sebserver.gbl.model.Domain; -import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; -import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus; -import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction; -import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction.InstructionType; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCallError; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.DisableClientConnection; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.PropagateInstruction; - -@Lazy -@Service -@GuiProfile -public class InstructionProcessor { - - private static final Logger log = LoggerFactory.getLogger(InstructionProcessor.class); - - private final RestService restService; - private final JSONMapper jsonMapper; - - protected InstructionProcessor(final PageService pageService) { - this.restService = pageService.getRestService(); - this.jsonMapper = pageService.getJSONMapper(); - } - - public void propagateSebQuitInstruction( - final Long examId, - final String connectionToken, - final PageContext pageContext) { - - propagateSebQuitInstruction( - examId, - p -> Stream.of(connectionToken).collect(Collectors.toSet()), - pageContext); - - } - - public void propagateSebQuitInstruction( - final Long examId, - final Function, Set> selectionFunction, - final PageContext pageContext) { - - final Set connectionTokens = selectionFunction - .apply(ClientConnection.getStatusPredicate(ConnectionStatus.ACTIVE)); - - if (connectionTokens.isEmpty()) { - // TODO message - return; - } - - if (log.isDebugEnabled()) { - log.debug("Propagate SEB quit instruction for exam: {} and connections: {}", - examId, - connectionTokens); - } - - final ClientInstruction clientInstruction = new ClientInstruction( - null, - examId, - InstructionType.SEB_QUIT, - StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR), - null); - - processInstruction(() -> { - return this.restService.getBuilder(PropagateInstruction.class) - .withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId)) - .withBody(clientInstruction) - .call() - .getOrThrow(); - }, - pageContext); - - } - - public void disableConnection( - final Long examId, - final Function, Set> selectionFunction, - final PageContext pageContext) { - - final Set connectionTokens = selectionFunction - .apply(ClientConnection.getStatusPredicate( - ConnectionStatus.CONNECTION_REQUESTED, - ConnectionStatus.UNDEFINED, - ConnectionStatus.CLOSED, - ConnectionStatus.AUTHENTICATED)); - - if (connectionTokens.isEmpty()) { - // TOOD message - return; - } - - if (log.isDebugEnabled()) { - log.debug("Disable SEB client connections for exam: {} and connections: {}", - examId, - connectionTokens); - } - - processInstruction(() -> { - return this.restService.getBuilder(DisableClientConnection.class) - .withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId)) - .withFormParam( - Domain.CLIENT_CONNECTION.ATTR_CONNECTION_TOKEN, - StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR)) - .call() - .getOrThrow(); - }, - pageContext); - - } - - private void processInstruction(final Supplier apiCall, final PageContext pageContext) { - try { - final String response = apiCall.get(); - - if (StringUtils.isNotBlank(response)) { - try { - final Collection errorMessage = this.jsonMapper.readValue( - response, - Constants.TYPE_REFERENCE_API_MESSAGE); - - pageContext.notifyUnexpectedError(new RestCallError( - "Failed to propagate SEB client instruction: ", - errorMessage)); - - } catch (final Exception e) { - log.error("Failed to parse error response: {}", response); - } - } - } catch (final Exception e) { - log.error("Failed to propagate SEB client instruction: ", e); - } - } - -} +/* + * Copyright (c) 2019 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.service.session; + +import java.util.Collection; +import java.util.Set; +import java.util.function.Function; +import java.util.function.Predicate; +import java.util.function.Supplier; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.api.APIMessage; +import ch.ethz.seb.sebserver.gbl.api.JSONMapper; +import ch.ethz.seb.sebserver.gbl.model.Domain; +import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; +import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus; +import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction; +import ch.ethz.seb.sebserver.gbl.model.session.ClientInstruction.InstructionType; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCallError; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.DisableClientConnection; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.PropagateInstruction; + +@Lazy +@Service +@GuiProfile +public class InstructionProcessor { + + private static final Logger log = LoggerFactory.getLogger(InstructionProcessor.class); + + private final RestService restService; + private final JSONMapper jsonMapper; + + protected InstructionProcessor(final PageService pageService) { + this.restService = pageService.getRestService(); + this.jsonMapper = pageService.getJSONMapper(); + } + + public void propagateSebQuitInstruction( + final Long examId, + final String connectionToken, + final PageContext pageContext) { + + propagateSebQuitInstruction( + examId, + p -> Stream.of(connectionToken).collect(Collectors.toSet()), + pageContext); + + } + + public void propagateSebQuitInstruction( + final Long examId, + final Function, Set> selectionFunction, + final PageContext pageContext) { + + final Set connectionTokens = selectionFunction + .apply(ClientConnection.getStatusPredicate(ConnectionStatus.ACTIVE)); + + if (connectionTokens.isEmpty()) { + // TODO message + return; + } + + if (log.isDebugEnabled()) { + log.debug("Propagate SEB quit instruction for exam: {} and connections: {}", + examId, + connectionTokens); + } + + final ClientInstruction clientInstruction = new ClientInstruction( + null, + examId, + InstructionType.SEB_QUIT, + StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR), + null); + + processInstruction(() -> this.restService.getBuilder(PropagateInstruction.class) + .withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId)) + .withBody(clientInstruction) + .call() + .getOrThrow(), + pageContext); + + } + + public void disableConnection( + final Long examId, + final Function, Set> selectionFunction, + final PageContext pageContext) { + + final Set connectionTokens = selectionFunction + .apply(ClientConnection.getStatusPredicate( + ConnectionStatus.CONNECTION_REQUESTED, + ConnectionStatus.UNDEFINED, + ConnectionStatus.CLOSED, + ConnectionStatus.AUTHENTICATED)); + + if (connectionTokens.isEmpty()) { + return; + } + + if (log.isDebugEnabled()) { + log.debug("Disable SEB client connections for exam: {} and connections: {}", + examId, + connectionTokens); + } + + processInstruction(() -> this.restService.getBuilder(DisableClientConnection.class) + .withURIVariable(API.PARAM_MODEL_ID, String.valueOf(examId)) + .withFormParam( + Domain.CLIENT_CONNECTION.ATTR_CONNECTION_TOKEN, + StringUtils.join(connectionTokens, Constants.LIST_SEPARATOR)) + .call() + .getOrThrow(), + pageContext); + + } + + private void processInstruction(final Supplier apiCall, final PageContext pageContext) { + try { + final String response = apiCall.get(); + + if (StringUtils.isNotBlank(response)) { + try { + final Collection errorMessage = this.jsonMapper.readValue( + response, + Constants.TYPE_REFERENCE_API_MESSAGE); + + pageContext.notifyUnexpectedError(new RestCallError( + "Failed to propagate SEB client instruction: ", + errorMessage)); + + } catch (final Exception e) { + log.error("Failed to parse error response: {}", response); + } + } + } catch (final Exception e) { + log.error("Failed to propagate SEB client instruction: ", e); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java b/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java index 256ea49f..e8558a46 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java @@ -1,751 +1,743 @@ -/* - * Copyright (c) 2019 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.table; - -import static ch.ethz.seb.sebserver.gui.service.i18n.PolyglotPageService.POLYGLOT_WIDGET_FUNCTION_KEY; - -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.Objects; -import java.util.Set; -import java.util.function.BiConsumer; -import java.util.function.Consumer; -import java.util.function.Function; -import java.util.function.Predicate; -import java.util.function.Supplier; -import java.util.stream.Collectors; - -import org.eclipse.rap.rwt.RWT; -import org.eclipse.swt.SWT; -import org.eclipse.swt.graphics.Point; -import org.eclipse.swt.graphics.Rectangle; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Event; -import org.eclipse.swt.widgets.Table; -import org.eclipse.swt.widgets.TableColumn; -import org.eclipse.swt.widgets.TableItem; -import org.eclipse.swt.widgets.Widget; -import org.joda.time.DateTime; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.util.MultiValueMap; -import org.springframework.web.util.HtmlUtils; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.model.Entity; -import ch.ethz.seb.sebserver.gbl.model.EntityKey; -import ch.ethz.seb.sebserver.gbl.model.GrantEntity; -import ch.ethz.seb.sebserver.gbl.model.Page; -import ch.ethz.seb.sebserver.gbl.model.PageSortOrder; -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.service.page.PageMessageException; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.CurrentUser; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.ImageIcon; -import io.micrometer.core.instrument.util.StringUtils; - -public class EntityTable { - - private static final Logger log = LoggerFactory.getLogger(EntityTable.class); - - private static final LocTextKey DEFAULT_SORT_COLUMN_TOOLTIP_KEY = - new LocTextKey("sebserver.table.column.sort.default.tooltip"); - private static final String COLUMN_DEFINITION = "COLUMN_DEFINITION"; - private static final String TABLE_ROW_DATA = "TABLE_ROW_DATA"; - private static final int HEADER_HEIGHT = 40; - private static final int ROW_HEIGHT = 25; - - private final String name; - private final String filterAttrName; - private final String sortAttrName; - private final String sortOrderAttrName; - private final String currentPageAttrName; - private final boolean markupEnabled; - - final PageService pageService; - final WidgetFactory widgetFactory; - final RestCall> restCall; - final Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter; - final I18nSupport i18nSupport; - final PageContext pageContext; - - final List> columns; - final LocTextKey emptyMessage; - - final Composite composite; - private final TableFilter filter; - private final Table table; - private final TableNavigator navigator; - private final MultiValueMap staticQueryParams; - private final BiConsumer rowDecorator; - private final Consumer> selectionListener; - - int pageNumber = 1; - int pageSize; - String sortColumn = null; - PageSortOrder sortOrder = PageSortOrder.ASCENDING; - boolean columnsWithSameWidth = true; - boolean hideNavigation = false; - - EntityTable( - final String name, - final boolean markupEnabled, - final int type, - final PageContext pageContext, - final RestCall> restCall, - final Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter, - final PageService pageService, - final List> columns, - final int pageSize, - final LocTextKey emptyMessage, - final Function, PageAction> defaultActionFunction, - final boolean hideNavigation, - final MultiValueMap staticQueryParams, - final BiConsumer rowDecorator, - final Consumer> selectionListener) { - - this.name = name; - this.filterAttrName = name + "_filter"; - this.sortAttrName = name + "_sort"; - this.sortOrderAttrName = name + "_sortOrder"; - this.currentPageAttrName = name + "_currentPage"; - this.markupEnabled = markupEnabled; - - this.composite = new Composite(pageContext.getParent(), type); - this.pageService = pageService; - this.i18nSupport = pageService.getI18nSupport(); - this.pageContext = pageContext; - this.widgetFactory = pageService.getWidgetFactory(); - this.restCall = restCall; - this.restCallAdapter = (restCallAdapter != null) ? restCallAdapter : Function.identity(); - this.columns = Utils.immutableListOf(columns); - this.emptyMessage = emptyMessage; - this.hideNavigation = hideNavigation; - - final GridLayout layout = new GridLayout(); - layout.horizontalSpacing = 0; - layout.verticalSpacing = 0; - layout.marginHeight = 0; - this.composite.setLayout(layout); - GridData gridData = new GridData(SWT.FILL, SWT.TOP, true, false); - this.composite.setLayoutData(gridData); - this.staticQueryParams = staticQueryParams; - this.rowDecorator = rowDecorator; - this.selectionListener = selectionListener; - this.pageSize = pageSize; - this.filter = - columns - .stream() - .map(column -> column.getFilterAttribute()) - .filter(Objects::nonNull) - .findFirst() - .isPresent() ? new TableFilter<>(this) : null; - - this.table = this.widgetFactory.tableLocalized(this.composite); - final GridLayout gridLayout = new GridLayout(columns.size(), true); - this.table.setLayout(gridLayout); - gridData = new GridData(SWT.FILL, SWT.TOP, true, false); - this.table.setLayoutData(gridData); - this.table.addListener(SWT.Resize, this::adaptColumnWidth); - @SuppressWarnings("unchecked") - final Consumer locFunction = (Consumer
) this.table.getData(POLYGLOT_WIDGET_FUNCTION_KEY); - final Consumer
newLocFunction = t -> { - updateValues(this); - locFunction.accept(t); - }; - this.table.setData(POLYGLOT_WIDGET_FUNCTION_KEY, newLocFunction); - - this.table.setHeaderVisible(true); - this.table.setLinesVisible(true); - this.table.setData(RWT.CUSTOM_ITEM_HEIGHT, ROW_HEIGHT); - if (this.markupEnabled) { - this.table.setData(RWT.MARKUP_ENABLED, Boolean.TRUE); - } - - if (defaultActionFunction != null) { - final PageAction defaultAction = defaultActionFunction.apply(this); - if (defaultAction != null) { - this.table.addListener(SWT.MouseDoubleClick, event -> { - // if the action has its own selection function, apply this - EntityKey selection = defaultAction.getSingleSelection(); - if (selection == null) { - // otherwise use current selection of this table - selection = getSingleSelection(); - } - if (selection != null) { - this.pageService.executePageAction( - defaultAction.withEntityKey(selection)); - } - }); - } - } - this.table.addListener(SWT.MouseDown, event -> { - if (event.button == Constants.RWT_MOUSE_BUTTON_1) { - return; - } - final Rectangle bounds = event.getBounds(); - final Point point = new Point(bounds.x, bounds.y); - final TableItem item = this.table.getItem(point); - if (item == null) { - return; - } - - for (int i = 0; i < columns.size(); i++) { - final Rectangle itemBoundes = item.getBounds(i); - if (itemBoundes.contains(point)) { - handleCellSelection(item, i); - return; - } - } - }); - - this.table.addListener(SWT.Selection, event -> { - this.notifySelectionChange(); - }); - - this.navigator = new TableNavigator(this); - - createTableColumns(); - this.pageNumber = initCurrentPageFromUserAttr(); - initFilterFromUserAttrs(); - initSortFromUserAttr(); - updateTableRows( - this.pageNumber, - this.pageSize, - this.sortColumn, - this.sortOrder); - } - - public String getName() { - return this.name; - } - - public EntityType getEntityType() { - if (this.restCall != null) { - return this.restCall.getEntityType(); - } - - return null; - } - - public PageContext getPageContext() { - if (this.pageContext == null) { - return null; - } - - return this.pageContext.copy(); - } - - public boolean hasAnyContent() { - return this.table.getItemCount() > 0; - } - - public void setPageSize(final int pageSize) { - this.pageSize = pageSize; - updateTableRows( - this.pageNumber, - this.pageSize, - this.sortColumn, - this.sortOrder); - } - - public void selectPage(final int pageSelection) { - // verify input - this.pageNumber = pageSelection; - if (this.pageNumber < 1) { - this.pageNumber = 1; - } - - updateTableRows( - this.pageNumber, - this.pageSize, - this.sortColumn, - this.sortOrder); - - updateCurrentPageAttr(); - } - - public void reset() { - this.applySort(null); - this.table.setSortColumn(null); - this.table.setSortDirection(SWT.NONE); - applyFilter(); - } - - public void applyFilter() { - try { - - updateFilterUserAttrs(); - this.selectPage(1); - - } catch (final Exception e) { - log.error("Unexpected error while trying to apply filter: ", e); - } - } - - public void applySort(final String columnName) { - try { - this.sortColumn = columnName; - this.sortOrder = PageSortOrder.ASCENDING; - - if (columnName != null) { - updateTableRows( - this.pageNumber, - this.pageSize, - this.sortColumn, - this.sortOrder); - } - - updateSortUserAttr(); - - } catch (final Exception e) { - log.error("Unexpected error while trying to apply sort: ", e); - } - } - - public void changeSortOrder() { - try { - this.sortOrder = (this.sortOrder == PageSortOrder.ASCENDING) - ? PageSortOrder.DESCENDING - : PageSortOrder.ASCENDING; - - updateTableRows( - this.pageNumber, - this.pageSize, - this.sortColumn, - this.sortOrder); - - updateSortUserAttr(); - - } catch (final Exception e) { - log.error("Unexpected error while trying to apply sort: ", e); - } - } - - public EntityKey getSingleSelection() { - final TableItem[] selection = this.table.getSelection(); - if (selection == null || selection.length == 0) { - return null; - } - - return getRowDataId(selection[0]); - } - - public ROW getFirstRowData() { - if (!this.hasAnyContent()) { - return null; - } - - final TableItem item = this.table.getItem(0); - if (item == null) { - return null; - } - - return getRowData(item); - } - - public ROW getSingleSelectedROWData() { - final TableItem[] selection = this.table.getSelection(); - if (selection == null || selection.length == 0) { - return null; - } - - return getRowData(selection[0]); - } - - public Set getSelectedROWData() { - final TableItem[] selection = this.table.getSelection(); - if (selection == null || selection.length == 0) { - return Collections.emptySet(); - } - - return Arrays.asList(selection) - .stream() - .map(this::getRowData) - .collect(Collectors.toSet()); - } - - public Set getSelection() { - return getSelection(null); - } - - public Set getSelection(final Predicate grantCheck) { - final TableItem[] selection = this.table.getSelection(); - if (selection == null) { - return Collections.emptySet(); - } - - return Arrays.asList(selection) - .stream() - .filter(item -> grantCheck == null || grantCheck.test(getRowData(item))) - .map(this::getRowDataId) - .collect(Collectors.toSet()); - } - - public Supplier> getGrantedSelection( - final CurrentUser currentUser, - final LocTextKey denyMessage) { - - return () -> getSelection(e -> { - if (!(e instanceof GrantEntity)) { - return true; - } - if (currentUser.entityGrantCheck((GrantEntity) e).m()) { - return true; - } else { - throw new PageMessageException(denyMessage); - } - }); - } - - private TableColumn getTableColumn(final String name) { - return Arrays.asList(this.table.getColumns()) - .stream() - .filter(col -> { - @SuppressWarnings("unchecked") - final ColumnDefinition def = (ColumnDefinition) col.getData(COLUMN_DEFINITION); - return name.equals(def.columnName); - }) - .findFirst() - .orElse(null); - } - - private void createTableColumns() { - final String sortText = this.i18nSupport.getText(DEFAULT_SORT_COLUMN_TOOLTIP_KEY, ""); - - for (final ColumnDefinition column : this.columns) { - - final LocTextKey _tooltip = column.getTooltip(); - final LocTextKey tooltip = (_tooltip != null && this.i18nSupport.hasText(_tooltip)) - ? (column.isSortable()) - ? new LocTextKey(_tooltip.name, sortText) - : new LocTextKey(_tooltip.name, "") - : (column.isSortable()) - ? DEFAULT_SORT_COLUMN_TOOLTIP_KEY - : null; - - final TableColumn tableColumn = this.widgetFactory.tableColumnLocalized( - this.table, - column.displayName, - tooltip); - - tableColumn.addListener(SWT.Resize, this::adaptColumnWidthChange); - tableColumn.setData(COLUMN_DEFINITION, column); - - if (column.isSortable()) { - tableColumn.addListener(SWT.Selection, event -> { - if (!column.columnName.equals(this.sortColumn)) { - applySort(column.columnName); - this.table.setSortColumn(tableColumn); - this.table.setSortDirection(SWT.UP); - } else { - changeSortOrder(); - this.table.setSortDirection( - (this.sortOrder == PageSortOrder.ASCENDING) ? SWT.UP : SWT.DOWN); - } - }); - } - - if (column.getWidthProportion() > 0) { - this.columnsWithSameWidth = false; - } - } - } - - private void updateTableRows( - final int pageNumber, - final int pageSize, - final String sortColumn, - final PageSortOrder sortOrder) { - - // first remove all rows if there are some - this.table.removeAll(); - - // get page data and create rows - this.restCall.newBuilder() - .withPaging(pageNumber, pageSize) - .withSorting(sortColumn, sortOrder) - .withQueryParams((this.filter != null) ? this.filter.getFilterParameter() : null) - .withQueryParams(this.staticQueryParams) - .apply(this.restCallAdapter) - .call() - .map(this::createTableRowsFromPage) - .map(this.navigator::update) - .onError(this.pageContext::notifyUnexpectedError); - - this.composite.getParent().layout(true, true); - PageService.updateScrolledComposite(this.composite); - this.notifySelectionChange(); - } - - private Page createTableRowsFromPage(final Page page) { - if (page.isEmpty()) { - final GridData gridData = (GridData) this.table.getLayoutData(); - gridData.heightHint = ROW_HEIGHT; - return page; - } - - final GridData gridData = (GridData) this.table.getLayoutData(); - gridData.heightHint = (this.pageNumber > 1) - ? (this.pageSize * ROW_HEIGHT) + HEADER_HEIGHT - : (page.content.size() * ROW_HEIGHT) + HEADER_HEIGHT; - - for (final ROW row : page.content) { - final TableItem item = new TableItem(this.table, SWT.NONE); - if (this.markupEnabled) { - item.setData(RWT.MARKUP_ENABLED, Boolean.TRUE); - } - item.setData(TABLE_ROW_DATA, row); - if (this.rowDecorator != null) { - this.rowDecorator.accept(item, row); - } - - int index = 0; - for (final ColumnDefinition column : this.columns) { - setValueToCell(item, index, column, column.valueSupplier.apply(row)); - index++; - } - } - - return page; - } - - private void adaptColumnWidth(final Event event) { - try { - int currentTableWidth = this.table.getParent().getClientArea().width; - // If we have all columns with filter we need some more space for the - // filter actions in the right hand side. This tweak gives enough space for that - if (this.filter != null && this.columns.size() == this.filter.size()) { - currentTableWidth -= 60; - } - - // The proportion size, the sum of all given proportion values - final int pSize = this.columns - .stream() - .filter(c -> c.getWidthProportion() > 0) - .reduce(0, - (acc, c) -> acc + c.getWidthProportion(), - (acc1, acc2) -> acc1 + acc2); - - // The unit size either with proportion or for a entire column if all columns are equal in size - final int columnUnitSize = (pSize > 0) - ? currentTableWidth / pSize - : currentTableWidth / this.columns.size(); - - // Apply the column width for each column - int index = 0; - for (final ColumnDefinition column : this.columns) { - final TableColumn tableColumn = this.table.getColumn(index); - final int newWidth = (pSize > 0) - ? columnUnitSize * column.getWidthProportion() - : columnUnitSize; - tableColumn.setWidth(newWidth); - if (this.filter != null) { - this.filter.adaptColumnWidth(this.table.indexOf(tableColumn), newWidth); - } - if (index == this.columns.size() - 1) { - tableColumn.setWidth(newWidth - 10); - } - - index++; - } - } catch (final Exception e) { - log.warn("Failed to adaptColumnWidth: ", e); - } - } - - private void adaptColumnWidthChange(final Event event) { - final Widget widget = event.widget; - if (widget instanceof TableColumn) { - final TableColumn tableColumn = ((TableColumn) widget); - if (this.filter != null) { - this.filter.adaptColumnWidth( - this.table.indexOf(tableColumn), - tableColumn.getWidth()); - } - } - } - - @SuppressWarnings("unchecked") - private ROW getRowData(final TableItem item) { - return (ROW) item.getData(TABLE_ROW_DATA); - } - - private EntityKey getRowDataId(final TableItem item) { - return getRowData(item).getEntityKey(); - } - - private void updateValues(final EntityTable table) { - final TableItem[] items = table.table.getItems(); - final TableColumn[] columns = table.table.getColumns(); - for (int i = 0; i < columns.length; i++) { - final ColumnDefinition columnDefinition = table.columns.get(i); - if (columnDefinition.isLocalized()) { - for (int j = 0; j < items.length; j++) { - @SuppressWarnings("unchecked") - final ROW rowData = (ROW) items[j].getData(TABLE_ROW_DATA); - setValueToCell(items[j], i, columnDefinition, columnDefinition.valueSupplier.apply(rowData)); - } - } - } - } - - private void setValueToCell( - final TableItem item, - final int index, - final ColumnDefinition columnDefinition, - final Object value) { - - if (value instanceof Boolean) { - addBooleanCell(item, index, value); - } else if (value instanceof DateTime) { - item.setText(index, this.i18nSupport.formatDisplayDateTime((DateTime) value)); - } else { - item.setText(index, renderTextValue( - (value != null) ? String.valueOf(value) : null, - columnDefinition)); - } - } - - private String renderTextValue(final String raw, final ColumnDefinition columnDefinition) { - if (StringUtils.isBlank(raw)) { - return Constants.EMPTY_NOTE; - } - if (!this.markupEnabled) { - return raw; - } - - if (columnDefinition.markupEnabled()) { - return raw; - } else { - return HtmlUtils.htmlEscapeHex(raw); - } - } - - private static void addBooleanCell(final TableItem item, final int index, final Object value) { - if ((Boolean) value) { - item.setImage(index, ImageIcon.YES.getImage(item.getDisplay())); - } else { - item.setImage(index, ImageIcon.NO.getImage(item.getDisplay())); - } - } - - private void handleCellSelection(final TableItem item, final int index) { - // TODO handle selection tool-tips on cell level - } - - private void notifySelectionChange() { - if (this.selectionListener == null) { - return; - } - - this.selectionListener.accept(this.getSelectedROWData()); - } - - private void updateCurrentPageAttr() { - try { - this.pageService - .getCurrentUser() - .putAttribute(this.currentPageAttrName, String.valueOf(this.pageNumber)); - } catch (final Exception e) { - log.error("Failed to put current page attribute to current user attributes", e); - } - } - - private int initCurrentPageFromUserAttr() { - try { - final String currentPage = this.pageService - .getCurrentUser() - .getAttribute(this.currentPageAttrName); - if (StringUtils.isNotBlank(currentPage)) { - return Integer.parseInt(currentPage); - } else { - return 1; - } - } catch (final Exception e) { - log.error("Failed to get sort attribute form current user attributes", e); - return 1; - } - } - - private void updateSortUserAttr() { - try { - this.pageService - .getCurrentUser() - .putAttribute(this.sortAttrName, this.sortColumn); - this.pageService - .getCurrentUser() - .putAttribute(this.sortOrderAttrName, this.sortOrder.name()); - } catch (final Exception e) { - log.error("Failed to put sort attribute to current user attributes", e); - } - } - - private void initSortFromUserAttr() { - try { - final String sort = this.pageService - .getCurrentUser() - .getAttribute(this.sortAttrName); - if (StringUtils.isNotBlank(sort)) { - this.sortColumn = sort; - final TableColumn tableColumn = getTableColumn(sort); - if (tableColumn != null) { - this.table.setSortColumn(tableColumn); - } - } - - final String sortOrder = this.pageService - .getCurrentUser() - .getAttribute(this.sortOrderAttrName); - if (StringUtils.isNotBlank(sortOrder)) { - this.sortOrder = PageSortOrder.valueOf(sortOrder); - this.table.setSortDirection(this.sortOrder == PageSortOrder.ASCENDING ? SWT.UP : SWT.DOWN); - } - - } catch (final Exception e) { - log.error("Failed to get sort attribute form current user attributes", e); - } - } - - private void updateFilterUserAttrs() { - if (this.filter != null) { - try { - this.pageService - .getCurrentUser() - .putAttribute(this.filterAttrName, this.filter.getFilterAttributes()); - } catch (final Exception e) { - log.error("Failed to put filter attributes to current user attributes", e); - } - } - } - - private void initFilterFromUserAttrs() { - if (this.filter != null) { - try { - this.filter.setFilterAttributes( - this.pageService - .getCurrentUser() - .getAttribute(this.filterAttrName)); - } catch (final Exception e) { - log.error("Failed to get filter attributes form current user attributes", e); - } - } - } - -} +/* + * Copyright (c) 2019 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.table; + +import static ch.ethz.seb.sebserver.gui.service.i18n.PolyglotPageService.POLYGLOT_WIDGET_FUNCTION_KEY; + +import java.util.Arrays; +import java.util.Collections; +import java.util.List; +import java.util.Objects; +import java.util.Set; +import java.util.function.BiConsumer; +import java.util.function.Consumer; +import java.util.function.Function; +import java.util.function.Predicate; +import java.util.function.Supplier; +import java.util.stream.Collectors; + +import org.eclipse.rap.rwt.RWT; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Point; +import org.eclipse.swt.graphics.Rectangle; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Table; +import org.eclipse.swt.widgets.TableColumn; +import org.eclipse.swt.widgets.TableItem; +import org.eclipse.swt.widgets.Widget; +import org.joda.time.DateTime; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.util.MultiValueMap; +import org.springframework.web.util.HtmlUtils; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.model.Entity; +import ch.ethz.seb.sebserver.gbl.model.EntityKey; +import ch.ethz.seb.sebserver.gbl.model.GrantEntity; +import ch.ethz.seb.sebserver.gbl.model.Page; +import ch.ethz.seb.sebserver.gbl.model.PageSortOrder; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.service.page.PageMessageException; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.CurrentUser; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.ImageIcon; +import io.micrometer.core.instrument.util.StringUtils; + +public class EntityTable { + + private static final Logger log = LoggerFactory.getLogger(EntityTable.class); + + private static final LocTextKey DEFAULT_SORT_COLUMN_TOOLTIP_KEY = + new LocTextKey("sebserver.table.column.sort.default.tooltip"); + private static final String COLUMN_DEFINITION = "COLUMN_DEFINITION"; + private static final String TABLE_ROW_DATA = "TABLE_ROW_DATA"; + private static final int HEADER_HEIGHT = 40; + private static final int ROW_HEIGHT = 25; + + private final String name; + private final String filterAttrName; + private final String sortAttrName; + private final String sortOrderAttrName; + private final String currentPageAttrName; + private final boolean markupEnabled; + + final PageService pageService; + final WidgetFactory widgetFactory; + final RestCall> restCall; + final Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter; + final I18nSupport i18nSupport; + final PageContext pageContext; + + final List> columns; + final LocTextKey emptyMessage; + + final Composite composite; + private final TableFilter filter; + private final Table table; + private final TableNavigator navigator; + private final MultiValueMap staticQueryParams; + private final BiConsumer rowDecorator; + private final Consumer> selectionListener; + + int pageNumber; + int pageSize; + String sortColumn = null; + PageSortOrder sortOrder = PageSortOrder.ASCENDING; + boolean columnsWithSameWidth = true; + boolean hideNavigation; + + EntityTable( + final String name, + final boolean markupEnabled, + final int type, + final PageContext pageContext, + final RestCall> restCall, + final Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter, + final PageService pageService, + final List> columns, + final int pageSize, + final LocTextKey emptyMessage, + final Function, PageAction> defaultActionFunction, + final boolean hideNavigation, + final MultiValueMap staticQueryParams, + final BiConsumer rowDecorator, + final Consumer> selectionListener) { + + this.name = name; + this.filterAttrName = name + "_filter"; + this.sortAttrName = name + "_sort"; + this.sortOrderAttrName = name + "_sortOrder"; + this.currentPageAttrName = name + "_currentPage"; + this.markupEnabled = markupEnabled; + + this.composite = new Composite(pageContext.getParent(), type); + this.pageService = pageService; + this.i18nSupport = pageService.getI18nSupport(); + this.pageContext = pageContext; + this.widgetFactory = pageService.getWidgetFactory(); + this.restCall = restCall; + this.restCallAdapter = (restCallAdapter != null) ? restCallAdapter : Function.identity(); + this.columns = Utils.immutableListOf(columns); + this.emptyMessage = emptyMessage; + this.hideNavigation = hideNavigation; + + final GridLayout layout = new GridLayout(); + layout.horizontalSpacing = 0; + layout.verticalSpacing = 0; + layout.marginHeight = 0; + this.composite.setLayout(layout); + GridData gridData = new GridData(SWT.FILL, SWT.TOP, true, false); + this.composite.setLayoutData(gridData); + this.staticQueryParams = staticQueryParams; + this.rowDecorator = rowDecorator; + this.selectionListener = selectionListener; + this.pageSize = pageSize; + this.filter = columns + .stream() + .map(ColumnDefinition::getFilterAttribute) + .anyMatch(Objects::nonNull) ? new TableFilter<>(this) : null; + + this.table = this.widgetFactory.tableLocalized(this.composite); + final GridLayout gridLayout = new GridLayout(columns.size(), true); + this.table.setLayout(gridLayout); + gridData = new GridData(SWT.FILL, SWT.TOP, true, false); + this.table.setLayoutData(gridData); + this.table.addListener(SWT.Resize, this::adaptColumnWidth); + @SuppressWarnings("unchecked") + final Consumer
locFunction = (Consumer
) this.table.getData(POLYGLOT_WIDGET_FUNCTION_KEY); + final Consumer
newLocFunction = t -> { + updateValues(this); + locFunction.accept(t); + }; + this.table.setData(POLYGLOT_WIDGET_FUNCTION_KEY, newLocFunction); + + this.table.setHeaderVisible(true); + this.table.setLinesVisible(true); + this.table.setData(RWT.CUSTOM_ITEM_HEIGHT, ROW_HEIGHT); + if (this.markupEnabled) { + this.table.setData(RWT.MARKUP_ENABLED, Boolean.TRUE); + } + + if (defaultActionFunction != null) { + final PageAction defaultAction = defaultActionFunction.apply(this); + if (defaultAction != null) { + this.table.addListener(SWT.MouseDoubleClick, event -> { + // if the action has its own selection function, apply this + EntityKey selection = defaultAction.getSingleSelection(); + if (selection == null) { + // otherwise use current selection of this table + selection = getSingleSelection(); + } + if (selection != null) { + this.pageService.executePageAction( + defaultAction.withEntityKey(selection)); + } + }); + } + } + this.table.addListener(SWT.MouseDown, event -> { + if (event.button == Constants.RWT_MOUSE_BUTTON_1) { + return; + } + final Rectangle bounds = event.getBounds(); + final Point point = new Point(bounds.x, bounds.y); + final TableItem item = this.table.getItem(point); + if (item == null) { + return; + } + + for (int i = 0; i < columns.size(); i++) { + final Rectangle itemBounds = item.getBounds(i); + if (itemBounds.contains(point)) { + handleCellSelection(item, i); + return; + } + } + }); + + this.table.addListener(SWT.Selection, event -> this.notifySelectionChange()); + + this.navigator = new TableNavigator(this); + + createTableColumns(); + this.pageNumber = initCurrentPageFromUserAttr(); + initFilterFromUserAttrs(); + initSortFromUserAttr(); + updateTableRows( + this.pageNumber, + this.pageSize, + this.sortColumn, + this.sortOrder); + } + + public String getName() { + return this.name; + } + + public EntityType getEntityType() { + if (this.restCall != null) { + return this.restCall.getEntityType(); + } + + return null; + } + + public PageContext getPageContext() { + if (this.pageContext == null) { + return null; + } + + return this.pageContext.copy(); + } + + public boolean hasAnyContent() { + return this.table.getItemCount() > 0; + } + + public void setPageSize(final int pageSize) { + this.pageSize = pageSize; + updateTableRows( + this.pageNumber, + this.pageSize, + this.sortColumn, + this.sortOrder); + } + + public void selectPage(final int pageSelection) { + // verify input + this.pageNumber = pageSelection; + if (this.pageNumber < 1) { + this.pageNumber = 1; + } + + updateTableRows( + this.pageNumber, + this.pageSize, + this.sortColumn, + this.sortOrder); + + updateCurrentPageAttr(); + } + + public void reset() { + this.applySort(null); + this.table.setSortColumn(null); + this.table.setSortDirection(SWT.NONE); + applyFilter(); + } + + public void applyFilter() { + try { + + updateFilterUserAttrs(); + this.selectPage(1); + + } catch (final Exception e) { + log.error("Unexpected error while trying to apply filter: ", e); + } + } + + public void applySort(final String columnName) { + try { + this.sortColumn = columnName; + this.sortOrder = PageSortOrder.ASCENDING; + + if (columnName != null) { + updateTableRows( + this.pageNumber, + this.pageSize, + this.sortColumn, + this.sortOrder); + } + + updateSortUserAttr(); + + } catch (final Exception e) { + log.error("Unexpected error while trying to apply sort: ", e); + } + } + + public void changeSortOrder() { + try { + this.sortOrder = (this.sortOrder == PageSortOrder.ASCENDING) + ? PageSortOrder.DESCENDING + : PageSortOrder.ASCENDING; + + updateTableRows( + this.pageNumber, + this.pageSize, + this.sortColumn, + this.sortOrder); + + updateSortUserAttr(); + + } catch (final Exception e) { + log.error("Unexpected error while trying to apply sort: ", e); + } + } + + public EntityKey getSingleSelection() { + final TableItem[] selection = this.table.getSelection(); + if (selection == null || selection.length == 0) { + return null; + } + + return getRowDataId(selection[0]); + } + + public ROW getFirstRowData() { + if (!this.hasAnyContent()) { + return null; + } + + final TableItem item = this.table.getItem(0); + if (item == null) { + return null; + } + + return getRowData(item); + } + + public ROW getSingleSelectedROWData() { + final TableItem[] selection = this.table.getSelection(); + if (selection == null || selection.length == 0) { + return null; + } + + return getRowData(selection[0]); + } + + public Set getSelectedROWData() { + final TableItem[] selection = this.table.getSelection(); + if (selection == null || selection.length == 0) { + return Collections.emptySet(); + } + + return Arrays.stream(selection) + .map(this::getRowData) + .collect(Collectors.toSet()); + } + + public Set getSelection() { + return getSelection(null); + } + + public Set getSelection(final Predicate grantCheck) { + final TableItem[] selection = this.table.getSelection(); + if (selection == null) { + return Collections.emptySet(); + } + + return Arrays.stream(selection) + .filter(item -> grantCheck == null || grantCheck.test(getRowData(item))) + .map(this::getRowDataId) + .collect(Collectors.toSet()); + } + + public Supplier> getGrantedSelection( + final CurrentUser currentUser, + final LocTextKey denyMessage) { + + return () -> getSelection(e -> { + if (!(e instanceof GrantEntity)) { + return true; + } + if (currentUser.entityGrantCheck((GrantEntity) e).m()) { + return true; + } else { + throw new PageMessageException(denyMessage); + } + }); + } + + private TableColumn getTableColumn(final String name) { + return Arrays.stream(this.table.getColumns()) + .filter(col -> { + @SuppressWarnings("unchecked") + final ColumnDefinition def = (ColumnDefinition) col.getData(COLUMN_DEFINITION); + return name.equals(def.columnName); + }) + .findFirst() + .orElse(null); + } + + private void createTableColumns() { + final String sortText = this.i18nSupport.getText(DEFAULT_SORT_COLUMN_TOOLTIP_KEY, ""); + + for (final ColumnDefinition column : this.columns) { + + final LocTextKey _tooltip = column.getTooltip(); + final LocTextKey tooltip = (_tooltip != null && this.i18nSupport.hasText(_tooltip)) + ? (column.isSortable()) + ? new LocTextKey(_tooltip.name, sortText) + : new LocTextKey(_tooltip.name, "") + : (column.isSortable()) + ? DEFAULT_SORT_COLUMN_TOOLTIP_KEY + : null; + + final TableColumn tableColumn = this.widgetFactory.tableColumnLocalized( + this.table, + column.displayName, + tooltip); + + tableColumn.addListener(SWT.Resize, this::adaptColumnWidthChange); + tableColumn.setData(COLUMN_DEFINITION, column); + + if (column.isSortable()) { + tableColumn.addListener(SWT.Selection, event -> { + if (!column.columnName.equals(this.sortColumn)) { + applySort(column.columnName); + this.table.setSortColumn(tableColumn); + this.table.setSortDirection(SWT.UP); + } else { + changeSortOrder(); + this.table.setSortDirection( + (this.sortOrder == PageSortOrder.ASCENDING) ? SWT.UP : SWT.DOWN); + } + }); + } + + if (column.getWidthProportion() > 0) { + this.columnsWithSameWidth = false; + } + } + } + + private void updateTableRows( + final int pageNumber, + final int pageSize, + final String sortColumn, + final PageSortOrder sortOrder) { + + // first remove all rows if there are some + this.table.removeAll(); + + // get page data and create rows + this.restCall.newBuilder() + .withPaging(pageNumber, pageSize) + .withSorting(sortColumn, sortOrder) + .withQueryParams((this.filter != null) ? this.filter.getFilterParameter() : null) + .withQueryParams(this.staticQueryParams) + .apply(this.restCallAdapter) + .call() + .map(this::createTableRowsFromPage) + .map(this.navigator::update) + .onError(this.pageContext::notifyUnexpectedError); + + this.composite.getParent().layout(true, true); + PageService.updateScrolledComposite(this.composite); + this.notifySelectionChange(); + } + + private Page createTableRowsFromPage(final Page page) { + if (page.isEmpty()) { + final GridData gridData = (GridData) this.table.getLayoutData(); + gridData.heightHint = ROW_HEIGHT; + return page; + } + + final GridData gridData = (GridData) this.table.getLayoutData(); + gridData.heightHint = (this.pageNumber > 1) + ? (this.pageSize * ROW_HEIGHT) + HEADER_HEIGHT + : (page.content.size() * ROW_HEIGHT) + HEADER_HEIGHT; + + for (final ROW row : page.content) { + final TableItem item = new TableItem(this.table, SWT.NONE); + if (this.markupEnabled) { + item.setData(RWT.MARKUP_ENABLED, Boolean.TRUE); + } + item.setData(TABLE_ROW_DATA, row); + if (this.rowDecorator != null) { + this.rowDecorator.accept(item, row); + } + + int index = 0; + for (final ColumnDefinition column : this.columns) { + setValueToCell(item, index, column, column.valueSupplier.apply(row)); + index++; + } + } + + return page; + } + + private void adaptColumnWidth(final Event event) { + try { + int currentTableWidth = this.table.getParent().getClientArea().width; + // If we have all columns with filter we need some more space for the + // filter actions in the right hand side. This tweak gives enough space for that + if (this.filter != null && this.columns.size() == this.filter.size()) { + currentTableWidth -= 60; + } + + // The proportion size, the sum of all given proportion values + final int pSize = this.columns + .stream() + .filter(c -> c.getWidthProportion() > 0) + .reduce(0, + (acc, c) -> acc + c.getWidthProportion(), + Integer::sum); + + // The unit size either with proportion or for a entire column if all columns are equal in size + final int columnUnitSize = (pSize > 0) + ? currentTableWidth / pSize + : currentTableWidth / this.columns.size(); + + // Apply the column width for each column + int index = 0; + for (final ColumnDefinition column : this.columns) { + final TableColumn tableColumn = this.table.getColumn(index); + final int newWidth = (pSize > 0) + ? columnUnitSize * column.getWidthProportion() + : columnUnitSize; + tableColumn.setWidth(newWidth); + if (this.filter != null) { + this.filter.adaptColumnWidth(this.table.indexOf(tableColumn), newWidth); + } + if (index == this.columns.size() - 1) { + tableColumn.setWidth(newWidth - 10); + } + + index++; + } + } catch (final Exception e) { + log.warn("Failed to adaptColumnWidth: ", e); + } + } + + private void adaptColumnWidthChange(final Event event) { + final Widget widget = event.widget; + if (widget instanceof TableColumn) { + final TableColumn tableColumn = ((TableColumn) widget); + if (this.filter != null) { + this.filter.adaptColumnWidth( + this.table.indexOf(tableColumn), + tableColumn.getWidth()); + } + } + } + + @SuppressWarnings("unchecked") + private ROW getRowData(final TableItem item) { + return (ROW) item.getData(TABLE_ROW_DATA); + } + + private EntityKey getRowDataId(final TableItem item) { + return getRowData(item).getEntityKey(); + } + + private void updateValues(final EntityTable table) { + final TableItem[] items = table.table.getItems(); + final TableColumn[] columns = table.table.getColumns(); + for (int i = 0; i < columns.length; i++) { + final ColumnDefinition columnDefinition = table.columns.get(i); + if (columnDefinition.isLocalized()) { + for (int j = 0; j < items.length; j++) { + @SuppressWarnings("unchecked") + final ROW rowData = (ROW) items[j].getData(TABLE_ROW_DATA); + setValueToCell(items[j], i, columnDefinition, columnDefinition.valueSupplier.apply(rowData)); + } + } + } + } + + private void setValueToCell( + final TableItem item, + final int index, + final ColumnDefinition columnDefinition, + final Object value) { + + if (value instanceof Boolean) { + addBooleanCell(item, index, value); + } else if (value instanceof DateTime) { + item.setText(index, this.i18nSupport.formatDisplayDateTime((DateTime) value)); + } else { + item.setText(index, renderTextValue( + (value != null) ? String.valueOf(value) : null, + columnDefinition)); + } + } + + private String renderTextValue(final String raw, final ColumnDefinition columnDefinition) { + if (StringUtils.isBlank(raw)) { + return Constants.EMPTY_NOTE; + } + if (!this.markupEnabled) { + return raw; + } + + if (columnDefinition.markupEnabled()) { + return raw; + } else { + return HtmlUtils.htmlEscapeHex(raw); + } + } + + private static void addBooleanCell(final TableItem item, final int index, final Object value) { + if ((Boolean) value) { + item.setImage(index, ImageIcon.YES.getImage(item.getDisplay())); + } else { + item.setImage(index, ImageIcon.NO.getImage(item.getDisplay())); + } + } + + private void handleCellSelection(final TableItem item, final int index) { + // TODO handle selection tool-tips on cell level + } + + private void notifySelectionChange() { + if (this.selectionListener == null) { + return; + } + + this.selectionListener.accept(this.getSelectedROWData()); + } + + private void updateCurrentPageAttr() { + try { + this.pageService + .getCurrentUser() + .putAttribute(this.currentPageAttrName, String.valueOf(this.pageNumber)); + } catch (final Exception e) { + log.error("Failed to put current page attribute to current user attributes", e); + } + } + + private int initCurrentPageFromUserAttr() { + try { + final String currentPage = this.pageService + .getCurrentUser() + .getAttribute(this.currentPageAttrName); + if (StringUtils.isNotBlank(currentPage)) { + return Integer.parseInt(currentPage); + } else { + return 1; + } + } catch (final Exception e) { + log.error("Failed to get sort attribute form current user attributes", e); + return 1; + } + } + + private void updateSortUserAttr() { + try { + this.pageService + .getCurrentUser() + .putAttribute(this.sortAttrName, this.sortColumn); + this.pageService + .getCurrentUser() + .putAttribute(this.sortOrderAttrName, this.sortOrder.name()); + } catch (final Exception e) { + log.error("Failed to put sort attribute to current user attributes", e); + } + } + + private void initSortFromUserAttr() { + try { + final String sort = this.pageService + .getCurrentUser() + .getAttribute(this.sortAttrName); + if (StringUtils.isNotBlank(sort)) { + this.sortColumn = sort; + final TableColumn tableColumn = getTableColumn(sort); + if (tableColumn != null) { + this.table.setSortColumn(tableColumn); + } + } + + final String sortOrder = this.pageService + .getCurrentUser() + .getAttribute(this.sortOrderAttrName); + if (StringUtils.isNotBlank(sortOrder)) { + this.sortOrder = PageSortOrder.valueOf(sortOrder); + this.table.setSortDirection(this.sortOrder == PageSortOrder.ASCENDING ? SWT.UP : SWT.DOWN); + } + + } catch (final Exception e) { + log.error("Failed to get sort attribute form current user attributes", e); + } + } + + private void updateFilterUserAttrs() { + if (this.filter != null) { + try { + this.pageService + .getCurrentUser() + .putAttribute(this.filterAttrName, this.filter.getFilterAttributes()); + } catch (final Exception e) { + log.error("Failed to put filter attributes to current user attributes", e); + } + } + } + + private void initFilterFromUserAttrs() { + if (this.filter != null) { + try { + this.filter.setFilterAttributes( + this.pageService + .getCurrentUser() + .getAttribute(this.filterAttrName)); + } catch (final Exception e) { + log.error("Failed to get filter attributes form current user attributes", e); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableBuilder.java b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableBuilder.java index 2b28beec..38df2298 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableBuilder.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableBuilder.java @@ -1,174 +1,174 @@ -/* - * Copyright (c) 2019 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.table; - -import java.util.ArrayList; -import java.util.List; -import java.util.Set; -import java.util.function.BiConsumer; -import java.util.function.BooleanSupplier; -import java.util.function.Consumer; -import java.util.function.Function; -import java.util.function.Supplier; - -import org.eclipse.swt.SWT; -import org.eclipse.swt.widgets.TableItem; -import org.springframework.util.LinkedMultiValueMap; -import org.springframework.util.MultiValueMap; - -import ch.ethz.seb.sebserver.gbl.model.Entity; -import ch.ethz.seb.sebserver.gbl.model.Page; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.page.PageContext; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction; -import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; - -public class TableBuilder { - - private final String name; - private final PageService pageService; - final RestCall> restCall; - private final MultiValueMap staticQueryParams; - final List> columns = new ArrayList<>(); - LocTextKey emptyMessage; - private Function, PageAction> defaultActionFunction; - private int pageSize = -1; - private int type = SWT.NONE; - private boolean hideNavigation = false; - private Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter; - private BiConsumer rowDecorator; - private Consumer> selectionListener; - private boolean markupEnabled = false; - - public TableBuilder( - final String name, - final PageService pageService, - final RestCall> restCall) { - - this.name = name; - this.pageService = pageService; - this.restCall = restCall; - this.staticQueryParams = new LinkedMultiValueMap<>(); - } - - public TableBuilder hideNavigation() { - this.hideNavigation = true; - return this; - } - - public TableBuilder withEmptyMessage(final LocTextKey emptyMessage) { - this.emptyMessage = emptyMessage; - return this; - } - - public TableBuilder withPaging(final int pageSize) { - this.pageSize = pageSize; - return this; - } - - public TableBuilder withColumn(final ColumnDefinition columnDefinition) { - this.columns.add(columnDefinition); - return this; - } - - public TableBuilder withMarkup() { - this.markupEnabled = true; - return this; - } - - public TableBuilder withColumnIf( - final BooleanSupplier condition, - final Supplier> columnDefSupplier) { - - if (condition != null && condition.getAsBoolean()) { - this.columns.add(columnDefSupplier.get()); - } - return this; - } - - public TableBuilder withRestCallAdapter( - final Function>.RestCallBuilder, RestCall>.RestCallBuilder> adapter) { - - this.restCallAdapter = adapter; - return this; - } - - public TableBuilder withMultiselection() { - this.type |= SWT.MULTI; - return this; - } - - public TableBuilder withSelectionListener(final Consumer> selectionListener) { - this.selectionListener = selectionListener; - return this; - } - - public TableBuilder withStaticFilter(final String name, final String value) { - this.staticQueryParams.add(name, value); - return this; - } - - public TableBuilder withDefaultActionIf( - final BooleanSupplier condition, - final Supplier actionSupplier) { - - if (condition.getAsBoolean()) { - return withDefaultAction(actionSupplier.get()); - } - - return this; - } - - public TableBuilder withDefaultAction(final PageAction action) { - this.defaultActionFunction = table -> PageAction.copyOf(action); - return this; - } - - public TableBuilder withDefaultActionIf( - final BooleanSupplier condition, - final Function, PageAction> defaultActionFunction) { - - if (condition.getAsBoolean()) { - return withDefaultAction(defaultActionFunction); - } - - return this; - } - - public TableBuilder withDefaultAction(final Function, PageAction> defaultActionFunction) { - this.defaultActionFunction = defaultActionFunction; - return this; - } - - public TableBuilder withRowDecorator(final BiConsumer rowDecorator) { - this.rowDecorator = rowDecorator; - return this; - } - - public EntityTable compose(final PageContext pageContext) { - return new EntityTable<>( - this.name, - this.markupEnabled, - this.type, - pageContext, - this.restCall, - this.restCallAdapter, - this.pageService, - this.columns, - this.pageSize, - this.emptyMessage, - this.defaultActionFunction, - this.hideNavigation, - this.staticQueryParams, - this.rowDecorator, - this.selectionListener); - } - -} +/* + * Copyright (c) 2019 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.table; + +import java.util.ArrayList; +import java.util.List; +import java.util.Set; +import java.util.function.BiConsumer; +import java.util.function.BooleanSupplier; +import java.util.function.Consumer; +import java.util.function.Function; +import java.util.function.Supplier; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.TableItem; +import org.springframework.util.LinkedMultiValueMap; +import org.springframework.util.MultiValueMap; + +import ch.ethz.seb.sebserver.gbl.model.Entity; +import ch.ethz.seb.sebserver.gbl.model.Page; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.page.PageContext; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction; +import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall; + +public class TableBuilder { + + private final String name; + private final PageService pageService; + final RestCall> restCall; + private final MultiValueMap staticQueryParams; + final List> columns = new ArrayList<>(); + LocTextKey emptyMessage; + private Function, PageAction> defaultActionFunction; + private int pageSize = -1; + private int type = SWT.NONE; + private boolean hideNavigation = false; + private Function>.RestCallBuilder, RestCall>.RestCallBuilder> restCallAdapter; + private BiConsumer rowDecorator; + private Consumer> selectionListener; + private boolean markupEnabled = false; + + public TableBuilder( + final String name, + final PageService pageService, + final RestCall> restCall) { + + this.name = name; + this.pageService = pageService; + this.restCall = restCall; + this.staticQueryParams = new LinkedMultiValueMap<>(); + } + + public TableBuilder hideNavigation() { + this.hideNavigation = true; + return this; + } + + public TableBuilder withEmptyMessage(final LocTextKey emptyMessage) { + this.emptyMessage = emptyMessage; + return this; + } + + public TableBuilder withPaging(final int pageSize) { + this.pageSize = pageSize; + return this; + } + + public TableBuilder withColumn(final ColumnDefinition columnDefinition) { + this.columns.add(columnDefinition); + return this; + } + + public TableBuilder withMarkup() { + this.markupEnabled = true; + return this; + } + + public TableBuilder withColumnIf( + final BooleanSupplier condition, + final Supplier> columnDefSupplier) { + + if (condition != null && condition.getAsBoolean()) { + this.columns.add(columnDefSupplier.get()); + } + return this; + } + + public TableBuilder withRestCallAdapter( + final Function>.RestCallBuilder, RestCall>.RestCallBuilder> adapter) { + + this.restCallAdapter = adapter; + return this; + } + + public TableBuilder withMultiSelection() { + this.type |= SWT.MULTI; + return this; + } + + public TableBuilder withSelectionListener(final Consumer> selectionListener) { + this.selectionListener = selectionListener; + return this; + } + + public TableBuilder withStaticFilter(final String name, final String value) { + this.staticQueryParams.add(name, value); + return this; + } + + public TableBuilder withDefaultActionIf( + final BooleanSupplier condition, + final Supplier actionSupplier) { + + if (condition.getAsBoolean()) { + return withDefaultAction(actionSupplier.get()); + } + + return this; + } + + public TableBuilder withDefaultAction(final PageAction action) { + this.defaultActionFunction = table -> PageAction.copyOf(action); + return this; + } + + public TableBuilder withDefaultActionIf( + final BooleanSupplier condition, + final Function, PageAction> defaultActionFunction) { + + if (condition.getAsBoolean()) { + return withDefaultAction(defaultActionFunction); + } + + return this; + } + + public TableBuilder withDefaultAction(final Function, PageAction> defaultActionFunction) { + this.defaultActionFunction = defaultActionFunction; + return this; + } + + public TableBuilder withRowDecorator(final BiConsumer rowDecorator) { + this.rowDecorator = rowDecorator; + return this; + } + + public EntityTable compose(final PageContext pageContext) { + return new EntityTable<>( + this.name, + this.markupEnabled, + this.type, + pageContext, + this.restCall, + this.restCallAdapter, + this.pageService, + this.columns, + this.pageSize, + this.emptyMessage, + this.defaultActionFunction, + this.hideNavigation, + this.staticQueryParams, + this.rowDecorator, + this.selectionListener); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableFilter.java b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableFilter.java index bde5d4ac..00713f65 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableFilter.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableFilter.java @@ -49,7 +49,7 @@ public class TableFilter { private static final LocTextKey DATE_TO_TEXT = new LocTextKey("sebserver.overall.date.to"); private static final LocTextKey ALL_TEXT = new LocTextKey("sebserver.overall.status.all"); - public static enum CriteriaType { + public enum CriteriaType { TEXT, SINGLE_SELECTION, DATE, @@ -82,7 +82,7 @@ public class TableFilter { public MultiValueMap getFilterParameter() { return this.components .stream() - .reduce(new LinkedMultiValueMap(), + .reduce(new LinkedMultiValueMap<>(), (map, comp) -> comp.putFilterParameter(map), (map1, map2) -> { map1.putAll(map2); @@ -92,8 +92,7 @@ public class TableFilter { public void reset() { this.components - .stream() - .forEach(comp -> comp.reset()); + .forEach(FilterComponent::reset); } private void buildComponents() { @@ -158,7 +157,7 @@ public class TableFilter { .append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR) .append(filter.getValue()) .append(Constants.LIST_SEPARATOR), - (sb1, sb2) -> sb1.append(sb2)); + StringBuilder::append); if (builder.length() > 0) { builder.deleteCharAt(builder.length() - 1); } @@ -171,22 +170,19 @@ public class TableFilter { } try { - Arrays.asList(StringUtils.split( + Arrays.stream(StringUtils.split( attribute, Constants.LIST_SEPARATOR_CHAR)) - .stream() .map(nameValue -> StringUtils.split( nameValue, Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR)) - .forEach(nameValue -> { - this.components - .stream() - .filter(filter -> nameValue[0].equals(filter.attribute.columnName)) - .findFirst() - .ifPresent(filter -> filter.setValue((nameValue.length > 1) - ? nameValue[1] - : StringUtils.EMPTY)); - }); + .forEach(nameValue -> this.components + .stream() + .filter(filter -> nameValue[0].equals(filter.attribute.columnName)) + .findFirst() + .ifPresent(filter -> filter.setValue((nameValue.length > 1) + ? nameValue[1] + : StringUtils.EMPTY))); } catch (final Exception e) { log.error("Failed to set filter attributes: ", e); } @@ -208,9 +204,7 @@ public class TableFilter { ImageIcon.SEARCH, inner, new LocTextKey("sebserver.overall.action.filter"), - event -> { - this.entityTable.applyFilter(); - }); + event -> this.entityTable.applyFilter()); imageButton.setLayoutData(gridData); final Label imageButton2 = this.entityTable.widgetFactory.imageButton( ImageIcon.CANCEL, @@ -276,7 +270,7 @@ public class TableFilter { } } - private class NullFilter extends FilterComponent { + private static class NullFilter extends FilterComponent { private Label label; @@ -493,7 +487,6 @@ public class TableFilter { private class DateRange extends FilterComponent { private Composite innerComposite; - //private final GridData rw1 = new GridData(SWT.FILL, SWT.FILL, true, true); private DateTime fromDateSelector; private DateTime toDateSelector; private DateTime fromTimeSelector; diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableNavigator.java b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableNavigator.java index fa01c40a..f5f5fa1e 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/table/TableNavigator.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/table/TableNavigator.java @@ -1,189 +1,179 @@ -/* - * Copyright (c) 2019 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.table; - -import org.eclipse.rap.rwt.RWT; -import org.eclipse.swt.SWT; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Label; - -import ch.ethz.seb.sebserver.gbl.model.Page; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; - -public class TableNavigator { - - private final static int PAGE_NAV_SIZE = 9; - - private final Composite composite; - private final EntityTable entityTable; - - TableNavigator(final EntityTable entityTable) { - this.composite = new Composite(entityTable.composite, SWT.NONE); - final GridData gridData = new GridData(SWT.LEFT, SWT.CENTER, true, true); - this.composite.setLayoutData(gridData); - final GridLayout layout = new GridLayout(3, false); - this.composite.setLayout(layout); - - this.entityTable = entityTable; - } - - public Page update(final Page pageData) { - // clear all - PageService.clearComposite(this.composite); - - if (pageData.isEmpty()) { - // show empty message - if (this.entityTable.emptyMessage != null) { - this.entityTable.widgetFactory.labelLocalized( - this.composite, - CustomVariant.TEXT_H3, - this.entityTable.emptyMessage); - } - return pageData; - } - - if (this.entityTable.hideNavigation) { - return pageData; - } - - final int pageNumber = pageData.getPageNumber(); - final int numberOfPages = pageData.getNumberOfPages(); - - createPagingHeader(pageNumber, numberOfPages); - - final Composite numNav = new Composite(this.composite, SWT.NONE); - final GridData gridData = new GridData(SWT.CENTER, SWT.TOP, true, false); - numNav.setLayoutData(gridData); - final GridLayout rowLayout = new GridLayout(PAGE_NAV_SIZE + 5, true); - numNav.setLayout(rowLayout); - - if (numberOfPages > 1) { - createBackwardLabel(pageNumber > 1, pageNumber, numNav); - final int pageNavSize = (numberOfPages > PAGE_NAV_SIZE) ? PAGE_NAV_SIZE : numberOfPages; - final int half = pageNavSize / 2; - int start = pageNumber - half; - if (start < 1) { - start = 1; - } - int end = start + pageNavSize; - if (end > numberOfPages) { - end = numberOfPages + 1; - start = end - pageNavSize; - } - - for (int i = start; i < end; i++) { - createPageNumberLabel(i, i != pageNumber, numNav); - } - - createForwardLabel(pageNumber < numberOfPages, pageNumber, numberOfPages, numNav); - } - - return pageData; - } - - private void createPagingHeader(final int page, final int of) { - final Label pageHeader = new Label(this.composite, SWT.NONE); - final GridData gridData = new GridData(SWT.CENTER, SWT.CENTER, true, false); - gridData.widthHint = 100; - gridData.minimumWidth = 100; - gridData.heightHint = 16; - pageHeader.setLayoutData(gridData); - pageHeader.setText("Page " + page + " / " + of); - } - - private void createPageNumberLabel( - final int page, - final boolean selectable, - final Composite parent) { - - final GridData rowData = new GridData(22, 16); - final Label pageLabel = new Label(parent, SWT.NONE); - pageLabel.setText(" " + String.valueOf(page) + " "); - pageLabel.setLayoutData(rowData); - pageLabel.setAlignment(SWT.CENTER); - if (selectable) { - pageLabel.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); - pageLabel.addListener(SWT.MouseDown, event -> { - this.entityTable.selectPage(page); - }); - } - } - - private void createForwardLabel( - final boolean visible, - final int pageNumber, - final int numberOfPages, - final Composite parent) { - - final GridData rowData = new GridData(22, 16); - final Label forward = new Label(parent, SWT.NONE); - forward.setText(">"); - forward.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); - forward.setLayoutData(rowData); - forward.setAlignment(SWT.CENTER); - if (visible) { - forward.addListener(SWT.MouseDown, event -> { - this.entityTable.selectPage(pageNumber + 1); - }); - } else { - forward.setVisible(false); - } - - final Label end = new Label(parent, SWT.NONE); - end.setText(">>"); - end.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); - end.setLayoutData(rowData); - end.setAlignment(SWT.CENTER); - if (visible) { - end.addListener(SWT.MouseDown, event -> { - this.entityTable.selectPage(numberOfPages); - }); - } else { - end.setVisible(false); - } - } - - private void createBackwardLabel( - final boolean visible, - final int pageNumber, - final Composite parent) { - - final GridData rowData = new GridData(22, 16); - final Label start = new Label(parent, SWT.NONE); - start.setText("<<"); - start.setLayoutData(rowData); - start.setAlignment(SWT.CENTER); - start.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); - if (visible) { - start.addListener(SWT.MouseDown, event -> { - this.entityTable.selectPage(1); - }); - } else { - start.setVisible(false); - } - - final Label backward = new Label(parent, SWT.NONE); - backward.setText("<"); - backward.setLayoutData(rowData); - backward.setAlignment(SWT.CENTER); - backward.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); - if (visible) { - backward.addListener(SWT.MouseDown, event -> { - this.entityTable.selectPage(pageNumber - 1); - }); - } else { - backward.setVisible(false); - } - - } - -} +/* + * Copyright (c) 2019 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.table; + +import org.eclipse.rap.rwt.RWT; +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Label; + +import ch.ethz.seb.sebserver.gbl.model.Page; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; + +public class TableNavigator { + + private final static int PAGE_NAV_SIZE = 9; + + private final Composite composite; + private final EntityTable entityTable; + + TableNavigator(final EntityTable entityTable) { + this.composite = new Composite(entityTable.composite, SWT.NONE); + final GridData gridData = new GridData(SWT.LEFT, SWT.CENTER, true, true); + this.composite.setLayoutData(gridData); + final GridLayout layout = new GridLayout(3, false); + this.composite.setLayout(layout); + + this.entityTable = entityTable; + } + + public Page update(final Page pageData) { + // clear all + PageService.clearComposite(this.composite); + + if (pageData.isEmpty()) { + // show empty message + if (this.entityTable.emptyMessage != null) { + this.entityTable.widgetFactory.labelLocalized( + this.composite, + CustomVariant.TEXT_H3, + this.entityTable.emptyMessage); + } + return pageData; + } + + if (this.entityTable.hideNavigation) { + return pageData; + } + + final int pageNumber = pageData.getPageNumber(); + final int numberOfPages = pageData.getNumberOfPages(); + + createPagingHeader(pageNumber, numberOfPages); + + final Composite numNav = new Composite(this.composite, SWT.NONE); + final GridData gridData = new GridData(SWT.CENTER, SWT.TOP, true, false); + numNav.setLayoutData(gridData); + final GridLayout rowLayout = new GridLayout(PAGE_NAV_SIZE + 5, true); + numNav.setLayout(rowLayout); + + if (numberOfPages > 1) { + createBackwardLabel(pageNumber > 1, pageNumber, numNav); + final int pageNavSize = Math.min(numberOfPages, PAGE_NAV_SIZE); + final int half = pageNavSize / 2; + int start = pageNumber - half; + if (start < 1) { + start = 1; + } + int end = start + pageNavSize; + if (end > numberOfPages) { + end = numberOfPages + 1; + start = end - pageNavSize; + } + + for (int i = start; i < end; i++) { + createPageNumberLabel(i, i != pageNumber, numNav); + } + + createForwardLabel(pageNumber < numberOfPages, pageNumber, numberOfPages, numNav); + } + + return pageData; + } + + private void createPagingHeader(final int page, final int of) { + final Label pageHeader = new Label(this.composite, SWT.NONE); + final GridData gridData = new GridData(SWT.CENTER, SWT.CENTER, true, false); + gridData.widthHint = 100; + gridData.minimumWidth = 100; + gridData.heightHint = 16; + pageHeader.setLayoutData(gridData); + pageHeader.setText("Page " + page + " / " + of); + } + + private void createPageNumberLabel( + final int page, + final boolean selectable, + final Composite parent) { + + final GridData rowData = new GridData(22, 16); + final Label pageLabel = new Label(parent, SWT.NONE); + pageLabel.setText(" " + page + " "); + pageLabel.setLayoutData(rowData); + pageLabel.setAlignment(SWT.CENTER); + if (selectable) { + pageLabel.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); + pageLabel.addListener(SWT.MouseDown, event -> this.entityTable.selectPage(page)); + } + } + + private void createForwardLabel( + final boolean visible, + final int pageNumber, + final int numberOfPages, + final Composite parent) { + + final GridData rowData = new GridData(22, 16); + final Label forward = new Label(parent, SWT.NONE); + forward.setText(">"); + forward.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); + forward.setLayoutData(rowData); + forward.setAlignment(SWT.CENTER); + if (visible) { + forward.addListener(SWT.MouseDown, event -> this.entityTable.selectPage(pageNumber + 1)); + } else { + forward.setVisible(false); + } + + final Label end = new Label(parent, SWT.NONE); + end.setText(">>"); + end.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); + end.setLayoutData(rowData); + end.setAlignment(SWT.CENTER); + if (visible) { + end.addListener(SWT.MouseDown, event -> this.entityTable.selectPage(numberOfPages)); + } else { + end.setVisible(false); + } + } + + private void createBackwardLabel( + final boolean visible, + final int pageNumber, + final Composite parent) { + + final GridData rowData = new GridData(22, 16); + final Label start = new Label(parent, SWT.NONE); + start.setText("<<"); + start.setLayoutData(rowData); + start.setAlignment(SWT.CENTER); + start.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); + if (visible) { + start.addListener(SWT.MouseDown, event -> this.entityTable.selectPage(1)); + } else { + start.setVisible(false); + } + + final Label backward = new Label(parent, SWT.NONE); + backward.setText("<"); + backward.setLayoutData(rowData); + backward.setAlignment(SWT.CENTER); + backward.setData(RWT.CUSTOM_VARIANT, CustomVariant.LIST_NAVIGATION.key); + if (visible) { + backward.addListener(SWT.MouseDown, event -> this.entityTable.selectPage(pageNumber - 1)); + } else { + backward.setVisible(false); + } + + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/FileUploadSelection.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/FileUploadSelection.java index 0784f730..1fe1e571 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/FileUploadSelection.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/FileUploadSelection.java @@ -1,204 +1,202 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.io.IOException; -import java.io.InputStream; -import java.io.PipedInputStream; -import java.io.PipedOutputStream; -import java.util.ArrayList; -import java.util.List; -import java.util.Locale; -import java.util.function.Consumer; - -import org.apache.commons.io.IOUtils; -import org.eclipse.rap.fileupload.FileDetails; -import org.eclipse.rap.fileupload.FileUploadHandler; -import org.eclipse.rap.fileupload.FileUploadReceiver; -import org.eclipse.rap.rwt.widgets.FileUpload; -import org.eclipse.swt.SWT; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Label; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; - -public class FileUploadSelection extends Composite { - - private static final Logger log = LoggerFactory.getLogger(FileUploadSelection.class); - - private static final long serialVersionUID = 5800153475027387363L; - - private static final LocTextKey PLEASE_SELECT_TEXT = - new LocTextKey("sebserver.overall.upload"); - - private final I18nSupport i18nSupport; - private final List supportedFileExtensions = new ArrayList<>(); - - private final boolean readonly; - private final FileUpload fileUpload; - private final Label fileName; - - private Consumer errorHandler; - private InputStream inputStream; - private final FileUploadHandler uploadHandler; - private final InputReceiver inputReceiver; - - public FileUploadSelection( - final Composite parent, - final I18nSupport i18nSupport, - final boolean readonly) { - - super(parent, SWT.NONE); - final GridLayout gridLayout = new GridLayout(2, false); - gridLayout.horizontalSpacing = 0; - gridLayout.marginHeight = 0; - gridLayout.marginWidth = 0; - gridLayout.verticalSpacing = 0; - super.setLayout(gridLayout); - - this.i18nSupport = i18nSupport; - this.readonly = readonly; - - if (readonly) { - this.fileName = new Label(this, SWT.NONE); - this.fileName.setText(i18nSupport.getText(PLEASE_SELECT_TEXT)); - this.fileName.setLayoutData(new GridData()); - this.fileUpload = null; - this.uploadHandler = null; - this.inputReceiver = null; - } else { - this.fileUpload = new FileUpload(this, SWT.NONE); - this.fileUpload.setImage(WidgetFactory.ImageIcon.IMPORT.getImage(parent.getDisplay())); - this.fileUpload.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, false, false)); - this.fileUpload.setToolTipText(Utils.formatLineBreaks(this.i18nSupport.getText(PLEASE_SELECT_TEXT))); - this.inputReceiver = new InputReceiver(); - this.uploadHandler = new FileUploadHandler(this.inputReceiver); - - this.fileName = new Label(this, SWT.NONE); - this.fileName.setText(i18nSupport.getText(PLEASE_SELECT_TEXT)); - this.fileName.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false)); - - this.fileUpload.addListener(SWT.Selection, event -> { - final String fileName = FileUploadSelection.this.fileUpload.getFileName(); - if (fileName == null || !fileSupported(fileName)) { - if (FileUploadSelection.this.errorHandler != null) { - final String text = i18nSupport.getText(new LocTextKey( - "sebserver.overall.upload.unsupported.file", - this.supportedFileExtensions.toString()), - "Unsupported image file type selected"); - FileUploadSelection.this.errorHandler.accept(text); - } - return; - } - FileUploadSelection.this.fileUpload.submit(this.uploadHandler.getUploadUrl()); - FileUploadSelection.this.fileName.setText(fileName); - FileUploadSelection.this.errorHandler.accept(null); - }); - - } - } - - public void close() { - if (this.inputReceiver != null) { - this.inputReceiver.close(); - } - } - - @Override - public void dispose() { - if (this.uploadHandler != null) { - this.uploadHandler.dispose(); - } - super.dispose(); - } - - public String getFileName() { - if (this.fileName != null) { - return this.fileName.getText(); - } - - return Constants.EMPTY_NOTE; - } - - public void setFileName(final String fileName) { - if (this.fileName != null && fileName != null) { - this.fileName.setText(fileName); - } - } - - public InputStream getInputStream() { - return this.inputStream; - } - - @Override - public void update() { - if (this.inputStream != null) { - this.fileName.setText(this.i18nSupport.getText(PLEASE_SELECT_TEXT)); - } - if (!this.readonly) { - this.fileUpload.setToolTipText(Utils.formatLineBreaks(this.i18nSupport.getText(PLEASE_SELECT_TEXT))); - } - } - - public FileUploadSelection setErrorHandler(final Consumer errorHandler) { - this.errorHandler = errorHandler; - return this; - } - - public FileUploadSelection withSupportFor(final String fileExtension) { - this.supportedFileExtensions.add(fileExtension); - return this; - } - - private boolean fileSupported(final String fileName) { - return this.supportedFileExtensions - .stream() - .filter(fileType -> fileName.toUpperCase(Locale.ROOT) - .endsWith(fileType.toUpperCase(Locale.ROOT))) - .findFirst() - .isPresent(); - } - - private final class InputReceiver extends FileUploadReceiver { - private PipedInputStream pIn = null; - private PipedOutputStream pOut = null; - - @Override - public void receive(final InputStream stream, final FileDetails details) throws IOException { - if (this.pIn != null || this.pOut != null) { - throw new IllegalStateException("InputReceiver already in use"); - } - - this.pIn = new PipedInputStream(); - this.pOut = new PipedOutputStream(this.pIn); - - FileUploadSelection.this.inputStream = this.pIn; - - try { - IOUtils.copyLarge(stream, this.pOut); - } catch (final Exception e) { - log.warn("IO error: {}", e.getMessage()); - } finally { - close(); - } - } - - void close() { - IOUtils.closeQuietly(this.pOut); - } - } - -} +/* + * Copyright (c) 2019 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.widget; + +import java.io.IOException; +import java.io.InputStream; +import java.io.PipedInputStream; +import java.io.PipedOutputStream; +import java.util.ArrayList; +import java.util.List; +import java.util.Locale; +import java.util.function.Consumer; + +import org.apache.commons.io.IOUtils; +import org.eclipse.rap.fileupload.FileDetails; +import org.eclipse.rap.fileupload.FileUploadHandler; +import org.eclipse.rap.fileupload.FileUploadReceiver; +import org.eclipse.rap.rwt.widgets.FileUpload; +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Label; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; + +public class FileUploadSelection extends Composite { + + private static final Logger log = LoggerFactory.getLogger(FileUploadSelection.class); + + private static final long serialVersionUID = 5800153475027387363L; + + private static final LocTextKey PLEASE_SELECT_TEXT = + new LocTextKey("sebserver.overall.upload"); + + private final I18nSupport i18nSupport; + private final List supportedFileExtensions = new ArrayList<>(); + + private final boolean readonly; + private final FileUpload fileUpload; + private final Label fileName; + + private Consumer errorHandler; + private InputStream inputStream; + private final FileUploadHandler uploadHandler; + private final InputReceiver inputReceiver; + + public FileUploadSelection( + final Composite parent, + final I18nSupport i18nSupport, + final boolean readonly) { + + super(parent, SWT.NONE); + final GridLayout gridLayout = new GridLayout(2, false); + gridLayout.horizontalSpacing = 0; + gridLayout.marginHeight = 0; + gridLayout.marginWidth = 0; + gridLayout.verticalSpacing = 0; + super.setLayout(gridLayout); + + this.i18nSupport = i18nSupport; + this.readonly = readonly; + + if (readonly) { + this.fileName = new Label(this, SWT.NONE); + this.fileName.setText(i18nSupport.getText(PLEASE_SELECT_TEXT)); + this.fileName.setLayoutData(new GridData()); + this.fileUpload = null; + this.uploadHandler = null; + this.inputReceiver = null; + } else { + this.fileUpload = new FileUpload(this, SWT.NONE); + this.fileUpload.setImage(WidgetFactory.ImageIcon.IMPORT.getImage(parent.getDisplay())); + this.fileUpload.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, false, false)); + this.fileUpload.setToolTipText(Utils.formatLineBreaks(this.i18nSupport.getText(PLEASE_SELECT_TEXT))); + this.inputReceiver = new InputReceiver(); + this.uploadHandler = new FileUploadHandler(this.inputReceiver); + + this.fileName = new Label(this, SWT.NONE); + this.fileName.setText(i18nSupport.getText(PLEASE_SELECT_TEXT)); + this.fileName.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false)); + + this.fileUpload.addListener(SWT.Selection, event -> { + final String fileName = FileUploadSelection.this.fileUpload.getFileName(); + if (fileName == null || !fileSupported(fileName)) { + if (FileUploadSelection.this.errorHandler != null) { + final String text = i18nSupport.getText(new LocTextKey( + "sebserver.overall.upload.unsupported.file", + this.supportedFileExtensions.toString()), + "Unsupported image file type selected"); + FileUploadSelection.this.errorHandler.accept(text); + } + return; + } + FileUploadSelection.this.fileUpload.submit(this.uploadHandler.getUploadUrl()); + FileUploadSelection.this.fileName.setText(fileName); + FileUploadSelection.this.errorHandler.accept(null); + }); + + } + } + + public void close() { + if (this.inputReceiver != null) { + this.inputReceiver.close(); + } + } + + @Override + public void dispose() { + if (this.uploadHandler != null) { + this.uploadHandler.dispose(); + } + super.dispose(); + } + + public String getFileName() { + if (this.fileName != null) { + return this.fileName.getText(); + } + + return Constants.EMPTY_NOTE; + } + + public void setFileName(final String fileName) { + if (this.fileName != null && fileName != null) { + this.fileName.setText(fileName); + } + } + + public InputStream getInputStream() { + return this.inputStream; + } + + @Override + public void update() { + if (this.inputStream != null) { + this.fileName.setText(this.i18nSupport.getText(PLEASE_SELECT_TEXT)); + } + if (!this.readonly) { + this.fileUpload.setToolTipText(Utils.formatLineBreaks(this.i18nSupport.getText(PLEASE_SELECT_TEXT))); + } + } + + public FileUploadSelection setErrorHandler(final Consumer errorHandler) { + this.errorHandler = errorHandler; + return this; + } + + public FileUploadSelection withSupportFor(final String fileExtension) { + this.supportedFileExtensions.add(fileExtension); + return this; + } + + private boolean fileSupported(final String fileName) { + return this.supportedFileExtensions + .stream() + .anyMatch(fileType -> fileName.toUpperCase(Locale.ROOT) + .endsWith(fileType.toUpperCase(Locale.ROOT))); + } + + private final class InputReceiver extends FileUploadReceiver { + private PipedInputStream pIn = null; + private PipedOutputStream pOut = null; + + @Override + public void receive(final InputStream stream, final FileDetails details) throws IOException { + if (this.pIn != null || this.pOut != null) { + throw new IllegalStateException("InputReceiver already in use"); + } + + this.pIn = new PipedInputStream(); + this.pOut = new PipedOutputStream(this.pIn); + + FileUploadSelection.this.inputStream = this.pIn; + + try { + IOUtils.copyLarge(stream, this.pOut); + } catch (final Exception e) { + log.warn("IO error: {}", e.getMessage()); + } finally { + close(); + } + } + + void close() { + IOUtils.closeQuietly(this.pOut); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/GridTable.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/GridTable.java index 3b6a182f..62df87f8 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/GridTable.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/GridTable.java @@ -1,422 +1,421 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.util.ArrayList; -import java.util.EnumSet; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.stream.Collectors; - -import org.apache.commons.lang3.BooleanUtils; -import org.apache.commons.lang3.StringUtils; -import org.eclipse.rap.rwt.RWT; -import org.eclipse.swt.SWT; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Button; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Event; -import org.eclipse.swt.widgets.Label; -import org.eclipse.swt.widgets.Listener; -import org.eclipse.swt.widgets.Text; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.model.sebconfig.AttributeType; -import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; -import ch.ethz.seb.sebserver.gui.service.page.PageService; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; -import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.ImageIcon; - -public class GridTable extends Composite { - - private static final long serialVersionUID = 8515260041931976458L; - private static final Logger log = LoggerFactory.getLogger(GridTable.class); - - public static final Set SUPPORTED_TYPES = EnumSet.of( - AttributeType.CHECKBOX, - AttributeType.TEXT_FIELD); - - private static final int ACTION_COLUMN_WIDTH = 20; - - private final WidgetFactory widgetFactory; - private final List columns; - private final Label addAction; - private final List rows; - private final String locTextKeyPrefix; - private Listener listener; - - public GridTable( - final Composite parent, - final List columnDefs, - final String locTextKeyPrefix, - final WidgetFactory widgetFactory) { - - super(parent, SWT.NONE); - - this.widgetFactory = widgetFactory; - this.locTextKeyPrefix = locTextKeyPrefix; - final GridLayout gridLayout = new GridLayout(columnDefs.size() + 1, false); - gridLayout.verticalSpacing = 1; - gridLayout.marginLeft = 0; - gridLayout.marginHeight = 5; - gridLayout.marginWidth = 0; - gridLayout.horizontalSpacing = 0; - this.setLayout(gridLayout); - - this.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); - - this.columns = new ArrayList<>(); - for (final ColumnDef columnDef : columnDefs) { - final Label label = widgetFactory.labelLocalized( - this, - new LocTextKey(locTextKeyPrefix + columnDef.name)); - final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); - label.setLayoutData(gridData); - this.columns.add(new Column(columnDef, gridData)); - } - - this.addAction = widgetFactory.imageButton( - ImageIcon.ADD_BOX, - this, - new LocTextKey(locTextKeyPrefix + "addAction"), - this::addRow); - final GridData gridData = new GridData(SWT.CENTER, SWT.FILL, true, true); - gridData.widthHint = ACTION_COLUMN_WIDTH; - this.addAction.setLayoutData(gridData); - - this.rows = new ArrayList<>(); - this.addListener(SWT.Resize, this::adaptColumnWidth); - - } - - public void setListener(final Listener listener) { - this.listener = listener; - } - - void addRow(final Event event) { - final List row = new ArrayList<>(); - for (final Column column : this.columns) { - row.add(createCell(column, column.columnDef.defaultValue)); - } - this.rows.add(new Row(row)); - - this.adaptLayout(); - - if (this.listener != null) { - this.listener.handleEvent(new Event()); - } - } - - public void adaptLayout() { - this.getParent().getParent().layout(true, true); - PageService.updateScrolledComposite(this); - } - - void addRow(final String values) { - if (StringUtils.isBlank(values)) { - return; - } - - final Map nameValueMap = new HashMap<>(); - for (final String valueMap : StringUtils.split(values, Constants.EMBEDDED_LIST_SEPARATOR)) { - final String[] nameValue = StringUtils.split(valueMap, Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR); - if (nameValue.length > 1) { - nameValueMap.put(nameValue[0], nameValue[1]); - } else { - nameValueMap.put(nameValue[0], null); - } - } - - final List row = new ArrayList<>(); - for (final Column column : this.columns) { - row.add(createCell(column, nameValueMap.get(column.columnDef.name))); - } - this.rows.add(new Row(row)); - } - - void deleteRow(final Row row) { - if (this.rows.remove(row)) { - row.dispose(); - } - - this.adaptLayout(); - } - - public String getValue() { - return StringUtils.join( - this.rows - .stream() - .map(row -> row.getValue()) - .collect(Collectors.toList()), - Constants.LIST_SEPARATOR); - } - - public void setValue(final String value) { - clearTable(); - - if (StringUtils.isBlank(value)) { - return; - } - - for (final String val : StringUtils.split(value, Constants.LIST_SEPARATOR)) { - addRow(val); - } - - this.adaptLayout(); - } - - private ControlAdapter createCell(final Column column, final String value) { - switch (column.columnDef.type) { - case CHECKBOX: { - final CheckBox checkBox = new CheckBox(this, column.columnDef, this.listener); - checkBox.setValue(value); - return checkBox; - } - case TEXT_FIELD: { - final TextField textField = new TextField(this, column.columnDef, this.listener); - textField.setValue(value); - return textField; - } - default: { - return new Dummy(this, column.columnDef); - } - } - } - - private void clearTable() { - for (final Row row : this.rows) { - row.dispose(); - } - this.rows.clear(); - } - - private void adaptColumnWidth(final Event event) { - try { - - final int currentTableWidth = super.getClientArea().width - ACTION_COLUMN_WIDTH; - final int widthUnit = currentTableWidth / this.columns - .stream() - .reduce(0, - (i, c2) -> i + c2.columnDef.widthFactor, - (i1, i2) -> i1 + i2); - - this.columns - .stream() - .forEach(c -> c.header.widthHint = c.columnDef.widthFactor * widthUnit); - - super.layout(true, true); - } catch (final Exception e) { - log.warn("Failed to adaptColumnWidth: ", e); - } - } - - final class Row { - final List cells; - final Label removeAction; - - protected Row(final List cells) { - this.cells = cells; - this.removeAction = GridTable.this.widgetFactory.imageButton( - ImageIcon.REMOVE_BOX, - GridTable.this, - new LocTextKey(GridTable.this.locTextKeyPrefix + "removeAction"), - event -> deleteRow(this)); - final GridData gridData = new GridData(SWT.CENTER, SWT.CENTER, true, true); - gridData.widthHint = ACTION_COLUMN_WIDTH; - this.removeAction.setLayoutData(gridData); - } - - void dispose() { - for (final ControlAdapter cell : this.cells) { - cell.dispose(); - } - this.removeAction.dispose(); - } - - String getValue() { - return StringUtils.join( - this.cells - .stream() - .map(cell -> cell.columnDef().name + Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR - + cell.getValue()) - .collect(Collectors.toList()), - Constants.EMBEDDED_LIST_SEPARATOR); - } - } - - public static final class ColumnDef { - final int widthFactor; - final String name; - final AttributeType type; - final String defaultValue; - - protected ColumnDef( - final int widthFactor, - final String name, - final AttributeType type, - final String defaultValue) { - - this.widthFactor = widthFactor; - this.name = name; - this.type = type; - this.defaultValue = defaultValue; - } - - public static final ColumnDef fromString( - final String string, - final Map defaultValueMap) { - - if (StringUtils.isBlank(string)) { - return null; - } - - final String[] split = StringUtils.split(string, Constants.COMPLEX_VALUE_SEPARATOR); - final AttributeType attributeType = AttributeType.valueOf(split[2]); - if (!SUPPORTED_TYPES.contains(attributeType)) { - throw new UnsupportedOperationException( - "The AttributeType : " + attributeType + " is not supported yet"); - } - - return new ColumnDef( - Integer.parseInt(split[0]), - split[1], - attributeType, - defaultValueMap.get(split[1])); - } - } - - private static class Column { - final ColumnDef columnDef; - final GridData header; - - protected Column(final ColumnDef columnDef, final GridData header) { - this.columnDef = columnDef; - this.header = header; - } - } - - interface ControlAdapter { - String getValue(); - - void setValue(String value); - - void dispose(); - - ColumnDef columnDef(); - } - - private static class Dummy implements ControlAdapter { - - private final Label label; - private final ColumnDef columnDef; - - Dummy(final Composite parent, final ColumnDef columnDef) { - this.label = new Label(parent, SWT.NONE); - this.label.setText("unsupported"); - this.columnDef = columnDef; - } - - @Override - public String getValue() { - return null; - } - - @Override - public void setValue(final String value) { - } - - @Override - public void dispose() { - this.label.dispose(); - } - - @Override - public ColumnDef columnDef() { - return this.columnDef; - } - } - - private static class CheckBox implements ControlAdapter { - - private final Button checkboxButton; - private final ColumnDef columnDef; - - CheckBox(final Composite parent, final ColumnDef columnDef, final Listener listener) { - this.checkboxButton = new Button(parent, SWT.CHECK); - this.checkboxButton.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, false)); - this.columnDef = columnDef; - if (listener != null) { - this.checkboxButton.addListener(SWT.Selection, listener); - } - } - - @Override - public String getValue() { - return this.checkboxButton.getSelection() - ? Constants.TRUE_STRING - : Constants.FALSE_STRING; - } - - @Override - public void setValue(final String value) { - this.checkboxButton.setSelection(BooleanUtils.toBoolean(value)); - } - - @Override - public void dispose() { - this.checkboxButton.dispose(); - } - - @Override - public ColumnDef columnDef() { - return this.columnDef; - } - } - - private static class TextField implements ControlAdapter { - - private final Text _textField; - private final ColumnDef columnDef; - - TextField(final Composite parent, final ColumnDef columnDef, final Listener listener) { - this._textField = new Text(parent, SWT.LEFT | SWT.BORDER); - this._textField.setData(RWT.CUSTOM_VARIANT, CustomVariant.CONFIG_INPUT_READONLY.key); - this._textField.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, false)); - this.columnDef = columnDef; - this._textField.addListener(SWT.FocusOut, listener); - this._textField.addListener(SWT.Traverse, listener); - } - - @Override - public String getValue() { - return this._textField.getText(); - } - - @Override - public void setValue(final String value) { - this._textField.setText((value != null) ? value : ""); - } - - @Override - public void dispose() { - this._textField.dispose(); - } - - @Override - public ColumnDef columnDef() { - return this.columnDef; - } - } - -} +/* + * Copyright (c) 2019 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.widget; + +import java.util.ArrayList; +import java.util.EnumSet; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; + +import org.apache.commons.lang3.BooleanUtils; +import org.apache.commons.lang3.StringUtils; +import org.eclipse.rap.rwt.RWT; +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Listener; +import org.eclipse.swt.widgets.Text; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.model.sebconfig.AttributeType; +import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.CustomVariant; +import ch.ethz.seb.sebserver.gui.widget.WidgetFactory.ImageIcon; + +public class GridTable extends Composite { + + private static final long serialVersionUID = 8515260041931976458L; + private static final Logger log = LoggerFactory.getLogger(GridTable.class); + + public static final Set SUPPORTED_TYPES = EnumSet.of( + AttributeType.CHECKBOX, + AttributeType.TEXT_FIELD); + + private static final int ACTION_COLUMN_WIDTH = 20; + + private final WidgetFactory widgetFactory; + private final List columns; + private final Label addAction; + private final List rows; + private final String locTextKeyPrefix; + private Listener listener; + + public GridTable( + final Composite parent, + final List columnDefs, + final String locTextKeyPrefix, + final WidgetFactory widgetFactory) { + + super(parent, SWT.NONE); + + this.widgetFactory = widgetFactory; + this.locTextKeyPrefix = locTextKeyPrefix; + final GridLayout gridLayout = new GridLayout(columnDefs.size() + 1, false); + gridLayout.verticalSpacing = 1; + gridLayout.marginLeft = 0; + gridLayout.marginHeight = 5; + gridLayout.marginWidth = 0; + gridLayout.horizontalSpacing = 0; + this.setLayout(gridLayout); + + this.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); + + this.columns = new ArrayList<>(); + for (final ColumnDef columnDef : columnDefs) { + final Label label = widgetFactory.labelLocalized( + this, + new LocTextKey(locTextKeyPrefix + columnDef.name)); + final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true); + label.setLayoutData(gridData); + this.columns.add(new Column(columnDef, gridData)); + } + + this.addAction = widgetFactory.imageButton( + ImageIcon.ADD_BOX, + this, + new LocTextKey(locTextKeyPrefix + "addAction"), + this::addRow); + final GridData gridData = new GridData(SWT.CENTER, SWT.FILL, true, true); + gridData.widthHint = ACTION_COLUMN_WIDTH; + this.addAction.setLayoutData(gridData); + + this.rows = new ArrayList<>(); + this.addListener(SWT.Resize, this::adaptColumnWidth); + + } + + public void setListener(final Listener listener) { + this.listener = listener; + } + + void addRow(final Event event) { + final List row = new ArrayList<>(); + for (final Column column : this.columns) { + row.add(createCell(column, column.columnDef.defaultValue)); + } + this.rows.add(new Row(row)); + + this.adaptLayout(); + + if (this.listener != null) { + this.listener.handleEvent(new Event()); + } + } + + public void adaptLayout() { + this.getParent().getParent().layout(true, true); + PageService.updateScrolledComposite(this); + } + + void addRow(final String values) { + if (StringUtils.isBlank(values)) { + return; + } + + final Map nameValueMap = new HashMap<>(); + for (final String valueMap : StringUtils.split(values, Constants.EMBEDDED_LIST_SEPARATOR)) { + final String[] nameValue = StringUtils.split(valueMap, Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR); + if (nameValue.length > 1) { + nameValueMap.put(nameValue[0], nameValue[1]); + } else { + nameValueMap.put(nameValue[0], null); + } + } + + final List row = new ArrayList<>(); + for (final Column column : this.columns) { + row.add(createCell(column, nameValueMap.get(column.columnDef.name))); + } + this.rows.add(new Row(row)); + } + + void deleteRow(final Row row) { + if (this.rows.remove(row)) { + row.dispose(); + } + + this.adaptLayout(); + } + + public String getValue() { + return StringUtils.join( + this.rows + .stream() + .map(Row::getValue) + .collect(Collectors.toList()), + Constants.LIST_SEPARATOR); + } + + public void setValue(final String value) { + clearTable(); + + if (StringUtils.isBlank(value)) { + return; + } + + for (final String val : StringUtils.split(value, Constants.LIST_SEPARATOR)) { + addRow(val); + } + + this.adaptLayout(); + } + + private ControlAdapter createCell(final Column column, final String value) { + switch (column.columnDef.type) { + case CHECKBOX: { + final CheckBox checkBox = new CheckBox(this, column.columnDef, this.listener); + checkBox.setValue(value); + return checkBox; + } + case TEXT_FIELD: { + final TextField textField = new TextField(this, column.columnDef, this.listener); + textField.setValue(value); + return textField; + } + default: { + return new Dummy(this, column.columnDef); + } + } + } + + private void clearTable() { + for (final Row row : this.rows) { + row.dispose(); + } + this.rows.clear(); + } + + private void adaptColumnWidth(final Event event) { + try { + + final int currentTableWidth = super.getClientArea().width - ACTION_COLUMN_WIDTH; + final int widthUnit = currentTableWidth / this.columns + .stream() + .reduce(0, + (i, c2) -> i + c2.columnDef.widthFactor, + Integer::sum); + + this.columns + .forEach(c -> c.header.widthHint = c.columnDef.widthFactor * widthUnit); + + super.layout(true, true); + } catch (final Exception e) { + log.warn("Failed to adaptColumnWidth: ", e); + } + } + + final class Row { + final List cells; + final Label removeAction; + + protected Row(final List cells) { + this.cells = cells; + this.removeAction = GridTable.this.widgetFactory.imageButton( + ImageIcon.REMOVE_BOX, + GridTable.this, + new LocTextKey(GridTable.this.locTextKeyPrefix + "removeAction"), + event -> deleteRow(this)); + final GridData gridData = new GridData(SWT.CENTER, SWT.CENTER, true, true); + gridData.widthHint = ACTION_COLUMN_WIDTH; + this.removeAction.setLayoutData(gridData); + } + + void dispose() { + for (final ControlAdapter cell : this.cells) { + cell.dispose(); + } + this.removeAction.dispose(); + } + + String getValue() { + return StringUtils.join( + this.cells + .stream() + .map(cell -> cell.columnDef().name + Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR + + cell.getValue()) + .collect(Collectors.toList()), + Constants.EMBEDDED_LIST_SEPARATOR); + } + } + + public static final class ColumnDef { + final int widthFactor; + final String name; + final AttributeType type; + final String defaultValue; + + protected ColumnDef( + final int widthFactor, + final String name, + final AttributeType type, + final String defaultValue) { + + this.widthFactor = widthFactor; + this.name = name; + this.type = type; + this.defaultValue = defaultValue; + } + + public static ColumnDef fromString( + final String string, + final Map defaultValueMap) { + + if (StringUtils.isBlank(string)) { + return null; + } + + final String[] split = StringUtils.split(string, Constants.COMPLEX_VALUE_SEPARATOR); + final AttributeType attributeType = AttributeType.valueOf(split[2]); + if (!SUPPORTED_TYPES.contains(attributeType)) { + throw new UnsupportedOperationException( + "The AttributeType : " + attributeType + " is not supported yet"); + } + + return new ColumnDef( + Integer.parseInt(split[0]), + split[1], + attributeType, + defaultValueMap.get(split[1])); + } + } + + private static class Column { + final ColumnDef columnDef; + final GridData header; + + protected Column(final ColumnDef columnDef, final GridData header) { + this.columnDef = columnDef; + this.header = header; + } + } + + interface ControlAdapter { + String getValue(); + + void setValue(String value); + + void dispose(); + + ColumnDef columnDef(); + } + + private static class Dummy implements ControlAdapter { + + private final Label label; + private final ColumnDef columnDef; + + Dummy(final Composite parent, final ColumnDef columnDef) { + this.label = new Label(parent, SWT.NONE); + this.label.setText("unsupported"); + this.columnDef = columnDef; + } + + @Override + public String getValue() { + return null; + } + + @Override + public void setValue(final String value) { + } + + @Override + public void dispose() { + this.label.dispose(); + } + + @Override + public ColumnDef columnDef() { + return this.columnDef; + } + } + + private static class CheckBox implements ControlAdapter { + + private final Button checkboxButton; + private final ColumnDef columnDef; + + CheckBox(final Composite parent, final ColumnDef columnDef, final Listener listener) { + this.checkboxButton = new Button(parent, SWT.CHECK); + this.checkboxButton.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, false)); + this.columnDef = columnDef; + if (listener != null) { + this.checkboxButton.addListener(SWT.Selection, listener); + } + } + + @Override + public String getValue() { + return this.checkboxButton.getSelection() + ? Constants.TRUE_STRING + : Constants.FALSE_STRING; + } + + @Override + public void setValue(final String value) { + this.checkboxButton.setSelection(BooleanUtils.toBoolean(value)); + } + + @Override + public void dispose() { + this.checkboxButton.dispose(); + } + + @Override + public ColumnDef columnDef() { + return this.columnDef; + } + } + + private static class TextField implements ControlAdapter { + + private final Text _textField; + private final ColumnDef columnDef; + + TextField(final Composite parent, final ColumnDef columnDef, final Listener listener) { + this._textField = new Text(parent, SWT.LEFT | SWT.BORDER); + this._textField.setData(RWT.CUSTOM_VARIANT, CustomVariant.CONFIG_INPUT_READONLY.key); + this._textField.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, false)); + this.columnDef = columnDef; + this._textField.addListener(SWT.FocusOut, listener); + this._textField.addListener(SWT.Traverse, listener); + } + + @Override + public String getValue() { + return this._textField.getText(); + } + + @Override + public void setValue(final String value) { + this._textField.setText((value != null) ? value : ""); + } + + @Override + public void dispose() { + this._textField.dispose(); + } + + @Override + public ColumnDef columnDef() { + return this.columnDef; + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/ImageUploadSelection.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/ImageUploadSelection.java index 4a40feb4..eec2cad2 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/ImageUploadSelection.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/ImageUploadSelection.java @@ -1,222 +1,213 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.io.ByteArrayInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.StandardCharsets; -import java.util.Arrays; -import java.util.Base64; -import java.util.Collections; -import java.util.HashSet; -import java.util.Locale; -import java.util.Set; -import java.util.function.Consumer; - -import org.apache.commons.codec.binary.Base64InputStream; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang3.StringUtils; -import org.eclipse.rap.fileupload.FileDetails; -import org.eclipse.rap.fileupload.FileUploadHandler; -import org.eclipse.rap.fileupload.FileUploadReceiver; -import org.eclipse.rap.rwt.RWT; -import org.eclipse.rap.rwt.widgets.FileUpload; -import org.eclipse.swt.SWT; -import org.eclipse.swt.graphics.Image; -import org.eclipse.swt.graphics.ImageData; -import org.eclipse.swt.graphics.Rectangle; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Composite; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; -import ch.ethz.seb.sebserver.gui.service.push.ServerPushContext; -import ch.ethz.seb.sebserver.gui.service.push.ServerPushService; - -public final class ImageUploadSelection extends Composite { - - private static final long serialVersionUID = 368264811155804533L; - private static final Logger log = LoggerFactory.getLogger(ImageUploadSelection.class); - - public static final Set SUPPORTED_IMAGE_FILES = Collections.unmodifiableSet(new HashSet<>(Arrays.asList( - ".png", - ".jpg", - ".jpeg"))); - - private final ServerPushService serverPushService; - - private final Composite imageCanvas; - private final FileUpload fileUpload; - private final int maxWidth; - private final int maxHeight; - - private Consumer errorHandler; - private String imageBase64 = null; - private boolean loadNewImage = false; - private boolean imageLoaded = false; - - ImageUploadSelection( - final Composite parent, - final ServerPushService serverPushService, - final I18nSupport i18nSupport, - final boolean readonly, - final int maxWidth, - final int maxHeight) { - - super(parent, SWT.NONE); - final GridLayout gridLayout = new GridLayout(1, false); - gridLayout.horizontalSpacing = 0; - gridLayout.marginHeight = 0; - gridLayout.marginWidth = 0; - gridLayout.marginLeft = 0; - gridLayout.verticalSpacing = 0; - super.setLayout(gridLayout); - - this.serverPushService = serverPushService; - this.maxWidth = maxWidth; - this.maxHeight = maxHeight; - - if (!readonly) { - this.fileUpload = new FileUpload(this, SWT.NONE); - this.fileUpload.setImage(WidgetFactory.ImageIcon.IMPORT.getImage(parent.getDisplay())); - final GridData gridData = new GridData(SWT.LEFT, SWT.TOP, false, false); - gridData.horizontalIndent = 0; - this.fileUpload.setLayoutData(gridData); - - final FileUploadHandler uploadHandler = new FileUploadHandler(new ImageReceiver()); - this.fileUpload.addListener(SWT.Selection, event -> { - final String fileName = ImageUploadSelection.this.fileUpload.getFileName(); - if (fileName == null || !fileSupported(fileName)) { - if (ImageUploadSelection.this.errorHandler != null) { - final String text = i18nSupport.getText( - "sebserver.institution.form.logoImage.unsupportedFileType", - "Unsupported image file type selected"); - ImageUploadSelection.this.errorHandler.accept(text); - } - - log.warn("Unsupported image file selected: {}", fileName); - - return; - } - ImageUploadSelection.this.loadNewImage = true; - ImageUploadSelection.this.imageLoaded = false; - ImageUploadSelection.this.fileUpload.submit(uploadHandler.getUploadUrl()); - - ImageUploadSelection.this.serverPushService.runServerPush( - new ServerPushContext(ImageUploadSelection.this, ImageUploadSelection::uploadInProgress), - 200, - ImageUploadSelection::update); - }); - } else { - this.fileUpload = null; - } - - this.imageCanvas = new Composite(this, SWT.NONE); - final GridData canvas = new GridData(SWT.FILL, SWT.FILL, true, true); - this.imageCanvas.setLayoutData(canvas); - } - - public void setErrorHandler(final Consumer errorHandler) { - this.errorHandler = errorHandler; - } - - public void setSelectionText(final String text) { - if (this.fileUpload != null) { - this.fileUpload.setToolTipText(Utils.formatLineBreaks(text)); - } - } - - public String getImageBase64() { - return this.imageBase64; - } - - public void setImageBase64(final String imageBase64) { - if (StringUtils.isBlank(imageBase64)) { - return; - } - - this.imageBase64 = imageBase64; - final Base64InputStream input = new Base64InputStream( - new ByteArrayInputStream(imageBase64.getBytes(StandardCharsets.UTF_8)), false); - - setImage(this, input); - } - - private static final boolean uploadInProgress(final ServerPushContext context) { - final ImageUploadSelection imageUpload = (ImageUploadSelection) context.getAnchor(); - return imageUpload.loadNewImage && !imageUpload.imageLoaded; - } - - private static final void update(final ServerPushContext context) { - final ImageUploadSelection imageUpload = (ImageUploadSelection) context.getAnchor(); - if (imageUpload.imageBase64 != null - && imageUpload.loadNewImage - && imageUpload.imageLoaded) { - - final Base64InputStream input = new Base64InputStream( - new ByteArrayInputStream( - imageUpload.imageBase64.getBytes(StandardCharsets.UTF_8)), - false); - - setImage(imageUpload, input); - context.layout(); - imageUpload.layout(); - imageUpload.loadNewImage = false; - imageUpload.errorHandler.accept(null); - } - } - - private static void setImage(final ImageUploadSelection imageUpload, final Base64InputStream input) { - imageUpload.imageCanvas.setData(RWT.CUSTOM_VARIANT, "bgLogoNoImage"); - - final Image image = new Image(imageUpload.imageCanvas.getDisplay(), input); - final Rectangle imageBounds = image.getBounds(); - final int width = (imageBounds.width > imageUpload.maxWidth) - ? imageUpload.maxWidth - : imageBounds.width; - final int height = (imageBounds.height > imageUpload.maxHeight) - ? imageUpload.maxHeight - : imageBounds.height; - final ImageData imageData = image.getImageData().scaledTo(width, height); - imageUpload.imageCanvas.setBackgroundImage(new Image(imageUpload.imageCanvas.getDisplay(), imageData)); - } - - private static boolean fileSupported(final String fileName) { - return SUPPORTED_IMAGE_FILES - .stream() - .filter(fileType -> fileName.toUpperCase(Locale.ROOT) - .endsWith(fileType.toUpperCase(Locale.ROOT))) - .findFirst() - .isPresent(); - } - - private final class ImageReceiver extends FileUploadReceiver { - @Override - public void receive(final InputStream stream, final FileDetails details) throws IOException { - - try { - final String contentType = details.getContentType(); - if (contentType != null && contentType.startsWith("image")) { - ImageUploadSelection.this.imageBase64 = Base64.getEncoder() - .encodeToString(IOUtils.toByteArray(stream)); - } - } catch (final Exception e) { - log.error("Error while trying to upload image", e); - } finally { - ImageUploadSelection.this.imageLoaded = true; - stream.close(); - } - } - } - -} +/* + * Copyright (c) 2019 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.widget; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.Base64; +import java.util.Collections; +import java.util.HashSet; +import java.util.Locale; +import java.util.Set; +import java.util.function.Consumer; + +import org.apache.commons.codec.binary.Base64InputStream; +import org.apache.commons.io.IOUtils; +import org.apache.commons.lang3.StringUtils; +import org.eclipse.rap.fileupload.FileDetails; +import org.eclipse.rap.fileupload.FileUploadHandler; +import org.eclipse.rap.fileupload.FileUploadReceiver; +import org.eclipse.rap.rwt.RWT; +import org.eclipse.rap.rwt.widgets.FileUpload; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.graphics.ImageData; +import org.eclipse.swt.graphics.Rectangle; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.i18n.I18nSupport; +import ch.ethz.seb.sebserver.gui.service.push.ServerPushContext; +import ch.ethz.seb.sebserver.gui.service.push.ServerPushService; + +public final class ImageUploadSelection extends Composite { + + private static final long serialVersionUID = 368264811155804533L; + private static final Logger log = LoggerFactory.getLogger(ImageUploadSelection.class); + + public static final Set SUPPORTED_IMAGE_FILES = Set.of(".png", ".jpg", ".jpeg"); + + private final ServerPushService serverPushService; + + private final Composite imageCanvas; + private final FileUpload fileUpload; + private final int maxWidth; + private final int maxHeight; + + private Consumer errorHandler; + private String imageBase64 = null; + private boolean loadNewImage = false; + private boolean imageLoaded = false; + + ImageUploadSelection( + final Composite parent, + final ServerPushService serverPushService, + final I18nSupport i18nSupport, + final boolean readonly, + final int maxWidth, + final int maxHeight) { + + super(parent, SWT.NONE); + final GridLayout gridLayout = new GridLayout(1, false); + gridLayout.horizontalSpacing = 0; + gridLayout.marginHeight = 0; + gridLayout.marginWidth = 0; + gridLayout.marginLeft = 0; + gridLayout.verticalSpacing = 0; + super.setLayout(gridLayout); + + this.serverPushService = serverPushService; + this.maxWidth = maxWidth; + this.maxHeight = maxHeight; + + if (!readonly) { + this.fileUpload = new FileUpload(this, SWT.NONE); + this.fileUpload.setImage(WidgetFactory.ImageIcon.IMPORT.getImage(parent.getDisplay())); + final GridData gridData = new GridData(SWT.LEFT, SWT.TOP, false, false); + gridData.horizontalIndent = 0; + this.fileUpload.setLayoutData(gridData); + + final FileUploadHandler uploadHandler = new FileUploadHandler(new ImageReceiver()); + this.fileUpload.addListener(SWT.Selection, event -> { + final String fileName = ImageUploadSelection.this.fileUpload.getFileName(); + if (fileName == null || !fileSupported(fileName)) { + if (ImageUploadSelection.this.errorHandler != null) { + final String text = i18nSupport.getText( + "sebserver.institution.form.logoImage.unsupportedFileType", + "Unsupported image file type selected"); + ImageUploadSelection.this.errorHandler.accept(text); + } + + log.warn("Unsupported image file selected: {}", fileName); + + return; + } + ImageUploadSelection.this.loadNewImage = true; + ImageUploadSelection.this.imageLoaded = false; + ImageUploadSelection.this.fileUpload.submit(uploadHandler.getUploadUrl()); + + ImageUploadSelection.this.serverPushService.runServerPush( + new ServerPushContext(ImageUploadSelection.this, ImageUploadSelection::uploadInProgress), + 200, + ImageUploadSelection::update); + }); + } else { + this.fileUpload = null; + } + + this.imageCanvas = new Composite(this, SWT.NONE); + final GridData canvas = new GridData(SWT.FILL, SWT.FILL, true, true); + this.imageCanvas.setLayoutData(canvas); + } + + public void setErrorHandler(final Consumer errorHandler) { + this.errorHandler = errorHandler; + } + + public void setSelectionText(final String text) { + if (this.fileUpload != null) { + this.fileUpload.setToolTipText(Utils.formatLineBreaks(text)); + } + } + + public String getImageBase64() { + return this.imageBase64; + } + + public void setImageBase64(final String imageBase64) { + if (StringUtils.isBlank(imageBase64)) { + return; + } + + this.imageBase64 = imageBase64; + final Base64InputStream input = new Base64InputStream( + new ByteArrayInputStream(imageBase64.getBytes(StandardCharsets.UTF_8)), false); + + setImage(this, input); + } + + private static boolean uploadInProgress(final ServerPushContext context) { + final ImageUploadSelection imageUpload = (ImageUploadSelection) context.getAnchor(); + return imageUpload.loadNewImage && !imageUpload.imageLoaded; + } + + private static void update(final ServerPushContext context) { + final ImageUploadSelection imageUpload = (ImageUploadSelection) context.getAnchor(); + if (imageUpload.imageBase64 != null + && imageUpload.loadNewImage + && imageUpload.imageLoaded) { + + final Base64InputStream input = new Base64InputStream( + new ByteArrayInputStream( + imageUpload.imageBase64.getBytes(StandardCharsets.UTF_8)), + false); + + setImage(imageUpload, input); + context.layout(); + imageUpload.layout(); + imageUpload.loadNewImage = false; + imageUpload.errorHandler.accept(null); + } + } + + private static void setImage(final ImageUploadSelection imageUpload, final Base64InputStream input) { + imageUpload.imageCanvas.setData(RWT.CUSTOM_VARIANT, "bgLogoNoImage"); + + final Image image = new Image(imageUpload.imageCanvas.getDisplay(), input); + final Rectangle imageBounds = image.getBounds(); + final int width = Math.min(imageBounds.width, imageUpload.maxWidth); + final int height = Math.min(imageBounds.height, imageUpload.maxHeight); + final ImageData imageData = image.getImageData().scaledTo(width, height); + imageUpload.imageCanvas.setBackgroundImage(new Image(imageUpload.imageCanvas.getDisplay(), imageData)); + } + + private static boolean fileSupported(final String fileName) { + return SUPPORTED_IMAGE_FILES + .stream() + .anyMatch(fileType -> fileName.toUpperCase(Locale.ROOT) + .endsWith(fileType.toUpperCase(Locale.ROOT))); + } + + private final class ImageReceiver extends FileUploadReceiver { + @Override + public void receive(final InputStream stream, final FileDetails details) throws IOException { + + try { + final String contentType = details.getContentType(); + if (contentType != null && contentType.startsWith("image")) { + ImageUploadSelection.this.imageBase64 = Base64.getEncoder() + .encodeToString(IOUtils.toByteArray(stream)); + } + } catch (final Exception e) { + log.error("Error while trying to upload image", e); + } finally { + ImageUploadSelection.this.imageLoaded = true; + stream.close(); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCheckbox.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCheckbox.java index 495f38f2..b4d25663 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCheckbox.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCheckbox.java @@ -8,12 +8,11 @@ package ch.ethz.seb.sebserver.gui.widget; -import java.util.Arrays; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; - +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.util.Tuple; +import ch.ethz.seb.sebserver.gbl.util.Tuple3; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.page.PageService; import org.apache.commons.lang3.StringUtils; import org.eclipse.swt.SWT; import org.eclipse.swt.layout.GridData; @@ -22,11 +21,10 @@ import org.eclipse.swt.widgets.Button; import org.eclipse.swt.widgets.Composite; import org.eclipse.swt.widgets.Listener; -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.util.Tuple; -import ch.ethz.seb.sebserver.gbl.util.Tuple3; -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.page.PageService; +import java.util.Arrays; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; public final class MultiSelectionCheckbox extends Composite implements Selection { @@ -121,7 +119,7 @@ public final class MultiSelectionCheckbox extends Composite implements Selection .stream() .filter(Button::getSelection) .map(button -> (String) button.getData(OPTION_VALUE)) - .collect(Collectors.toList()).toArray()); + .toArray()); } @Override diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCombo.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCombo.java index 27b4437f..054fbd75 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCombo.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/MultiSelectionCombo.java @@ -1,252 +1,230 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collection; -import java.util.List; -import java.util.Optional; -import java.util.stream.Collectors; - -import org.apache.commons.lang3.StringUtils; -import org.eclipse.rap.rwt.widgets.DropDown; -import org.eclipse.swt.SWT; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Control; -import org.eclipse.swt.widgets.Event; -import org.eclipse.swt.widgets.Label; -import org.eclipse.swt.widgets.Listener; -import org.eclipse.swt.widgets.Text; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.util.Tuple; -import ch.ethz.seb.sebserver.gui.service.page.PageService; - -public final class MultiSelectionCombo extends Composite implements Selection { - - private static final Logger log = LoggerFactory.getLogger(MultiSelectionCombo.class); - private static final long serialVersionUID = -7787134114963647332L; - - private final WidgetFactory widgetFactory; - - private final List selectionControls = new ArrayList<>(); - - private final List> valueMapping = new ArrayList<>(); - private final List> availableValues = new ArrayList<>(); - private final List> selectedValues = new ArrayList<>(); - - private final DropDown dropDown; - private final Text textInput; - private final GridData textCell; - private final Composite updateAnchor; - - private Listener listener = null; - - MultiSelectionCombo( - final Composite parent, - final WidgetFactory widgetFactory, - final String locTextPrefix, - final Composite updateAnchor) { - - super(parent, SWT.NONE); - this.widgetFactory = widgetFactory; - - final GridLayout gridLayout = new GridLayout(); - gridLayout.verticalSpacing = 1; - gridLayout.marginLeft = 0; - gridLayout.marginHeight = 0; - gridLayout.marginWidth = 0; - gridLayout.horizontalSpacing = 0; - setLayout(gridLayout); - - this.addListener(SWT.Resize, this::adaptColumnWidth); - this.textInput = widgetFactory.textInput(this); - this.textCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); - this.textInput.setLayoutData(this.textCell); - this.dropDown = new DropDown(this.textInput, SWT.NONE); - this.textInput.addListener(SWT.FocusIn, event -> { - openDropDown(); - }); - this.textInput.addListener(SWT.Modify, event -> { - openDropDown(); - }); - this.dropDown.addListener(SWT.Selection, event -> { - final int selectionIndex = this.dropDown.getSelectionIndex(); - if (selectionIndex >= 0) { - final String selectedItem = this.dropDown.getItems()[selectionIndex]; - addSelection(itemForName(selectedItem)); - } - }); - - this.updateAnchor = updateAnchor; - } - - private void openDropDown() { - final String text = this.textInput.getText(); - if (text == null) { - this.dropDown.setVisible(false); - return; - } - final Collection items = this.availableValues - .stream() - .filter(it -> it._2 != null && it._2.startsWith(text)) - .map(t -> t._2) - .collect(Collectors.toList()); - this.dropDown.setItems(items.toArray(new String[items.size()])); - this.dropDown.setSelectionIndex(0); - this.dropDown.setVisible(true); - } - - @Override - public Type type() { - return Type.MULTI_COMBO; - } - - @Override - public void setSelectionListener(final Listener listener) { - this.listener = listener; - } - - @Override - public void applyNewMapping(final List> mapping) { - this.valueMapping.clear(); - this.valueMapping.addAll(mapping); - this.clear(); - } - - @Override - public void select(final String keys) { - clear(); - if (StringUtils.isBlank(keys)) { - return; - } - - Arrays.asList(StringUtils.split(keys, Constants.LIST_SEPARATOR)) - .stream() - .map(this::itemForId) - .forEach(this::addSelection); - } - - @Override - public String getSelectionValue() { - if (this.selectedValues.isEmpty()) { - return null; - } - return this.selectedValues - .stream() - .map(t -> t._1) - .reduce("", (s1, s2) -> { - if (!StringUtils.isBlank(s1)) { - return s1.concat(Constants.LIST_SEPARATOR).concat(s2); - } else { - return s1.concat(s2); - } - }); - } - - @Override - public void clear() { - this.selectedValues.clear(); - this.selectionControls - .stream() - .forEach(Control::dispose); - this.selectionControls.clear(); - this.availableValues.clear(); - this.availableValues.addAll(this.valueMapping); - } - - private void addSelection(final Tuple item) { - if (item == null) { - return; - } - - this.selectedValues.add(item); - final Label label = this.widgetFactory.label(this, item._2); - label.setData(OPTION_VALUE, item._2); - final GridData textCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); - label.setLayoutData(textCell); - label.addListener(SWT.MouseDoubleClick, event -> { - removeComboSelection(event); - }); - this.selectionControls.add(label); - - this.availableValues.remove(item); - PageService.updateScrolledComposite(this); - this.updateAnchor.layout(true, true); - - } - - private void removeComboSelection(final Event event) { - if (event.widget == null) { - return; - } - - final String selectionKey = (String) event.widget.getData(OPTION_VALUE); - final Optional findFirst = this.selectionControls.stream() - .filter(t -> selectionKey.equals(t.getData(OPTION_VALUE))) - .findFirst(); - if (!findFirst.isPresent()) { - return; - } - - final Control control = findFirst.get(); - final int indexOf = this.selectionControls.indexOf(control); - this.selectionControls.remove(control); - control.dispose(); - - final Tuple value = this.selectedValues.remove(indexOf); - this.availableValues.add(value); - - PageService.updateScrolledComposite(this); - this.updateAnchor.layout(true, true); - if (this.listener != null) { - this.listener.handleEvent(event); - } - } - - private void adaptColumnWidth(final Event event) { - try { - final int currentTableWidth = this.getClientArea().width; - this.textCell.widthHint = currentTableWidth; - this.layout(); - } catch (final Exception e) { - log.warn("Failed to adaptColumnWidth: ", e); - } - } - - private Tuple itemForName(final String name) { - final Optional> findFirst = this.availableValues - .stream() - .filter(it -> it._2 != null && it._2.equals(name)) - .findFirst(); - if (findFirst.isPresent()) { - return findFirst.get(); - } - - return null; - } - - private Tuple itemForId(final String id) { - final Optional> findFirst = this.availableValues - .stream() - .filter(it -> it._1 != null && it._1.equals(id)) - .findFirst(); - if (findFirst.isPresent()) { - return findFirst.get(); - } - - return null; - } - -} +/* + * Copyright (c) 2019 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.widget; + +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.util.Tuple; +import ch.ethz.seb.sebserver.gui.service.page.PageService; +import org.apache.commons.lang3.StringUtils; +import org.eclipse.rap.rwt.widgets.DropDown; +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Listener; +import org.eclipse.swt.widgets.Text; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; + +public final class MultiSelectionCombo extends Composite implements Selection { + + private static final Logger log = LoggerFactory.getLogger(MultiSelectionCombo.class); + private static final long serialVersionUID = -7787134114963647332L; + + private final WidgetFactory widgetFactory; + + private final List selectionControls = new ArrayList<>(); + + private final List> valueMapping = new ArrayList<>(); + private final List> availableValues = new ArrayList<>(); + private final List> selectedValues = new ArrayList<>(); + + private final DropDown dropDown; + private final Text textInput; + private final GridData textCell; + private final Composite updateAnchor; + + private Listener listener = null; + + MultiSelectionCombo( + final Composite parent, + final WidgetFactory widgetFactory, + final String locTextPrefix, + final Composite updateAnchor) { + + super(parent, SWT.NONE); + this.widgetFactory = widgetFactory; + + final GridLayout gridLayout = new GridLayout(); + gridLayout.verticalSpacing = 1; + gridLayout.marginLeft = 0; + gridLayout.marginHeight = 0; + gridLayout.marginWidth = 0; + gridLayout.horizontalSpacing = 0; + setLayout(gridLayout); + + this.addListener(SWT.Resize, this::adaptColumnWidth); + this.textInput = widgetFactory.textInput(this); + this.textCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); + this.textInput.setLayoutData(this.textCell); + this.dropDown = new DropDown(this.textInput, SWT.NONE); + this.textInput.addListener(SWT.FocusIn, event -> openDropDown()); + this.textInput.addListener(SWT.Modify, event -> openDropDown()); + this.dropDown.addListener(SWT.Selection, event -> { + final int selectionIndex = this.dropDown.getSelectionIndex(); + if (selectionIndex >= 0) { + final String selectedItem = this.dropDown.getItems()[selectionIndex]; + addSelection(itemForName(selectedItem)); + } + }); + + this.updateAnchor = updateAnchor; + } + + private void openDropDown() { + final String text = this.textInput.getText(); + if (text == null) { + this.dropDown.setVisible(false); + return; + } + this.dropDown.setItems(this.availableValues + .stream() + .filter(it -> it._2 != null && it._2.startsWith(text)) + .map(t -> t._2).toArray(String[]::new)); + this.dropDown.setSelectionIndex(0); + this.dropDown.setVisible(true); + } + + @Override + public Type type() { + return Type.MULTI_COMBO; + } + + @Override + public void setSelectionListener(final Listener listener) { + this.listener = listener; + } + + @Override + public void applyNewMapping(final List> mapping) { + this.valueMapping.clear(); + this.valueMapping.addAll(mapping); + this.clear(); + } + + @Override + public void select(final String keys) { + clear(); + if (StringUtils.isBlank(keys)) { + return; + } + + Arrays.stream(StringUtils.split(keys, Constants.LIST_SEPARATOR)) + .map(this::itemForId) + .forEach(this::addSelection); + } + + @Override + public String getSelectionValue() { + if (this.selectedValues.isEmpty()) { + return null; + } + return this.selectedValues + .stream() + .map(t -> t._1) + .reduce("", (s1, s2) -> { + if (!StringUtils.isBlank(s1)) { + return s1.concat(Constants.LIST_SEPARATOR).concat(s2); + } else { + return s1.concat(s2); + } + }); + } + + @Override + public void clear() { + this.selectedValues.clear(); + this.selectionControls + .forEach(Control::dispose); + this.selectionControls.clear(); + this.availableValues.clear(); + this.availableValues.addAll(this.valueMapping); + } + + private void addSelection(final Tuple item) { + if (item == null) { + return; + } + + this.selectedValues.add(item); + final Label label = this.widgetFactory.label(this, item._2); + label.setData(OPTION_VALUE, item._2); + final GridData textCell = new GridData(SWT.LEFT, SWT.CENTER, true, true); + label.setLayoutData(textCell); + label.addListener(SWT.MouseDoubleClick, this::removeComboSelection); + this.selectionControls.add(label); + + this.availableValues.remove(item); + PageService.updateScrolledComposite(this); + this.updateAnchor.layout(true, true); + + } + + private void removeComboSelection(final Event event) { + if (event.widget == null) { + return; + } + + final String selectionKey = (String) event.widget.getData(OPTION_VALUE); + final Optional findFirst = this.selectionControls.stream() + .filter(t -> selectionKey.equals(t.getData(OPTION_VALUE))) + .findFirst(); + if (!findFirst.isPresent()) { + return; + } + + final Control control = findFirst.get(); + final int indexOf = this.selectionControls.indexOf(control); + this.selectionControls.remove(control); + control.dispose(); + + final Tuple value = this.selectedValues.remove(indexOf); + this.availableValues.add(value); + + PageService.updateScrolledComposite(this); + this.updateAnchor.layout(true, true); + if (this.listener != null) { + this.listener.handleEvent(event); + } + } + + private void adaptColumnWidth(final Event event) { + try { + this.textCell.widthHint = this.getClientArea().width; + this.layout(); + } catch (final Exception e) { + log.warn("Failed to adaptColumnWidth: ", e); + } + } + + private Tuple itemForName(final String name) { + final Optional> findFirst = this.availableValues + .stream() + .filter(it -> it._2 != null && it._2.equals(name)) + .findFirst(); + return findFirst.orElse(null); + } + + private Tuple itemForId(final String id) { + final Optional> findFirst = this.availableValues + .stream() + .filter(it -> it._1 != null && it._1.equals(id)) + .findFirst(); + return findFirst.orElse(null); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/RadioSelection.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/RadioSelection.java index 08e1617b..db0f9204 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/RadioSelection.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/RadioSelection.java @@ -1,122 +1,121 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; - -import org.apache.commons.lang3.StringUtils; -import org.eclipse.swt.SWT; -import org.eclipse.swt.layout.GridData; -import org.eclipse.swt.layout.GridLayout; -import org.eclipse.swt.widgets.Button; -import org.eclipse.swt.widgets.Composite; -import org.eclipse.swt.widgets.Listener; - -import ch.ethz.seb.sebserver.gbl.util.Tuple; -import ch.ethz.seb.sebserver.gbl.util.Utils; -import ch.ethz.seb.sebserver.gui.service.page.PageService; - -public final class RadioSelection extends Composite implements Selection { - - private static final long serialVersionUID = 7937242481193100852L; - - private Listener listener = null; - private final Map radioButtons; - - RadioSelection(final Composite parent) { - super(parent, SWT.NONE); - final GridLayout gridLayout = new GridLayout(1, true); - gridLayout.verticalSpacing = 1; - gridLayout.marginLeft = 0; - gridLayout.marginHeight = 0; - gridLayout.marginWidth = 0; - setLayout(gridLayout); - - this.radioButtons = new LinkedHashMap<>(); - } - - @Override - public Type type() { - return Type.RADIO; - } - - @Override - public void applyNewMapping(final List> mapping) { - final String selectionValue = getSelectionValue(); - this.radioButtons.clear(); - PageService.clearComposite(this); - - for (final Tuple tuple : mapping) { - final Button button = new Button(this, SWT.RADIO); - button.setText(tuple._2); - final GridData gridData = new GridData(SWT.FILL, SWT.CENTER, true, true); - button.setLayoutData(gridData); - button.setData(OPTION_VALUE, tuple._1); - button.addListener(SWT.Selection, event -> { - if (this.listener != null) { - this.listener.handleEvent(event); - } - }); - this.radioButtons.put(tuple._1, button); - } - - if (StringUtils.isNotBlank(selectionValue)) { - select(selectionValue); - } - } - - @Override - public void applyToolTipsForItems(final List> mapping) { - mapping - .stream() - .filter(tuple -> StringUtils.isNotBlank(tuple._2)) - .forEach(tuple -> { - final Button button = this.radioButtons.get(tuple._1); - if (button != null) { - button.setToolTipText(Utils.formatLineBreaks(tuple._2)); - } - }); - } - - @Override - public void select(final String key) { - clear(); - if (StringUtils.isNotBlank(key) && this.radioButtons.containsKey(key)) { - this.radioButtons.get(key).setSelection(true); - } - } - - @Override - public String getSelectionValue() { - return this.radioButtons - .values() - .stream() - .filter(button -> button.getSelection()) - .findFirst() - .map(button -> (String) button.getData(OPTION_VALUE)) - .orElse(null); - } - - @Override - public void clear() { - this.radioButtons - .values() - .stream() - .forEach(button -> button.setSelection(false)); - - } - - @Override - public void setSelectionListener(final Listener listener) { - this.listener = listener; - } - -} +/* + * Copyright (c) 2019 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.widget; + +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; + +import org.apache.commons.lang3.StringUtils; +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Listener; + +import ch.ethz.seb.sebserver.gbl.util.Tuple; +import ch.ethz.seb.sebserver.gbl.util.Utils; +import ch.ethz.seb.sebserver.gui.service.page.PageService; + +public final class RadioSelection extends Composite implements Selection { + + private static final long serialVersionUID = 7937242481193100852L; + + private Listener listener = null; + private final Map radioButtons; + + RadioSelection(final Composite parent) { + super(parent, SWT.NONE); + final GridLayout gridLayout = new GridLayout(1, true); + gridLayout.verticalSpacing = 1; + gridLayout.marginLeft = 0; + gridLayout.marginHeight = 0; + gridLayout.marginWidth = 0; + setLayout(gridLayout); + + this.radioButtons = new LinkedHashMap<>(); + } + + @Override + public Type type() { + return Type.RADIO; + } + + @Override + public void applyNewMapping(final List> mapping) { + final String selectionValue = getSelectionValue(); + this.radioButtons.clear(); + PageService.clearComposite(this); + + for (final Tuple tuple : mapping) { + final Button button = new Button(this, SWT.RADIO); + button.setText(tuple._2); + final GridData gridData = new GridData(SWT.FILL, SWT.CENTER, true, true); + button.setLayoutData(gridData); + button.setData(OPTION_VALUE, tuple._1); + button.addListener(SWT.Selection, event -> { + if (this.listener != null) { + this.listener.handleEvent(event); + } + }); + this.radioButtons.put(tuple._1, button); + } + + if (StringUtils.isNotBlank(selectionValue)) { + select(selectionValue); + } + } + + @Override + public void applyToolTipsForItems(final List> mapping) { + mapping + .stream() + .filter(tuple -> StringUtils.isNotBlank(tuple._2)) + .forEach(tuple -> { + final Button button = this.radioButtons.get(tuple._1); + if (button != null) { + button.setToolTipText(Utils.formatLineBreaks(tuple._2)); + } + }); + } + + @Override + public void select(final String key) { + clear(); + if (StringUtils.isNotBlank(key) && this.radioButtons.containsKey(key)) { + this.radioButtons.get(key).setSelection(true); + } + } + + @Override + public String getSelectionValue() { + return this.radioButtons + .values() + .stream() + .filter(Button::getSelection) + .findFirst() + .map(button -> (String) button.getData(OPTION_VALUE)) + .orElse(null); + } + + @Override + public void clear() { + this.radioButtons + .values() + .forEach(button -> button.setSelection(false)); + + } + + @Override + public void setSelectionListener(final Listener listener) { + this.listener = listener; + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/Selection.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/Selection.java index 03b68cf3..2b1e6c3f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/Selection.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/Selection.java @@ -1,65 +1,65 @@ -/* - * Copyright (c) 2019 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.widget; - -import java.util.List; - -import org.eclipse.swt.widgets.Control; -import org.eclipse.swt.widgets.Listener; - -import ch.ethz.seb.sebserver.gbl.util.Tuple; - -public interface Selection { - - static final String OPTION_VALUE = "OPTION_VALUE"; - - enum Type { - SINGLE, - SINGLE_COMBO, - RADIO, - MULTI, - MULTI_COMBO, - MULTI_CHECKBOX, - COLOR, - } - - Type type(); - - void applyNewMapping(final List> mapping); - - void select(final String keys); - - String getSelectionValue(); - - default String getSelectionReadableValue() { - return getSelectionValue(); - } - - void clear(); - - void setVisible(boolean visible); - - void setSelectionListener(Listener listener); - - void setToolTipText(String tooltipText); - - default void applyToolTipsForItems(final List> mapping) { - throw new UnsupportedOperationException("Must be implemented for this specific Selection"); - } - - default Control adaptToControl() { - return (Control) this; - } - - @SuppressWarnings("unchecked") - default T getTypeInstance() { - return (T) this; - } - -} +/* + * Copyright (c) 2019 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.widget; + +import java.util.List; + +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Listener; + +import ch.ethz.seb.sebserver.gbl.util.Tuple; + +public interface Selection { + + String OPTION_VALUE = "OPTION_VALUE"; + + enum Type { + SINGLE, + SINGLE_COMBO, + RADIO, + MULTI, + MULTI_COMBO, + MULTI_CHECKBOX, + COLOR, + } + + Type type(); + + void applyNewMapping(final List> mapping); + + void select(final String keys); + + String getSelectionValue(); + + default String getSelectionReadableValue() { + return getSelectionValue(); + } + + void clear(); + + void setVisible(boolean visible); + + void setSelectionListener(Listener listener); + + void setToolTipText(String tooltipText); + + default void applyToolTipsForItems(final List> mapping) { + throw new UnsupportedOperationException("Must be implemented for this specific Selection"); + } + + default Control adaptToControl() { + return (Control) this; + } + + @SuppressWarnings("unchecked") + default T getTypeInstance() { + return (T) this; + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/SingleSelection.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/SingleSelection.java index 33f4686b..6c8ef8dd 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/SingleSelection.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/SingleSelection.java @@ -87,7 +87,7 @@ public final class SingleSelection extends Combo implements Selection { @Override public void clear() { super.clearSelection(); - super.setItems(this.valueMapping.toArray(new String[this.valueMapping.size()])); + super.setItems(this.valueMapping.toArray(new String[0])); } @Override diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/widget/WidgetFactory.java b/src/main/java/ch/ethz/seb/sebserver/gui/widget/WidgetFactory.java index 4675e3c0..03ec4d87 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/widget/WidgetFactory.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/widget/WidgetFactory.java @@ -123,7 +123,7 @@ public class WidgetFactory { private ImageData image = null; private ImageData greyedImage = null; - private ImageIcon(final String fileName) { + ImageIcon(final String fileName) { this.fileName = fileName; } @@ -199,7 +199,7 @@ public class WidgetFactory { public final String key; - private CustomVariant(final String key) { + CustomVariant(final String key) { this.key = key; } } @@ -268,7 +268,7 @@ public class WidgetFactory { * @param parent The parent Composite * @return the scrolled Composite to add the form content */ public Composite createPopupScrollComposite(final Composite parent) { - final Composite grid = PageService.createManagedVScrolledComposite( + return PageService.createManagedVScrolledComposite( parent, scrolledComposite -> { final Composite g = new Composite(scrolledComposite, SWT.NONE); @@ -277,7 +277,6 @@ public class WidgetFactory { return g; }, false); - return grid; } public Composite createWarningPanel(final Composite parent) { @@ -733,7 +732,7 @@ public class WidgetFactory { new FileUploadSelection(parent, this.i18nSupport, readonly); if (supportedFiles != null) { - supportedFiles.forEach(ext -> fileUploadSelection.withSupportFor(ext)); + supportedFiles.forEach(fileUploadSelection::withSupportFor); } return fileUploadSelection; } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java b/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java index eebfb41a..0e55f57c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java @@ -1,113 +1,106 @@ -/* - * 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.webservice; - -import java.net.InetAddress; -import java.net.UnknownHostException; - -import javax.annotation.PreDestroy; - -import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; -import org.springframework.boot.context.event.ApplicationReadyEvent; -import org.springframework.context.ApplicationEventPublisher; -import org.springframework.context.ApplicationListener; -import org.springframework.context.annotation.Import; -import org.springframework.core.env.Environment; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.SEBServerInit; -import ch.ethz.seb.sebserver.SEBServerInitEvent; -import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; - -@Component -@WebServiceProfile -@Import(DataSourceAutoConfiguration.class) -public class WebserviceInit implements ApplicationListener { - - private final SEBServerInit sebServerInit; - private final Environment environment; - private final WebserviceInfo webserviceInfo; - private final AdminUserInitializer adminUserInitializer; - private final ApplicationEventPublisher applicationEventPublisher; - - protected WebserviceInit( - final SEBServerInit sebServerInit, - final Environment environment, - final WebserviceInfo webserviceInfo, - final AdminUserInitializer adminUserInitializer, - final ApplicationEventPublisher applicationEventPublisher) { - - this.sebServerInit = sebServerInit; - this.environment = environment; - this.webserviceInfo = webserviceInfo; - this.adminUserInitializer = adminUserInitializer; - this.applicationEventPublisher = applicationEventPublisher; - } - - @Override - public void onApplicationEvent(final ApplicationReadyEvent event) { - - this.sebServerInit.init(); - - SEBServerInit.INIT_LOGGER.info("----> **** Webservice starting up... ****"); - - SEBServerInit.INIT_LOGGER.info("----> "); - SEBServerInit.INIT_LOGGER.info("----> Init Database with flyway..."); - SEBServerInit.INIT_LOGGER.info("----> TODO "); - - // TODO integration of Flyway for database initialization and migration: https://flywaydb.org - // see also https://flywaydb.org/getstarted/firststeps/api - - SEBServerInit.INIT_LOGGER.info("----> "); - SEBServerInit.INIT_LOGGER.info("----> Intitialize Services..."); - SEBServerInit.INIT_LOGGER.info("----> "); - - this.applicationEventPublisher.publishEvent(new SEBServerInitEvent(this)); - - SEBServerInit.INIT_LOGGER.info("----> "); - SEBServerInit.INIT_LOGGER.info("----> **** Webservice successfully started up! **** "); - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> *** Info:"); - - try { - SEBServerInit.INIT_LOGGER.info("----> Server address: {}", this.environment.getProperty("server.address")); - SEBServerInit.INIT_LOGGER.info("----> Server port: {}", this.environment.getProperty("server.port")); - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> Local-Host address: {}", InetAddress.getLocalHost().getHostAddress()); - SEBServerInit.INIT_LOGGER.info("----> Local-Host name: {}", InetAddress.getLocalHost().getHostName()); - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> Remote-Host address: {}", - InetAddress.getLoopbackAddress().getHostAddress()); - SEBServerInit.INIT_LOGGER.info("----> Remote-Host name: {}", - InetAddress.getLoopbackAddress().getHostName()); - } catch (final UnknownHostException e) { - SEBServerInit.INIT_LOGGER.error("Unknown Host: ", e); - } - - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> External-Host URL: {}", this.webserviceInfo.getExternalServerURL()); - SEBServerInit.INIT_LOGGER.info("----> LMS-External-Address-Alias: {}", - this.webserviceInfo.getLmsExternalAddressAlias()); - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> HTTP Scheme {}", this.webserviceInfo.getHttpScheme()); - SEBServerInit.INIT_LOGGER.info("---->"); - SEBServerInit.INIT_LOGGER.info("----> Property Override Test: {}", this.webserviceInfo.getTestProperty()); - - // Create an initial admin account if requested and not already in the data-base - this.adminUserInitializer.initAdminAccount(); - - } - - @PreDestroy - public void gracefulShutdown() { - SEBServerInit.INIT_LOGGER.info("**** Gracefully Shutdown of SEB Server instance {} ****", - this.webserviceInfo.getHostAddress()); - } - -} +/* + * 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.webservice; + +import java.net.InetAddress; +import java.net.UnknownHostException; + +import javax.annotation.PreDestroy; + +import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; +import org.springframework.boot.context.event.ApplicationReadyEvent; +import org.springframework.context.ApplicationEventPublisher; +import org.springframework.context.ApplicationListener; +import org.springframework.context.annotation.Import; +import org.springframework.core.env.Environment; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.SEBServerInit; +import ch.ethz.seb.sebserver.SEBServerInitEvent; +import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; + +@Component +@WebServiceProfile +@Import(DataSourceAutoConfiguration.class) +public class WebserviceInit implements ApplicationListener { + + private final SEBServerInit sebServerInit; + private final Environment environment; + private final WebserviceInfo webserviceInfo; + private final AdminUserInitializer adminUserInitializer; + private final ApplicationEventPublisher applicationEventPublisher; + + protected WebserviceInit( + final SEBServerInit sebServerInit, + final Environment environment, + final WebserviceInfo webserviceInfo, + final AdminUserInitializer adminUserInitializer, + final ApplicationEventPublisher applicationEventPublisher) { + + this.sebServerInit = sebServerInit; + this.environment = environment; + this.webserviceInfo = webserviceInfo; + this.adminUserInitializer = adminUserInitializer; + this.applicationEventPublisher = applicationEventPublisher; + } + + @Override + public void onApplicationEvent(final ApplicationReadyEvent event) { + + this.sebServerInit.init(); + + SEBServerInit.INIT_LOGGER.info("----> **** Webservice starting up... ****"); + + SEBServerInit.INIT_LOGGER.info("----> "); + SEBServerInit.INIT_LOGGER.info("----> Intitialize Services..."); + SEBServerInit.INIT_LOGGER.info("----> "); + + this.applicationEventPublisher.publishEvent(new SEBServerInitEvent(this)); + + SEBServerInit.INIT_LOGGER.info("----> "); + SEBServerInit.INIT_LOGGER.info("----> **** Webservice successfully started up! **** "); + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> *** Info:"); + + try { + SEBServerInit.INIT_LOGGER.info("----> Server address: {}", this.environment.getProperty("server.address")); + SEBServerInit.INIT_LOGGER.info("----> Server port: {}", this.environment.getProperty("server.port")); + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> Local-Host address: {}", InetAddress.getLocalHost().getHostAddress()); + SEBServerInit.INIT_LOGGER.info("----> Local-Host name: {}", InetAddress.getLocalHost().getHostName()); + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> Remote-Host address: {}", + InetAddress.getLoopbackAddress().getHostAddress()); + SEBServerInit.INIT_LOGGER.info("----> Remote-Host name: {}", + InetAddress.getLoopbackAddress().getHostName()); + } catch (final UnknownHostException e) { + SEBServerInit.INIT_LOGGER.error("Unknown Host: ", e); + } + + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> External-Host URL: {}", this.webserviceInfo.getExternalServerURL()); + SEBServerInit.INIT_LOGGER.info("----> LMS-External-Address-Alias: {}", + this.webserviceInfo.getLmsExternalAddressAlias()); + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> HTTP Scheme {}", this.webserviceInfo.getHttpScheme()); + SEBServerInit.INIT_LOGGER.info("---->"); + SEBServerInit.INIT_LOGGER.info("----> Property Override Test: {}", this.webserviceInfo.getTestProperty()); + + // Create an initial admin account if requested and not already in the data-base + this.adminUserInitializer.initAdminAccount(); + + } + + @PreDestroy + public void gracefulShutdown() { + SEBServerInit.INIT_LOGGER.info("**** Gracefully Shutdown of SEB Server instance {} ****", + this.webserviceInfo.getHostAddress()); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/lms/LmsAPIService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/lms/LmsAPIService.java index cbdf65ab..33e27a37 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/lms/LmsAPIService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/lms/LmsAPIService.java @@ -23,6 +23,7 @@ import ch.ethz.seb.sebserver.gbl.model.institution.LmsSetup; import ch.ethz.seb.sebserver.gbl.model.institution.LmsSetupTestResult; import ch.ethz.seb.sebserver.gbl.util.Result; import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap; +import org.joda.time.DateTimeZone; /** Defines the LMS API access service interface with all functionality needed to access * a LMS API within a given LmsSetup configuration. @@ -101,12 +102,12 @@ public interface LmsAPIService { static Predicate quizFilterPredicate(final FilterMap filterMap) { final String name = filterMap.getQuizName(); final DateTime from = filterMap.getQuizFromTime(); - //final DateTime now = DateTime.now(DateTimeZone.UTC); return q -> { final boolean nameFilter = StringUtils.isBlank(name) || (q.name != null && q.name.contains(name)); final boolean startTimeFilter = (from == null) || (q.startTime != null && (q.startTime.isEqual(from) || q.startTime.isAfter(from))); - return nameFilter && startTimeFilter /* && endTimeFilter */; + final boolean currentlyRunning = DateTime.now(DateTimeZone.UTC).isBefore(q.endTime); + return nameFilter && (startTimeFilter || currentlyRunning) ; }; } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/ClientConfigService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/ClientConfigService.java index 1a3f1a61..dda1369f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/ClientConfigService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/ClientConfigService.java @@ -60,10 +60,10 @@ public interface ClientConfigService { unless = "#result.hasError()") Result getClientConfigDetails(String clientName); - @CacheEvict( - cacheNames = EXAM_CLIENT_DETAILS_CACHE, - allEntries = true) - @EventListener(BulkActionEvent.class) - void flushClientConfigData(BulkActionEvent event); - + /** Internally used to check OAuth2 access for a active SebClientConfig. + * + * @param config the SebClientConfig to check access + * @return true if the system was able to gain an access token for the client. False otherwise + */ + boolean checkAccess(SebClientConfig config); } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/impl/ClientConfigServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/impl/ClientConfigServiceImpl.java index 7be036c9..f851615f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/impl/ClientConfigServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/sebconfig/impl/ClientConfigServiceImpl.java @@ -10,8 +10,7 @@ package ch.ethz.seb.sebserver.webservice.servicelayer.sebconfig.impl; import ch.ethz.seb.sebserver.WebSecurityConfig; import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType; -import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.api.API; import ch.ethz.seb.sebserver.gbl.model.EntityKey; import ch.ethz.seb.sebserver.gbl.model.institution.Institution; import ch.ethz.seb.sebserver.gbl.model.sebconfig.SebClientConfig; @@ -19,8 +18,6 @@ import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; import ch.ethz.seb.sebserver.gbl.util.Result; import ch.ethz.seb.sebserver.gbl.util.Utils; import ch.ethz.seb.sebserver.webservice.WebserviceInfo; -import ch.ethz.seb.sebserver.webservice.servicelayer.bulkaction.impl.BulkAction; -import ch.ethz.seb.sebserver.webservice.servicelayer.bulkaction.impl.BulkActionEvent; import ch.ethz.seb.sebserver.webservice.servicelayer.client.ClientCredentialService; import ch.ethz.seb.sebserver.webservice.servicelayer.client.ClientCredentials; import ch.ethz.seb.sebserver.webservice.servicelayer.dao.InstitutionDAO; @@ -38,12 +35,21 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.context.annotation.Lazy; +import org.springframework.http.HttpEntity; +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpMethod; +import org.springframework.http.HttpStatus; +import org.springframework.http.MediaType; +import org.springframework.http.ResponseEntity; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.security.oauth2.common.OAuth2AccessToken; import org.springframework.security.oauth2.provider.ClientDetails; import org.springframework.security.oauth2.provider.client.BaseClientDetails; import org.springframework.security.oauth2.provider.token.TokenStore; import org.springframework.stereotype.Service; +import org.springframework.util.LinkedMultiValueMap; +import org.springframework.util.MultiValueMap; +import org.springframework.web.client.RestTemplate; import java.io.IOException; import java.io.InputStream; @@ -51,6 +57,7 @@ import java.io.OutputStream; import java.io.PipedInputStream; import java.io.PipedOutputStream; import java.nio.charset.StandardCharsets; +import java.util.Base64; import java.util.Collection; import java.util.Collections; import java.util.UUID; @@ -316,19 +323,48 @@ public class ClientConfigServiceImpl implements ClientConfigService { } @Override - public void flushClientConfigData(final BulkActionEvent event) { + public boolean checkAccess(SebClientConfig config) { + if(!config.isActive()) { + return false; + } + try { - final BulkAction bulkAction = event.getBulkAction(); + RestTemplate restTemplate = new RestTemplate(); + String externalServerURL = webserviceInfo.getExternalServerURL() + + API.OAUTH_TOKEN_ENDPOINT; - if (bulkAction.type == BulkActionType.DEACTIVATE || - bulkAction.type == BulkActionType.HARD_DELETE) { + MultiValueMap headers = new LinkedMultiValueMap<>(); + headers.add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_FORM_URLENCODED_VALUE); + ClientCredentials credentials = sebClientConfigDAO + .getSebClientCredentials(config.getModelId()) + .getOrThrow(); + CharSequence plainClientSecret = clientCredentialService.getPlainClientSecret(credentials); + String basicAuth = credentials.clientId + + String.valueOf(Constants.COLON) + + plainClientSecret; + String encoded = Base64.getEncoder() + .encodeToString(basicAuth.getBytes()); - bulkAction.extractKeys(EntityType.SEB_CLIENT_CONFIGURATION) - .forEach(this::flushClientConfigData); + headers.add(HttpHeaders.AUTHORIZATION, "Basic " + encoded); + HttpEntity entity = new HttpEntity<>( + "grant_type=client_credentials&scope=read write", + headers); + + ResponseEntity exchange = restTemplate.exchange( + externalServerURL, + HttpMethod.POST, + entity, + String.class); + + if (exchange.getStatusCode().value() == HttpStatus.OK.value()) { + return true; + } else { + log.warn("Failed to check access SebClientConfig {} response: {}", config, exchange.getStatusCode()); + return false; } - - } catch (final Exception e) { - log.error("Unexpected error while trying to flush ClientConfig data ", e); + } catch (Exception e) { + log.warn("Failed to check access for SebClientConfig: {} cause: {}", config, e.getMessage()); + return false; } } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ClientIndicatorFactory.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ClientIndicatorFactory.java index 122fef6d..3b5d71fc 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ClientIndicatorFactory.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ClientIndicatorFactory.java @@ -1,109 +1,121 @@ -/* - * 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.webservice.servicelayer.session.impl; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Collections; -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.ApplicationContext; -import org.springframework.context.annotation.Lazy; -import org.springframework.stereotype.Component; - -import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; -import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.IndicatorType; -import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; -import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; -import ch.ethz.seb.sebserver.webservice.servicelayer.dao.IndicatorDAO; -import ch.ethz.seb.sebserver.webservice.servicelayer.session.ClientIndicator; - -@Lazy -@Component -@WebServiceProfile -public class ClientIndicatorFactory { - - private static final Logger log = LoggerFactory.getLogger(ClientIndicatorFactory.class); - - private final ApplicationContext applicationContext; - private final IndicatorDAO indicatorDAO; - private final boolean enableCaching; - - @Autowired - public ClientIndicatorFactory( - final ApplicationContext applicationContext, - final IndicatorDAO indicatorDAO, - @Value("${sebserver.webservice.api.exam.enable-indicator-cache:true}") final boolean enableCaching) { - - this.applicationContext = applicationContext; - this.indicatorDAO = indicatorDAO; - this.enableCaching = enableCaching; - } - - public List createFor(final ClientConnection clientConnection) { - final List result = new ArrayList<>(); - - if (clientConnection.examId == null) { - return result; - } - - try { - - final Collection examIndicators = this.indicatorDAO - .allForExam(clientConnection.examId) - .getOrThrow(); - - boolean pingIndicatorAvailable = false; - - for (final Indicator indicatorDef : examIndicators) { - try { - - final ClientIndicator indicator = this.applicationContext - .getBean(indicatorDef.type.name(), ClientIndicator.class); - - if (!pingIndicatorAvailable) { - pingIndicatorAvailable = indicatorDef.type == IndicatorType.LAST_PING; - } - - indicator.init( - indicatorDef, - clientConnection.id, - this.enableCaching); - - result.add(indicator); - } catch (final Exception e) { - log.warn("No Indicator with type: {} found as registered bean. Ignore this one.", indicatorDef.type, - e); - } - } - - // If there is no ping interval indicator set from the exam, we add a hidden one - // to at least create missing ping events and track missing state - if (!pingIndicatorAvailable) { - final PingIntervalClientIndicator pingIndicator = this.applicationContext - .getBean(PingIntervalClientIndicator.class); - pingIndicator.hidden = true; - result.add(pingIndicator); - } - - } catch (final RuntimeException e) { - log.error("Failed to create ClientIndicator for ClientConnection: {}", clientConnection); - throw e; - } catch (final Exception e) { - log.error("Failed to create ClientIndicator for ClientConnection: {}", clientConnection); - } - - return Collections.unmodifiableList(result); - } - -} +/* + * 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.webservice.servicelayer.session.impl; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.ApplicationContext; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; +import ch.ethz.seb.sebserver.gbl.model.exam.Indicator.IndicatorType; +import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; +import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; +import ch.ethz.seb.sebserver.webservice.servicelayer.dao.IndicatorDAO; +import ch.ethz.seb.sebserver.webservice.servicelayer.session.ClientIndicator; + +@Lazy +@Component +@WebServiceProfile +public class ClientIndicatorFactory { + + private static final Logger log = LoggerFactory.getLogger(ClientIndicatorFactory.class); + + private final ApplicationContext applicationContext; + private final IndicatorDAO indicatorDAO; + private final boolean enableCaching; + + @Autowired + public ClientIndicatorFactory( + final ApplicationContext applicationContext, + final IndicatorDAO indicatorDAO, + @Value("${sebserver.webservice.api.exam.enable-indicator-cache:true}") final boolean enableCaching) { + + this.applicationContext = applicationContext; + this.indicatorDAO = indicatorDAO; + this.enableCaching = enableCaching; + } + + public List createFor(final ClientConnection clientConnection) { + final List result = new ArrayList<>(); + + if (clientConnection.examId == null) { + return result; + } + + try { + + final Collection examIndicators = this.indicatorDAO + .allForExam(clientConnection.examId) + .getOrThrow(); + + boolean pingIndicatorAvailable = false; + + for (final Indicator indicatorDef : examIndicators) { + try { + + final ClientIndicator indicator = this.applicationContext + .getBean(indicatorDef.type.name(), ClientIndicator.class); + + if (!pingIndicatorAvailable) { + pingIndicatorAvailable = indicatorDef.type == IndicatorType.LAST_PING; + } + + indicator.init( + indicatorDef, + clientConnection.id, + this.enableCaching); + + result.add(indicator); + } catch (final Exception e) { + log.warn("No Indicator with type: {} found as registered bean. Ignore this one.", indicatorDef.type, + e); + } + } + + // If there is no ping interval indicator set from the exam, we add a hidden one + // to at least create missing ping events and track missing state + if (!pingIndicatorAvailable) { + final PingIntervalClientIndicator pingIndicator = this.applicationContext + .getBean(PingIntervalClientIndicator.class); + pingIndicator.hidden = true; + final Indicator indicator = new Indicator( + null, + clientConnection.examId, + "hidden_ping_indicator", + IndicatorType.LAST_PING, + "", + Arrays.asList(new Indicator.Threshold(5000d, ""))); + pingIndicator.init( + indicator, + clientConnection.id, + this.enableCaching); + result.add(pingIndicator); + } + + } catch (final RuntimeException e) { + log.error("Failed to create ClientIndicator for ClientConnection: {}", clientConnection); + throw e; + } catch (final Exception e) { + log.error("Failed to create ClientIndicator for ClientConnection: {}", clientConnection); + } + + return Collections.unmodifiableList(result); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/PingIntervalClientIndicator.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/PingIntervalClientIndicator.java index 648e4354..1a584210 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/PingIntervalClientIndicator.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/PingIntervalClientIndicator.java @@ -38,7 +38,6 @@ public final class PingIntervalClientIndicator extends AbstractPingIndicator { long pingErrorThreshold; boolean missingPing = false; - boolean hidden = false; public PingIntervalClientIndicator(final ClientEventExtensionMapper clientEventExtensionMapper) { diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java index 343b1b3e..be6cbefe 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java @@ -1,135 +1,141 @@ -/* - * Copyright (c) 2019 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.webservice.weblayer.api; - -import java.util.Collection; - -import org.mybatis.dynamic.sql.SqlTable; -import org.springframework.http.MediaType; -import org.springframework.util.MultiValueMap; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RequestParam; -import org.springframework.web.bind.annotation.RestController; - -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType; -import ch.ethz.seb.sebserver.gbl.api.EntityType; -import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; -import ch.ethz.seb.sebserver.gbl.model.EntityKey; -import ch.ethz.seb.sebserver.gbl.model.GrantEntity; -import ch.ethz.seb.sebserver.gbl.model.Page; -import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent; -import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent; -import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; -import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecordDynamicSqlSupport; -import ch.ethz.seb.sebserver.webservice.servicelayer.PaginationService; -import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.AuthorizationService; -import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.PermissionDeniedException; -import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.UserService; -import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.impl.SEBServerUser; -import ch.ethz.seb.sebserver.webservice.servicelayer.bulkaction.BulkActionService; -import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ClientEventDAO; -import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ExamDAO; -import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap; -import ch.ethz.seb.sebserver.webservice.servicelayer.dao.UserActivityLogDAO; -import ch.ethz.seb.sebserver.webservice.servicelayer.validation.BeanValidationService; - -@WebServiceProfile -@RestController -@RequestMapping("${sebserver.webservice.api.admin.endpoint}" + API.SEB_CLIENT_EVENT_ENDPOINT) -public class ClientEventController extends ReadonlyEntityController { - - private final ExamDAO examDAO; - private final ClientEventDAO clientEventDAO; - - protected ClientEventController( - final AuthorizationService authorization, - final BulkActionService bulkActionService, - final ClientEventDAO entityDAO, - final UserActivityLogDAO userActivityLogDAO, - final PaginationService paginationService, - final BeanValidationService beanValidationService, - final ExamDAO examDAO) { - - super(authorization, - bulkActionService, - entityDAO, - userActivityLogDAO, - paginationService, - beanValidationService); - - this.examDAO = examDAO; - this.clientEventDAO = entityDAO; - } - - @RequestMapping( - path = API.SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT, - method = RequestMethod.GET, - consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE, - produces = MediaType.APPLICATION_JSON_UTF8_VALUE) - public Page getExtendedPage( - @RequestParam( - name = API.PARAM_INSTITUTION_ID, - required = true, - defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId, - @RequestParam(name = Page.ATTR_PAGE_NUMBER, required = false) final Integer pageNumber, - @RequestParam(name = Page.ATTR_PAGE_SIZE, required = false) final Integer pageSize, - @RequestParam(name = Page.ATTR_SORT, required = false) final String sort, - @RequestParam final MultiValueMap allRequestParams) { - - // at least current user must have base read access for specified entity type within its own institution - checkReadPrivilege(institutionId); - - final FilterMap filterMap = new FilterMap(allRequestParams); - - // if current user has no read access for specified entity type within other institution - // then the current users institutionId is put as a SQL filter criteria attribute to extends query performance - if (!this.authorization.hasGrant(PrivilegeType.READ, getGrantEntityType())) { - filterMap.putIfAbsent(API.PARAM_INSTITUTION_ID, String.valueOf(institutionId)); - } - - return this.paginationService.getPage( - pageNumber, - pageSize, - sort, - getSQLTableOfEntity().name(), - () -> this.clientEventDAO.allMatchingExtended(filterMap, this::hasReadAccess)) - .getOrThrow(); - } - - @Override - public Collection getDependencies(final String modelId, final BulkActionType bulkActionType) { - throw new UnsupportedOperationException(); - } - - @Override - protected SqlTable getSQLTableOfEntity() { - return ClientEventRecordDynamicSqlSupport.clientEventRecord; - } - - @Override - protected GrantEntity toGrantEntity(final ClientEvent entity) { - return this.examDAO - .byClientConnection(entity.connectionId) - .getOrThrow(); - } - - @Override - protected void checkReadPrivilege(final Long institutionId) { - final SEBServerUser currentUser = this.authorization.getUserService().getCurrentUser(); - if (currentUser.institutionId().longValue() != institutionId.longValue()) { - throw new PermissionDeniedException( - EntityType.CLIENT_EVENT, - PrivilegeType.READ, - currentUser.getUserInfo()); - } - } - -} +/* + * Copyright (c) 2019 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.webservice.weblayer.api; + +import java.util.Collection; + +import org.mybatis.dynamic.sql.SqlTable; +import org.springframework.http.MediaType; +import org.springframework.util.MultiValueMap; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestMethod; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; + +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType; +import ch.ethz.seb.sebserver.gbl.api.EntityType; +import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; +import ch.ethz.seb.sebserver.gbl.model.EntityKey; +import ch.ethz.seb.sebserver.gbl.model.GrantEntity; +import ch.ethz.seb.sebserver.gbl.model.Page; +import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent; +import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent; +import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; +import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecordDynamicSqlSupport; +import ch.ethz.seb.sebserver.webservice.servicelayer.PaginationService; +import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.AuthorizationService; +import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.PermissionDeniedException; +import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.UserService; +import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.impl.SEBServerUser; +import ch.ethz.seb.sebserver.webservice.servicelayer.bulkaction.BulkActionService; +import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ClientEventDAO; +import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ExamDAO; +import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap; +import ch.ethz.seb.sebserver.webservice.servicelayer.dao.UserActivityLogDAO; +import ch.ethz.seb.sebserver.webservice.servicelayer.validation.BeanValidationService; + +@WebServiceProfile +@RestController +@RequestMapping("${sebserver.webservice.api.admin.endpoint}" + API.SEB_CLIENT_EVENT_ENDPOINT) +public class ClientEventController extends ReadonlyEntityController { + + private final ExamDAO examDAO; + private final ClientEventDAO clientEventDAO; + + protected ClientEventController( + final AuthorizationService authorization, + final BulkActionService bulkActionService, + final ClientEventDAO entityDAO, + final UserActivityLogDAO userActivityLogDAO, + final PaginationService paginationService, + final BeanValidationService beanValidationService, + final ExamDAO examDAO) { + + super(authorization, + bulkActionService, + entityDAO, + userActivityLogDAO, + paginationService, + beanValidationService); + + this.examDAO = examDAO; + this.clientEventDAO = entityDAO; + } + + @RequestMapping( + path = API.SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT, + method = RequestMethod.GET, + consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE, + produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + public Page getExtendedPage( + @RequestParam( + name = API.PARAM_INSTITUTION_ID, + required = true, + defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId, + @RequestParam(name = Page.ATTR_PAGE_NUMBER, required = false) final Integer pageNumber, + @RequestParam(name = Page.ATTR_PAGE_SIZE, required = false) final Integer pageSize, + @RequestParam(name = Page.ATTR_SORT, required = false) final String sort, + @RequestParam final MultiValueMap allRequestParams) { + + // at least current user must have base read access for specified entity type within its own institution + checkReadPrivilege(institutionId); + + final FilterMap filterMap = new FilterMap(allRequestParams); + + // if current user has no read access for specified entity type within other institution + // then the current users institutionId is put as a SQL filter criteria attribute to extends query performance + if (!this.authorization.hasGrant(PrivilegeType.READ, getGrantEntityType())) { + filterMap.putIfAbsent(API.PARAM_INSTITUTION_ID, String.valueOf(institutionId)); + } + + try { + + return this.paginationService.getPage( + pageNumber, + pageSize, + sort, + getSQLTableOfEntity().name(), + () -> this.clientEventDAO.allMatchingExtended(filterMap, this::hasReadAccess)) + .getOrThrow(); + } catch (Exception e) { + e.printStackTrace(); + throw e; + } + } + + @Override + public Collection getDependencies(final String modelId, final BulkActionType bulkActionType) { + throw new UnsupportedOperationException(); + } + + @Override + protected SqlTable getSQLTableOfEntity() { + return ClientEventRecordDynamicSqlSupport.clientEventRecord; + } + + @Override + protected GrantEntity toGrantEntity(final ClientEvent entity) { + return this.examDAO + .byClientConnection(entity.connectionId) + .get(); + } + + @Override + protected void checkReadPrivilege(final Long institutionId) { + final SEBServerUser currentUser = this.authorization.getUserService().getCurrentUser(); + if (currentUser.institutionId().longValue() != institutionId.longValue()) { + throw new PermissionDeniedException( + EntityType.CLIENT_EVENT, + PrivilegeType.READ, + currentUser.getUserInfo()); + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/SebClientConfigController.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/SebClientConfigController.java index bd73fe1b..129085e8 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/SebClientConfigController.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/SebClientConfigController.java @@ -39,6 +39,7 @@ import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.client.RestTemplate; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletResponse; @@ -144,6 +145,15 @@ public class SebClientConfigController extends ActivatableEntityController notifySaved(SebClientConfig entity) { + if (entity.isActive()) { + // try to get access token for SEB client + sebClientConfigService.checkAccess(entity); + } + return super.notifySaved(entity); + } + private SebClientConfig checkPasswordMatch(final SebClientConfig entity) { Collection errors = new ArrayList<>(); if (entity.hasEncryptionSecret() && !entity.encryptSecret.equals(entity.encryptSecretConfirm)) { diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/oauth/DefaultTokenServicesFallback.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/oauth/DefaultTokenServicesFallback.java index 66b93c00..4165faeb 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/oauth/DefaultTokenServicesFallback.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/oauth/DefaultTokenServicesFallback.java @@ -1,58 +1,59 @@ -/* - * Copyright (c) 2019 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.webservice.weblayer.oauth; - -import org.apache.commons.lang3.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.dao.DuplicateKeyException; -import org.springframework.security.core.AuthenticationException; -import org.springframework.security.oauth2.common.OAuth2AccessToken; -import org.springframework.security.oauth2.provider.OAuth2Authentication; -import org.springframework.security.oauth2.provider.token.DefaultTokenServices; - -public class DefaultTokenServicesFallback extends DefaultTokenServices { - - private static final Logger log = LoggerFactory.getLogger(DefaultTokenServicesFallback.class); - - @Override - public OAuth2AccessToken createAccessToken(final OAuth2Authentication authentication) - throws AuthenticationException { - - try { - return super.createAccessToken(authentication); - } catch (final DuplicateKeyException e) { - - log.info( - "Catched DuplicateKeyException, try to handle it by trying to get the already stored access token after waited some time"); - - final String clientName = authentication.getName(); - if (StringUtils.isNotBlank(clientName)) { - - // wait a second... - try { - Thread.sleep(1000); - } catch (final InterruptedException e1) { - log.warn("Failed to sleep: {}", e1.getMessage()); - } - - final OAuth2AccessToken accessToken = this.getAccessToken(authentication); - if (accessToken != null) { - log.info("Found original accees token for client: {} token: {}", clientName, accessToken); - return accessToken; - } - } - - // If no access token is available, propagate the original exception - log.error("Unable the handle DuplicateKeyException properly", e); - throw e; - } - } - -} +/* + * Copyright (c) 2019 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.webservice.weblayer.oauth; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.dao.DuplicateKeyException; +import org.springframework.security.core.AuthenticationException; +import org.springframework.security.oauth2.common.OAuth2AccessToken; +import org.springframework.security.oauth2.provider.OAuth2Authentication; +import org.springframework.security.oauth2.provider.token.DefaultTokenServices; + +// TODO check if we can apply some caching here to get better performance for SEB client connection attempts +public class DefaultTokenServicesFallback extends DefaultTokenServices { + + private static final Logger log = LoggerFactory.getLogger(DefaultTokenServicesFallback.class); + + @Override + public OAuth2AccessToken createAccessToken(final OAuth2Authentication authentication) + throws AuthenticationException { + + try { + return super.createAccessToken(authentication); + } catch (final DuplicateKeyException e) { + + log.warn( + "Caught DuplicateKeyException, try to handle it by trying to get the already stored access token after waited some time"); + + final String clientName = authentication.getName(); + if (StringUtils.isNotBlank(clientName)) { + + // wait some time... + try { + Thread.sleep(500); + } catch (final InterruptedException e1) { + log.warn("Failed to sleep: {}", e1.getMessage()); + } + + final OAuth2AccessToken accessToken = this.getAccessToken(authentication); + if (accessToken != null) { + log.debug("Found original access token for client: {} ", clientName); + return accessToken; + } + } + + // If no access token is available, propagate the original exception + log.error("Unable the handle DuplicateKeyException properly", e); + throw e; + } + } + +} diff --git a/src/main/resources/config/application-dev-ws.properties b/src/main/resources/config/application-dev-ws.properties index ebe238e2..5ec2c44f 100644 --- a/src/main/resources/config/application-dev-ws.properties +++ b/src/main/resources/config/application-dev-ws.properties @@ -1,71 +1,71 @@ -server.address=localhost -server.port=8090 - -logging.file=log/sebserver.log - -# data source configuration -spring.datasource.initialize=true -spring.datasource.initialization-mode=always -spring.datasource.url=jdbc:mariadb://localhost:3306/SEBServer?createDatabaseIfNotExist=true&verifyServerCertificate=false&useSSL=false&requireSSL=false -spring.datasource.driver-class-name=org.mariadb.jdbc.Driver -spring.flyway.enabled=true -spring.flyway.locations=classpath:config/sql/base,classpath:config/sql/dev -spring.flyway.baselineOnMigrate=true -spring.datasource.hikari.initializationFailTimeout=30000 -spring.datasource.hikari.connectionTimeout=30000 -spring.datasource.hikari.idleTimeout=600000 -spring.datasource.hikari.maxLifetime=1800000 - -sebserver.http.client.connect-timeout=15000 -sebserver.http.client.connection-request-timeout=10000 -sebserver.http.client.read-timeout=20000 - -# webservice configuration -sebserver.init.adminaccount.gen-on-init=false -sebserver.webservice.distributed=false -sebserver.webservice.http.scheme=http -sebserver.webservice.http.external.servername= -sebserver.webservice.http.external.port= -sebserver.webservice.http.redirect.gui=/gui - - -sebserver.webservice.api.admin.endpoint=/admin-api/v1 -sebserver.webservice.api.admin.accessTokenValiditySeconds=3600 -sebserver.webservice.api.admin.refreshTokenValiditySeconds=-1 -sebserver.webservice.api.exam.config.init.permittedProcesses=config/initialPermittedProcesses.xml -sebserver.webservice.api.exam.config.init.prohibitedProcesses=config/initialProhibitedProcesses.xml -sebserver.webservice.api.exam.update-interval=1 * * * * * -sebserver.webservice.api.exam.time-prefix=0 -sebserver.webservice.api.exam.time-suffix=0 -sebserver.webservice.api.exam.endpoint=/exam-api -sebserver.webservice.api.exam.endpoint.discovery=${sebserver.webservice.api.exam.endpoint}/discovery -sebserver.webservice.api.exam.endpoint.v1=${sebserver.webservice.api.exam.endpoint}/v1 -sebserver.webservice.api.exam.accessTokenValiditySeconds=3600 -sebserver.webservice.api.exam.event-handling-strategy=ASYNC_BATCH_STORE_STRATEGY -sebserver.webservice.api.exam.enable-indicator-cache=true -sebserver.webservice.api.pagination.maxPageSize=500 -# comma separated list of known possible OpenEdX API access token request endpoints -sebserver.webservice.lms.openedx.api.token.request.paths=/oauth2/access_token -sebserver.webservice.lms.moodle.api.token.request.paths= -sebserver.webservice.lms.address.alias=lms.mockup.com=lms.address.alias - -# NOTE: This is a temporary work-around for SEB Restriction API within Open edX SEB integration plugin to -# apply on load-balanced infrastructure or infrastructure that has several layers of cache. -# The reason for this is that the API (Open edX system) internally don't apply a resource-change that is -# done within HTTP API call immediately from an outside perspective. -# After a resource-change on the API is done, the system toggles between the old and the new resource -# while constantly calling GET. This usually happens for about a minute or two then it stabilizes on the new resource -# -# This may source on load-balancing or internally caching on Open edX side. -# To mitigate this effect the SEB Server can be configured to apply a resource-change on the -# API several times in a row to flush as match caches and reach as match as possible server instances. -# -# Since this is a brute-force method to mitigate the problem, this should only be a temporary -# work-around until a better solution on Open edX SEB integration side has been found and applied. -#sebserver.webservice.lms.openedx.seb.restriction.push-count=10 - -# actuator configuration -management.server.port=${server.port} -management.endpoints.web.base-path=/management -management.endpoints.web.exposure.include=logfile,loggers,jolokia +server.address=localhost +server.port=8090 + +logging.file=log/sebserver.log + +# data source configuration +spring.datasource.initialize=true +spring.datasource.initialization-mode=always +spring.datasource.url=jdbc:mariadb://localhost:3306/SEBServer?createDatabaseIfNotExist=true&verifyServerCertificate=false&useSSL=false&requireSSL=false +spring.datasource.driver-class-name=org.mariadb.jdbc.Driver +spring.flyway.enabled=true +spring.flyway.locations=classpath:config/sql/base,classpath:config/sql/dev +spring.flyway.baselineOnMigrate=true +spring.datasource.hikari.initializationFailTimeout=30000 +spring.datasource.hikari.connectionTimeout=30000 +spring.datasource.hikari.idleTimeout=600000 +spring.datasource.hikari.maxLifetime=1800000 + +sebserver.http.client.connect-timeout=15000 +sebserver.http.client.connection-request-timeout=10000 +sebserver.http.client.read-timeout=20000 + +# webservice configuration +sebserver.init.adminaccount.gen-on-init=false +sebserver.webservice.distributed=false +sebserver.webservice.http.scheme=http +sebserver.webservice.http.external.servername= +sebserver.webservice.http.external.port=${server.port} +sebserver.webservice.http.redirect.gui=/gui + + +sebserver.webservice.api.admin.endpoint=/admin-api/v1 +sebserver.webservice.api.admin.accessTokenValiditySeconds=3600 +sebserver.webservice.api.admin.refreshTokenValiditySeconds=-1 +sebserver.webservice.api.exam.config.init.permittedProcesses=config/initialPermittedProcesses.xml +sebserver.webservice.api.exam.config.init.prohibitedProcesses=config/initialProhibitedProcesses.xml +sebserver.webservice.api.exam.update-interval=1 * * * * * +sebserver.webservice.api.exam.time-prefix=0 +sebserver.webservice.api.exam.time-suffix=0 +sebserver.webservice.api.exam.endpoint=/exam-api +sebserver.webservice.api.exam.endpoint.discovery=${sebserver.webservice.api.exam.endpoint}/discovery +sebserver.webservice.api.exam.endpoint.v1=${sebserver.webservice.api.exam.endpoint}/v1 +sebserver.webservice.api.exam.accessTokenValiditySeconds=3600 +sebserver.webservice.api.exam.event-handling-strategy=ASYNC_BATCH_STORE_STRATEGY +sebserver.webservice.api.exam.enable-indicator-cache=true +sebserver.webservice.api.pagination.maxPageSize=500 +# comma separated list of known possible OpenEdX API access token request endpoints +sebserver.webservice.lms.openedx.api.token.request.paths=/oauth2/access_token +sebserver.webservice.lms.moodle.api.token.request.paths= +sebserver.webservice.lms.address.alias=lms.mockup.com=lms.address.alias + +# NOTE: This is a temporary work-around for SEB Restriction API within Open edX SEB integration plugin to +# apply on load-balanced infrastructure or infrastructure that has several layers of cache. +# The reason for this is that the API (Open edX system) internally don't apply a resource-change that is +# done within HTTP API call immediately from an outside perspective. +# After a resource-change on the API is done, the system toggles between the old and the new resource +# while constantly calling GET. This usually happens for about a minute or two then it stabilizes on the new resource +# +# This may source on load-balancing or internally caching on Open edX side. +# To mitigate this effect the SEB Server can be configured to apply a resource-change on the +# API several times in a row to flush as match caches and reach as match as possible server instances. +# +# Since this is a brute-force method to mitigate the problem, this should only be a temporary +# work-around until a better solution on Open edX SEB integration side has been found and applied. +#sebserver.webservice.lms.openedx.seb.restriction.push-count=10 + +# actuator configuration +management.server.port=${server.port} +management.endpoints.web.base-path=/management +management.endpoints.web.exposure.include=logfile,loggers,jolokia management.endpoints.web.path-mapping.jolokia=jmx \ No newline at end of file diff --git a/src/main/resources/config/sql/base/V2__insert_data_v1_0.sql b/src/main/resources/config/sql/base/V2__insert_data_v1_0.sql index a84e0b6d..b0db0d93 100644 --- a/src/main/resources/config/sql/base/V2__insert_data_v1_0.sql +++ b/src/main/resources/config/sql/base/V2__insert_data_v1_0.sql @@ -175,7 +175,7 @@ INSERT INTO configuration_attribute VALUES (304, 'enablePrivateClipboard', 'CHECKBOX', null, null, null, null, 'true'), (305, 'enableLogging', 'CHECKBOX', null, null, null, null, 'false'), (306, 'logDirectoryWin', 'TEXT_FIELD', null, null, null, null, ''), - (307, 'logDirectoryOSX', 'TEXT_FIELD', null, null, null, null, 'NSTemporaryDirectory'), + (307, 'logDirectoryOSX', 'TEXT_FIELD', null, null, null, null, 'F'), (308, 'minMacOSVersion', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7', null, null, '0'), (309, 'enableAppSwitcherCheck', 'CHECKBOX', null, null, null, null, 'true'), (310, 'forceAppFolderInstall', 'CHECKBOX', null, null, null, null, 'true'), @@ -263,7 +263,7 @@ INSERT INTO configuration_attribute VALUES (927, 'mobileStatusBarAppearanceExtended', 'SINGLE_SELECTION', null, '0,1,2,3,4', null, null, '1'), (928, 'newBrowserWindowShowURL', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '1'), (929, 'pinEmbeddedCertificates', 'CHECKBOX', null, null, null, null, 'false'), - (930, 'sendBrowserExamKey', 'CHECKBOX', null, null, null, null, 'false'), + (930, 'sendBrowserExamKey', 'CHECKBOX', null, null, null, null, 'true'), (931, 'showNavigationButtons', 'CHECKBOX', null, null, null, null, 'false'), (932, 'showScanQRCodeButton', 'CHECKBOX', null, null, null, null, 'false'), (933, 'startResource', 'TEXT_FIELD', null, null, null, null, ''), diff --git a/src/test/resources/data-test-additional.sql b/src/test/resources/data-test-additional.sql index fb30ed4b..814671c3 100644 --- a/src/test/resources/data-test-additional.sql +++ b/src/test/resources/data-test-additional.sql @@ -1,866 +1,866 @@ -INSERT IGNORE INTO lms_setup VALUES - (1, 1, 'test', 'MOCKUP', 'http://', 'ccdfa2330533ed6c316a8ffbd64a3197d4a79956ac7ee4c1162f7bdb1a27234fe8793615a51074351e', '8d14b78ecdcbec1d010d414a7208dbe5c411f1fa735c35c7427d840453093a3730d1bc0abe13b9b1a8', null, null, null, null, null, 1) - ; - -INSERT IGNORE INTO seb_client_configuration VALUES - (1, 1, 'test', '2019-07-02 09:22:50', 'test', '98ac3c953abf5948d9d13c81cab580819ee2624c76d6d4147d4896a5b79f49956d382c08c93cb3b9ae350b32', null, 1) - ; - -INSERT IGNORE INTO exam VALUES - (1, 1, 1, 'quiz1', 'super-admin', 'super-admin', 'MANAGED', null, null, 'UP_COMING', 1, 0, null, 1), - (2, 1, 1, 'quiz6', 'super-admin', 'super-admin', 'MANAGED', null, null, 'RUNNING', 1, 0, null, 1) - ; - -INSERT IGNORE INTO indicator VALUES - (1, 2, 'LAST_PING', 'Ping', 'dcdcdc') - ; - -INSERT IGNORE INTO threshold VALUES - (1, 1, 1000.0000, '22b14c'), - (2, 1, 2000.0000, 'ff7e00'), - (3, 1, 5000.0000, 'ed1c24') - ; - -INSERT IGNORE INTO view VALUES - (1, 'general', 4, 1, 0), - (2, 'user_interface', 12, 2, 0), - (3, 'browser', 12, 3, 0), - (4, 'down_upload', 12, 4, 0), - (5, 'exam', 12, 5, 0), - (6, 'applications', 12, 6, 0), - (7, 'resources', 12, 7, 0), - (8, 'network', 12, 8, 0), - (9, 'security', 12, 9, 0), - (10, 'registry', 12, 10, 0), - (11, 'hooked_keys', 12, 11, 0); - -INSERT IGNORE INTO configuration_attribute VALUES - (1, 'hashedAdminPassword', 'PASSWORD_FIELD', null, null, null, null, null), - (2, 'allowQuit', 'CHECKBOX', null, null, null, null, 'true'), - (3, 'ignoreExitKeys', 'CHECKBOX', null, null, null, null, 'false'), - (4, 'hashedQuitPassword', 'PASSWORD_FIELD', null, null, null, null, null), - (5, 'exitKey1', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '2'), - (6, 'exitKey2', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '10'), - (7, 'exitKey3', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '5'), - - (8, 'browserViewMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), - (9, 'enableTouchExit', 'CHECKBOX', null, null, null, null, 'false'), - (10, 'mainBrowserWindowWidth', 'COMBO_SELECTION', null, '50%,100%,800,1000', 'WindowsSizeValidator', null, '100%'), - (11, 'mainBrowserWindowHeight', 'COMBO_SELECTION', null, '80%,100%,600,800', 'WindowsSizeValidator', null, '100%'), - (12, 'mainBrowserWindowPositioning', 'SINGLE_SELECTION', null, '0,1,2', null, null, '1'), - (13, 'enableBrowserWindowToolbar', 'CHECKBOX', null, null, null, null, 'false'), - (14, 'hideBrowserWindowToolbar', 'CHECKBOX', null, null, null, null, 'false'), - (15, 'showMenuBar', 'CHECKBOX', null, null, null, null, 'false'), - (16, 'showTaskBar', 'CHECKBOX', null, null, null, null, 'true'), - (17, 'taskBarHeight', 'COMBO_SELECTION', null, '40,60,80', 'IntegerTypeValidator', null, '40'), - (18, 'showReloadButton', 'CHECKBOX', null, null, null, null, 'true'), - (19, 'showTime', 'CHECKBOX', null, null, null, null, 'true'), - (20, 'showInputLanguage', 'CHECKBOX', null, null, null, null, 'false'), - (21, 'enableZoomPage', 'CHECKBOX', null, null, null, null, 'true'), - (22, 'enableZoomText', 'CHECKBOX', null, null, null, null, 'true'), - (23, 'zoomMode', 'RADIO_SELECTION', null, '0,1', null, null, '0'), - (24, 'audioControlEnabled', 'CHECKBOX', null, null, null, null, 'false'), - (25, 'audioMute', 'CHECKBOX', null, null, null, null, 'false'), - (26, 'audioSetVolumeLevel', 'CHECKBOX', null, null, null, null, 'false'), - (27, 'audioVolumeLevel', 'SLIDER', null, '0,100', null, null, '25'), - (28, 'allowSpellCheck', 'CHECKBOX', null, null, null, null, 'false'), - (29, 'allowDictionaryLookup', 'CHECKBOX', null, null, null, null, 'false'), - (30, 'allowSpellCheckDictionary', 'MULTI_CHECKBOX_SELECTION', null, 'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI', null, null, 'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), - - (31, 'newBrowserWindowByLinkPolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '2'), - (32, 'newBrowserWindowByLinkBlockForeign', 'CHECKBOX', null, null, null, null, 'false'), - (33, 'newBrowserWindowByLinkWidth', 'COMBO_SELECTION', null, '50%,100%,800,1000', 'WindowsSizeValidator', null, '100%'), - (34, 'newBrowserWindowByLinkHeight', 'COMBO_SELECTION', null, '80%,100%,600,800', 'WindowsSizeValidator', null, '100%'), - (35, 'newBrowserWindowByLinkPositioning', 'SINGLE_SELECTION', null, '0,1,2', null, null, '2'), - (36, 'enablePlugIns', 'CHECKBOX', null, null, null, null, 'true'), - (37, 'enableJavaScript', 'CHECKBOX', null, null, null, null, 'true'), - (38, 'enableJava', 'CHECKBOX', null, null, null, null, 'false'), - (39, 'blockPopUpWindows', 'CHECKBOX', null, null, null, null, 'false'), - (40, 'allowVideoCapture', 'CHECKBOX', null, null, null, null, 'false'), - (41, 'allowAudioCapture', 'CHECKBOX', null, null, null, null, 'false'), - (42, 'allowBrowsingBackForward', 'CHECKBOX', null, null, null, null, 'false'), - (43, 'newBrowserWindowNavigation', 'CHECKBOX', null, null, null, null, 'true'), - (44, 'browserWindowAllowReload', 'CHECKBOX', null, null, null, null, 'true'), - (45, 'newBrowserWindowAllowReload', 'CHECKBOX', null, null, null, null, 'true'), - (46, 'showReloadWarning', 'CHECKBOX', null, null, null, null, 'true'), - (47, 'newBrowserWindowShowReloadWarning', 'CHECKBOX', null, null, null, null, 'false'), - (48, 'removeBrowserProfile', 'CHECKBOX', null, null, null, null, 'false'), - (49, 'removeLocalStorage', 'CHECKBOX', null, null, null, null, 'false'), - (50, 'browserUserAgent', 'TEXT_FIELD', null, null, null, null, null), - (51, 'browserUserAgentWinDesktopMode', 'RADIO_SELECTION', null, '0,1', null, null, '0'), - (52, 'browserUserAgentWinDesktopModeCustom', 'TEXT_FIELD', null, null, null, null, null), - (53, 'browserUserAgentWinTouchMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), - (54, 'browserUserAgentWinTouchModeCustom', 'TEXT_FIELD', null, null, null, null, null), - (55, 'browserUserAgentMac', 'RADIO_SELECTION', null, '0,1', null, null, '0'), - (56, 'browserUserAgentMacCustom', 'TEXT_FIELD', null, null, null, null, null), - (57, 'enableSebBrowser', 'CHECKBOX', null, null, null, null, 'true'), - (58, 'browserWindowTitleSuffix', 'TEXT_FIELD', null, null, null, null, null), - - (59, 'allowDownUploads', 'CHECKBOX', null, null, null, null, 'true'), - (60, 'downloadDirectoryWin', 'TEXT_FIELD', null, null, null, null, null), - (61, 'downloadDirectoryOSX', 'TEXT_FIELD', null, null, null, null, null), - (62, 'openDownloads', 'CHECKBOX', null, null, null, null, 'false'), - (63, 'chooseFileToUploadPolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), - (64, 'downloadPDFFiles', 'CHECKBOX', null, null, null, null, 'true'), - (65, 'allowPDFPlugIn', 'CHECKBOX', null, null, null, null, 'true'), - (66, 'downloadAndOpenSebConfig', 'CHECKBOX', null, null, null, null, 'true'), - - (67, 'quitURL', 'TEXT_FIELD', null, null, null, null, null), - (68, 'quitURLConfirm', 'CHECKBOX', null, null, null, null, 'true'), - (69, 'restartExamUseStartURL', 'CHECKBOX', null, null, null, null, 'false'), - (70, 'restartExamURL', 'TEXT_FIELD', null, null, null, null, null), - (71, 'restartExamText', 'TEXT_FIELD', null, null, null, null, null), - (72, 'restartExamPasswordProtected', 'CHECKBOX', null, null, null, null, 'true'), - - (73, 'permittedProcesses', 'TABLE', null, null, null, null, null), - (74, 'permittedProcesses.active', 'CHECKBOX', 73, null, null, null, 'true'), - (75, 'permittedProcesses.os', 'SINGLE_SELECTION', 73, '0,1', null, null, '1'), - (76, 'permittedProcesses.title', 'TEXT_FIELD', 73, null, null, null, ''), - (77, 'permittedProcesses.description', 'TEXT_FIELD', 73, null, null, null, ''), - (78, 'permittedProcesses.executable', 'TEXT_FIELD', 73, null, null, null, ''), - (79, 'permittedProcesses.originalName', 'TEXT_FIELD', 73, null, null, null, ''), - (80, 'permittedProcesses.allowedExecutables', 'TEXT_FIELD', 73, null, null, null, ''), - (81, 'permittedProcesses.path', 'TEXT_FIELD', 73, null, null, null, ''), - (82, 'permittedProcesses.arguments', 'INLINE_TABLE', 73, '1:active:CHECKBOX|4:argument:TEXT_FIELD', null, null, null), - (85, 'permittedProcesses.identifier', 'TEXT_FIELD', 73, null, null, null, ''), - (86, 'permittedProcesses.iconInTaskbar', 'CHECKBOX', 73, null, null, null, 'true'), - (87, 'permittedProcesses.autostart', 'CHECKBOX', 73, null, null, null, 'false'), - (88, 'permittedProcesses.runInBackground', 'CHECKBOX', 73, null, null, null, 'false'), - (89, 'permittedProcesses.allowUserToChooseApp', 'CHECKBOX', 73, null, null, null, 'false'), - (90, 'permittedProcesses.strongKill', 'CHECKBOX', 73, null, null, null, 'false'), - (91, 'allowSwitchToApplications', 'CHECKBOX', null, null, null, null, 'false'), - (92, 'allowFlashFullscreen', 'CHECKBOX', null, null, null, null, 'false'), - - (93, 'prohibitedProcesses', 'TABLE', null, null, null, null, null), - (94, 'prohibitedProcesses.active', 'CHECKBOX', 93, null, null, null, 'true'), - (95, 'prohibitedProcesses.os', 'SINGLE_SELECTION', 93, '0,1', null, null, '1'), - (96, 'prohibitedProcesses.executable', 'TEXT_FIELD', 93, null, null, null, ''), - (97, 'prohibitedProcesses.description', 'TEXT_FIELD', 93, null, null, null, ''), - (98, 'prohibitedProcesses.originalName', 'TEXT_FIELD', 93, null, null, null, ''), - (99, 'prohibitedProcesses.identifier', 'TEXT_FIELD', 93, null, null, null, ''), - (100, 'prohibitedProcesses.strongKill', 'CHECKBOX', 93, null, null, null, 'false'), - (101, 'prohibitedProcesses.currentUser', 'CHECKBOX', 93, null, null, null, 'false'), - (102, 'prohibitedProcesses.user', 'TEXT_FIELD', 93, null, null, null, null), - - (200, 'URLFilterEnable', 'CHECKBOX', null, null, null, null, 'false'), - (201, 'URLFilterEnableContentFilter', 'CHECKBOX', null, null, null, null, 'false'), - (202, 'URLFilterRules', 'TABLE', null, null, null, null, null), - (203, 'URLFilterRules.active', 'CHECKBOX', 202, null, null, null, 'true'), - (204, 'URLFilterRules.regex', 'CHECKBOX', 202, null, null, null, 'false'), - (205, 'URLFilterRules.expression', 'TEXT_FIELD', 202, null, null, null, ''), - (206, 'URLFilterRules.action', 'SINGLE_SELECTION', 202, '0,1', null, null, '0'), - - (210, 'proxySettingsPolicy', 'RADIO_SELECTION', null, '0,1', null, null, '0'), - (220, 'proxies', 'COMPOSITE_TABLE', null, 'active,TABLE_ENTRY|autoDiscovery,autoConfiguration,http,https,ftp,socks,rtsp', null, null, null), - (221, 'ExcludeSimpleHostnames', 'CHECKBOX', 220, null, null, 'showInView=true,createDefaultValue=true', 'false'), - (222, 'ExceptionsList', 'TEXT_AREA', 220, null, null, 'showInView=true,createDefaultValue=true', null), - (223, 'FTPPassive', 'CHECKBOX', 220, null, null, 'showInView=true,createDefaultValue=true', 'true'), - (231, 'AutoDiscoveryEnabled', 'CHECKBOX', 220, null, null, 'groupId=autoDiscovery,createDefaultValue=true', 'false'), - (233, 'AutoConfigurationEnabled', 'CHECKBOX', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', 'false'), - (234, 'AutoConfigurationURL', 'TEXT_FIELD', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', null), - (235, 'AutoConfigurationJavaScript', 'TEXT_AREA', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', null), - (236, 'HTTPEnable', 'CHECKBOX', 220, null, null, 'groupId=http,createDefaultValue=true', 'false'), - (237, 'HTTPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), - (238, 'HTTPPort', 'INTEGER', 220, null, null, 'groupId=http,createDefaultValue=true', '80'), - (239, 'HTTPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=http,createDefaultValue=true', 'false'), - (240, 'HTTPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), - (241, 'HTTPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), - (242, 'HTTPSEnable', 'CHECKBOX', 220, null, null, 'groupId=https,createDefaultValue=true', 'false'), - (243, 'HTTPSProxy', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), - (244, 'HTTPSPort', 'INTEGER', 220, null, null, 'groupId=https,createDefaultValue=true', '443'), - (245, 'HTTPSRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=https,createDefaultValue=true', 'false'), - (246, 'HTTPSUsername', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), - (247, 'HTTPSPassword', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), - (248, 'FTPEnable', 'CHECKBOX', 220, null, null, 'groupId=ftp,createDefaultValue=true', 'false'), - (249, 'FTPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), - (250, 'FTPPort', 'INTEGER', 220, null, null, 'groupId=ftp,createDefaultValue=true', '21'), - (251, 'FTPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=ftp,createDefaultValue=true', 'false'), - (252, 'FTPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), - (253, 'FTPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), - (254, 'SOCKSEnable', 'CHECKBOX', 220, null, null, 'groupId=socks,createDefaultValue=true', 'false'), - (255, 'SOCKSProxy', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), - (256, 'SOCKSPort', 'INTEGER', 220, null, null, 'groupId=socks,createDefaultValue=true', '1080'), - (257, 'SOCKSRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=socks,createDefaultValue=true', 'false'), - (258, 'SOCKSUsername', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), - (259, 'SOCKSPassword', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), - (260, 'RTSPEnable', 'CHECKBOX', 220, null, null, 'groupId=rtsp,createDefaultValue=true', 'false'), - (261, 'RTSPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), - (262, 'RTSPPort', 'INTEGER', 220, null, null, 'groupId=rtsp,createDefaultValue=true', '554'), - (263, 'RTSPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=rtsp,createDefaultValue=true', 'false'), - (264, 'RTSPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), - (265, 'RTSPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), - - - (300, 'sebServicePolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '2'), - (301, 'kioskMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), - (302, 'allowVirtualMachine', 'CHECKBOX', null, null, null, null, 'false'), - (303, 'allowScreenSharing', 'CHECKBOX', null, null, null, null, 'false'), - (304, 'enablePrivateClipboard', 'CHECKBOX', null, null, null, null, 'true'), - (305, 'enableLogging', 'CHECKBOX', null, null, null, null, 'false'), - (306, 'logDirectoryWin', 'TEXT_FIELD', null, null, null, null, ''), - (307, 'logDirectoryOSX', 'TEXT_FIELD', null, null, null, null, 'NSTemporaryDirectory'), - (308, 'minMacOSVersion', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7', null, null, '0'), - (309, 'enableAppSwitcherCheck', 'CHECKBOX', null, null, null, null, 'true'), - (310, 'forceAppFolderInstall', 'CHECKBOX', null, null, null, null, 'true'), - (311, 'allowUserAppFolderInstall', 'CHECKBOX', null, null, null, null, 'false'), - (312, 'allowSiri', 'CHECKBOX', null, null, null, null, 'false'), - (313, 'detectStoppedProcess', 'CHECKBOX', null, null, null, null, 'true'), - (314, 'allowDisplayMirroring', 'CHECKBOX', null, null, null, null, 'false'), - (315, 'allowedDisplaysMaxNumber', 'COMBO_SELECTION', null, '1,2,3', null, null, '1'), - (316, 'allowedDisplayBuiltin', 'CHECKBOX', null, null, null, null, 'true'), - (317, 'logLevel', 'SINGLE_SELECTION', null, '0,1,2,3,4', null, null, '1'), - - (400, 'insideSebEnableSwitchUser', 'CHECKBOX', null, null, null, null, 'false'), - (401, 'insideSebEnableLockThisComputer', 'CHECKBOX', null, null, null, null, 'false'), - (402, 'insideSebEnableChangeAPassword', 'CHECKBOX', null, null, null, null, 'false'), - (403, 'insideSebEnableStartTaskManager', 'CHECKBOX', null, null, null, null, 'false'), - (404, 'insideSebEnableLogOff', 'CHECKBOX', null, null, null, null, 'false'), - (405, 'insideSebEnableShutDown', 'CHECKBOX', null, null, null, null, 'false'), - (406, 'insideSebEnableVmWareClientShade', 'CHECKBOX', null, null, null, null, 'false'), - (407, 'insideSebEnableEaseOfAccess', 'CHECKBOX', null, null, null, null, 'false'), - (408, 'insideSebEnableNetworkConnectionSelector', 'CHECKBOX', null, null, null, null, 'false'), - - (500, 'enableEsc', 'CHECKBOX', null, null, null, null, 'false'), - (501, 'enablePrintScreen', 'CHECKBOX', null, null, null, null, 'false'), - (502, 'enableCtrlEsc', 'CHECKBOX', null, null, null, null, 'false'), - (503, 'enableAltEsc', 'CHECKBOX', null, null, null, null, 'false'), - (504, 'enableAltTab', 'CHECKBOX', null, null, null, null, 'true'), - (505, 'enableAltF4', 'CHECKBOX', null, null, null, null, 'false'), - (506, 'enableStartMenu', 'CHECKBOX', null, null, null, null, 'false'), - (507, 'enableRightMouse', 'CHECKBOX', null, null, null, null, 'false'), - (508, 'enableAltMouseWheel', 'CHECKBOX', null, null, null, null, 'false'), - - (509, 'enableF1', 'CHECKBOX', null, null, null, null, 'false'), - (510, 'enableF2', 'CHECKBOX', null, null, null, null, 'false'), - (511, 'enableF3', 'CHECKBOX', null, null, null, null, 'false'), - (512, 'enableF4', 'CHECKBOX', null, null, null, null, 'false'), - (513, 'enableF5', 'CHECKBOX', null, null, null, null, 'false'), - (514, 'enableF6', 'CHECKBOX', null, null, null, null, 'false'), - (515, 'enableF7', 'CHECKBOX', null, null, null, null, 'false'), - (516, 'enableF8', 'CHECKBOX', null, null, null, null, 'false'), - (517, 'enableF9', 'CHECKBOX', null, null, null, null, 'false'), - (518, 'enableF10', 'CHECKBOX', null, null, null, null, 'false'), - (519, 'enableF11', 'CHECKBOX', null, null, null, null, 'false'), - (520, 'enableF12', 'CHECKBOX', null, null, null, null, 'false'), - - (800, 'browserMessagingSocket', 'TEXT_FIELD', null, null, null, null, 'ws://localhost:8706'), - (801, 'browserMessagingPingTime', 'INTEGER', null, null, null, null, '120000'), - (802, 'allowPreferencesWindow', 'CHECKBOX', null, null, null, null, 'true'), - (803, 'useAsymmetricOnlyEncryption', 'CHECKBOX', null, null, null, null, 'false'), - (804, 'touchOptimized', 'CHECKBOX', null, null, null, null, 'false'), - (805, 'browserScreenKeyboard', 'CHECKBOX', null, null, null, null, 'false'), - (806, 'newBrowserWindowByScriptPolicy', 'INTEGER', null, null, null, null, '2'), - (807, 'newBrowserWindowByScriptBlockForeign', 'CHECKBOX', null, null, null, null, 'false'), - (808, 'monitorProcesses', 'CHECKBOX', null, null, null, null, 'false'), - (809, 'blacklistURLFilter', 'TEXT_FIELD', null, null, null, null, ''), - (810, 'whitelistURLFilter', 'TEXT_FIELD', null, null, null, null, ''), - (812, 'allowWlan', 'CHECKBOX', null, null, null, null, 'false'), - (813, 'hookKeys', 'CHECKBOX', null, null, null, null, 'true'), - - (900, 'examSessionClearCookiesOnEnd', 'CHECKBOX', null, null, null, null, 'true'), - (901, 'examSessionClearCookiesOnStart', 'CHECKBOX', null, null, null, null, 'true'), - (902, 'showBackToStartButton', 'CHECKBOX', null, null, null, null, 'true'), - (903, 'showSettingsInApp', 'CHECKBOX', null, null, null, null, 'false'), - (904, 'browserUserAgentWinTouchModeIPad', 'TEXT_FIELD', null, null, null, null, 'Mozilla/5.0 (iPad; CPU OS 12_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/12.1.2 Mobile/15E148 Safari/604.1'), - (905, 'mobileAllowPictureInPictureMediaPlayback', 'CHECKBOX', null, null, null, null, 'false'), - (906, 'lockOnMessageSocketClose', 'CHECKBOX', null, null, null, null, 'false'), - (907, 'enableDrawingEditor', 'CHECKBOX', null, null, null, null, 'false'), - (908, 'URLFilterMessage', 'RADIO_SELECTION', null, '0,1', null, null, '0'), - (909, 'allowDictation', 'CHECKBOX', null, null, null, null, 'false'), - (910, 'allowUserSwitching', 'CHECKBOX', null, null, null, null, 'false'), - (911, 'allowiOSBetaVersionNumber', 'SINGLE_SELECTION', null, '0,12', null, null, '0'), - (912, 'allowiOSVersionNumberMajor', 'SINGLE_SELECTION', null, '9,10,11,12', null, null, '9'), - (913, 'allowiOSVersionNumberMinor', 'INTEGER', null, null, null, null, '3'), - (914, 'allowiOSVersionNumberPatch', 'INTEGER', null, null, null, null, '5'), - (915, 'browserURLSalt', 'CHECKBOX', null, null, null, null, 'true'), - (917, 'browserUserAgentiOS', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '0'), - (918, 'browserUserAgentiOSCustom', 'TEXT_FIELD', null, null, null, null, ''), - (919, 'browserWindowShowURL', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '0'), - (920, 'mobileAllowQRCodeConfig', 'CHECKBOX', null, null, null, null, 'false'), - (921, 'mobileAllowSingleAppMode', 'CHECKBOX', null, null, null, null, 'false'), - (922, 'mobileEnableASAM', 'CHECKBOX', null, null, null, null, 'true'), - (923, 'mobileEnableGuidedAccessLinkTransform', 'CHECKBOX', null, null, null, null, 'false'), - (924, 'mobilePreventAutoLock', 'CHECKBOX', null, null, null, null, 'true'), - (925, 'mobileShowSettings', 'CHECKBOX', null, null, null, null, 'false'), - (926, 'mobileStatusBarAppearance', 'SINGLE_SELECTION', null, '0,1,2', null, null, '1'), - (927, 'mobileStatusBarAppearanceExtended', 'SINGLE_SELECTION', null, '0,1,2,3,4', null, null, '1'), - (928, 'newBrowserWindowShowURL', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '1'), - (929, 'pinEmbeddedCertificates', 'CHECKBOX', null, null, null, null, 'false'), - (930, 'sendBrowserExamKey', 'CHECKBOX', null, null, null, null, 'false'), - (931, 'showNavigationButtons', 'CHECKBOX', null, null, null, null, 'false'), - (932, 'showScanQRCodeButton', 'CHECKBOX', null, null, null, null, 'false'), - (933, 'startResource', 'TEXT_FIELD', null, null, null, null, ''), - - (1000, 'originatorVersion', 'TEXT_FIELD', null, null, null, null, 'SEB_Server_0.3.0'), - (1001, 'sebConfigPurpose', 'RADIO_SELECTION', null, '0,1', null, null, '0') - - ; - -INSERT IGNORE INTO orientation VALUES - (1, 1, 0, 1, null, 1, 1, 1, 2, 'LEFT'), - (2, 2, 0, 1, null, 1, 3, 1, 1, 'LEFT'), - (3, 3, 0, 1, null, 1, 4, 1, 1, 'LEFT'), - (4, 4, 0, 1, null, 1, 5, 1, 2, 'LEFT'), - (5, 5, 0, 1, 'exitSequence', 2, 1, 1, 1, 'NONE'), - (6, 6, 0, 1, 'exitSequence', 2, 2, 1, 1, 'NONE'), - (7, 7, 0, 1, 'exitSequence', 2, 3, 1, 1, 'NONE'), - - (8, 8, 0, 2, 'browserViewMode', 0, 0, 3, 3, 'NONE'), - (9, 9, 0, 2, 'browserViewMode', 3, 2, 4, 1, 'NONE'), - (10, 10, 0, 2, 'winsize', 1, 4, 2, 1, 'LEFT'), - (11, 11, 0, 2, 'winsize', 1, 5, 2, 1, 'LEFT'), - (12, 12, 0, 2, 'winsize', 5, 4, 2, 1, 'LEFT_SPAN'), - (13, 13, 0, 2, 'wintoolbar', 0, 6, 3, 1, 'NONE'), - (14, 14, 0, 2, 'wintoolbar', 3, 6, 4, 1, 'NONE'), - (15, 15, 0, 2, 'wintoolbar', 0, 7, 3, 1, 'NONE'), - (16, 16, 0, 2, 'taskbar', 0, 9, 3, 1, 'NONE'), - (17, 17, 0, 2, 'taskbar', 5, 9, 2, 1, 'LEFT_SPAN'), - (18, 18, 0, 2, 'taskbar', 0, 10, 3, 1, 'NONE'), - (19, 19, 0, 2, 'taskbar', 0, 11, 3, 1, 'NONE'), - (20, 20, 0, 2, 'taskbar', 0, 12, 3, 1, 'NONE'), - (21, 21, 0, 2, 'zoom', 0, 14, 3, 1, 'NONE'), - (22, 22, 0, 2, 'zoom', 0, 15, 3, 1, 'NONE'), - (23, 23, 0, 2, 'zoomMode', 3, 14, 4, 1, 'NONE'), - (24, 24, 0, 2, 'audio', 7, 0, 5, 1, 'NONE'), - (25, 25, 0, 2, 'audio', 7, 1, 5, 1, 'NONE'), - (26, 26, 0, 2, 'audio', 7, 2, 5, 1, 'NONE'), - (27, 27, 0, 2, 'audio', 7, 3, 5, 1, 'NONE'), - (28, 28, 0, 2, 'spellcheck', 7, 4, 5, 1, 'NONE'), - (29, 29, 0, 2, 'spellcheck', 7, 5, 5, 1, 'NONE'), - (30, 30, 0, 2, 'spellcheck', 7, 7, 5, 9, 'TOP'), - - (31, 31, 0, 3, 'newBrowserWindow', 0, 0, 3, 3, 'NONE'), - (32, 32, 0, 3, 'newBrowserWindow', 4, 0, 3, 1, 'NONE'), - (33, 33, 0, 3, 'newwinsize', 1, 4, 2, 1, 'LEFT'), - (34, 34, 0, 3, 'newwinsize', 1, 5, 2, 1, 'LEFT'), - (35, 35, 0, 3, 'newwinsize', 5, 4, 2, 1, 'LEFT_SPAN'), - (36, 36, 0, 3, 'browserSecurity', 0, 5, 4, 1, 'NONE'), - (37, 37, 0, 3, 'browserSecurity', 4, 5, 3, 1, 'NONE'), - (38, 38, 0, 3, 'browserSecurity', 0, 6, 4, 1, 'NONE'), - (39, 39, 0, 3, 'browserSecurity', 4, 6, 3, 1, 'NONE'), - (40, 40, 0, 3, 'browserSecurity', 0, 7, 4, 1, 'NONE'), - (41, 41, 0, 3, 'browserSecurity', 4, 7, 3, 1, 'NONE'), - (42, 42, 0, 3, 'browserSecurity', 0, 8, 4, 1, 'NONE'), - (43, 43, 0, 3, 'browserSecurity', 4, 8, 3, 1, 'NONE'), - (44, 44, 0, 3, 'browserSecurity', 0, 9, 4, 1, 'NONE'), - (45, 45, 0, 3, 'browserSecurity', 4, 9, 3, 1, 'NONE'), - (46, 46, 0, 3, 'browserSecurity', 0, 10, 4, 1, 'NONE'), - (47, 47, 0, 3, 'browserSecurity', 4, 10, 3, 1, 'NONE'), - (48, 48, 0, 3, 'browserSecurity', 0, 11, 4, 1, 'NONE'), - (49, 49, 0, 3, 'browserSecurity', 4, 11, 3, 1, 'NONE'), - - (50, 50, 0, 3, null, 7, 1, 5, 1, 'TOP'), - (51, 51, 0, 3, 'userAgentDesktop', 7, 2, 5, 2, 'NONE'), - (52, 52, 0, 3, 'userAgentDesktop', 7, 3, 5, 1, 'NONE'), - (53, 53, 0, 3, 'userAgentTouch', 7, 4, 5, 2, 'NONE'), - (54, 54, 0, 3, 'userAgentTouch', 7, 6, 5, 1, 'NONE'), - (55, 55, 0, 3, 'userAgentMac', 7, 8, 5, 2, 'NONE'), - (56, 56, 0, 3, 'userAgentMac', 7, 10, 5, 1, 'NONE'), - (57, 57, 0, 3, null, 0, 13, 6, 1, 'NONE'), - (58, 58, 0, 3, null, 7, 13, 5, 1, 'TOP'), - - (59, 59, 0, 4, null, 0, 0, 8, 1, 'NONE'), - (60, 60, 0, 4, null, 3, 1, 5, 1, 'LEFT_SPAN'), - (61, 61, 0, 4, null, 3, 2, 5, 1, 'LEFT_SPAN'), - (62, 62, 0, 4, null, 0, 3, 8, 1, 'NONE'), - (63, 63, 0, 4, null, 0, 5, 8, 2, 'TOP'), - (64, 64, 0, 4, null, 0, 8, 8, 1, 'NONE'), - (65, 65, 0, 4, null, 0, 9, 8, 1, 'NONE'), - (66, 66, 0, 4, null, 0, 10, 8, 1, 'NONE'), - - (67, 67, 0, 5, 'quitLink', 0, 1, 8, 1, 'TOP'), - (68, 68, 0, 5, 'quitLink', 0, 2, 8, 1, 'NONE'), - (69, 69, 0, 5, 'backToStart', 0, 4, 8, 1, 'NONE'), - (70, 70, 0, 5, 'backToStart', 0, 6, 8, 2, 'TOP'), - (71, 71, 0, 5, 'backToStart', 0, 8, 8, 2, 'TOP'), - (72, 72, 0, 5, 'backToStart', 0, 10, 8, 1, 'NONE'), - - (73, 73, 0, 6, null, 0, 2, 10, 6, 'TOP'), - (74, 74, 0, 6, null, 1, 1, 1, 1, 'LEFT'), - (75, 75, 0, 6, null, 2, 2, 1, 1, 'LEFT'), - (76, 76, 0, 6, null, 4, 4, 2, 1, 'LEFT'), - (77, 77, 0, 6, null, 0, 3, 1, 1, 'LEFT'), - (78, 78, 0, 6, null, 3, 4, 4, 1, 'LEFT'), - (79, 79, 0, 6, null, 0, 5, 1, 1, 'LEFT'), - (80, 80, 0, 6, null, 0, 6, 1, 1, 'LEFT'), - (81, 81, 0, 6, null, 0, 7, 1, 1, 'LEFT'), - - (82, 82, 0, 6, null, 0, 8, 1, 3, 'LEFT'), - - (85, 85, 0, 6, null, 0, 8, 1, 1, 'LEFT'), - (86, 86, 0, 6, null, 0, 7, 1, 1, 'LEFT'), - (87, 87, 0, 6, null, 0, 9, 1, 1, 'LEFT'), - (88, 88, 0, 6, null, 0, 10, 1, 1, 'LEFT'), - (89, 89, 0, 6, null, 0, 11, 1, 1, 'LEFT'), - (90, 90, 0, 6, null, 0, 12, 1, 1, 'LEFT'), - (91, 91, 0, 6, null, 0, 0, 5, 1, 'NONE'), - (92, 92, 0, 6, null, 5, 0, 5, 1, 'NONE'), - (93, 93, 0, 6, null, 0, 10, 10, 6, 'TOP'), - (94, 94, 0, 6, null, 1, 1, 1, 1, 'LEFT'), - (95, 95, 0, 6, null, 2, 2, 1, 1, 'LEFT'), - (96, 96, 0, 6, null, 3, 3, 4, 1, 'LEFT'), - (97, 97, 0, 6, null, 4, 5, 2, 1, 'LEFT'), - (98, 98, 0, 6, null, 0, 4, 1, 1, 'LEFT'), - (99, 99, 0, 6, null, 0, 6, 1, 1, 'LEFT'), - (100, 100, 0, 6, null, 0, 7, 1, 1, 'LEFT'), - - (200, 200, 0, 8, 'urlFilter', 0, 0, 3, 1, 'NONE'), - (201, 201, 0, 8, 'urlFilter', 3, 0, 4, 1, 'NONE'), - (202, 202, 0, 8, 'urlFilter', 0, 1, 12, 6, 'NONE'), - (203, 203, 0, 8, 'urlFilter', 1, 1, 1, 1, 'LEFT'), - (204, 204, 0, 8, 'urlFilter', 2, 2, 1, 1, 'LEFT'), - (205, 205, 0, 8, 'urlFilter', 3, 3, 4, 1, 'LEFT'), - (206, 206, 0, 8, 'urlFilter', 4, 4, 2, 1, 'LEFT'), - - (210, 210, 0, 8, 'proxies', 0, 6, 5, 2, 'NONE'), - (220, 220, 0, 8, 'proxies', 7, 7, 5, 7, 'TOP'), - (221, 221, 0, 8, 'proxies', 0, 8, 6, 1, 'NONE'), - (222, 222, 0, 8, 'proxies', 0, 10, 6, 2, 'TOP'), - (223, 223, 0, 8, 'proxies', 0, 11, 6, 1, 'NONE'), - - (231, 231, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - - (233, 233, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (234, 234, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (235, 235, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (236, 236, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (237, 237, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (238, 238, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (239, 239, 0, 8, null, 0, 3, 1, 1, 'LEFT'), - (240, 240, 0, 8, null, 0, 4, 1, 1, 'LEFT'), - (241, 241, 0, 8, null, 0, 5, 1, 1, 'LEFT'), - (242, 242, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (243, 243, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (244, 244, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (245, 245, 0, 8, null, 0, 3, 1, 1, 'LEFT'), - (246, 246, 0, 8, null, 0, 4, 1, 1, 'LEFT'), - (247, 247, 0, 8, null, 0, 5, 1, 1, 'LEFT'), - (248, 248, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (249, 249, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (250, 250, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (251, 251, 0, 8, null, 0, 3, 1, 1, 'LEFT'), - (252, 252, 0, 8, null, 0, 4, 1, 1, 'LEFT'), - (253, 253, 0, 8, null, 0, 5, 1, 1, 'LEFT'), - (254, 254, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (255, 255, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (256, 256, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (257, 257, 0, 8, null, 0, 3, 1, 1, 'LEFT'), - (258, 258, 0, 8, null, 0, 4, 1, 1, 'LEFT'), - (259, 259, 0, 8, null, 0, 5, 1, 1, 'LEFT'), - (260, 260, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), - (261, 261, 0, 8, null, 0, 1, 1, 1, 'LEFT'), - (262, 262, 0, 8, null, 0, 2, 1, 1, 'LEFT'), - (263, 263, 0, 8, null, 0, 3, 1, 1, 'LEFT'), - (264, 264, 0, 8, null, 0, 4, 1, 1, 'LEFT'), - (265, 265, 0, 8, null, 0, 5, 1, 1, 'LEFT'), - - - (300, 300, 0, 9, 'servicePolicy', 0, 0, 4, 3, 'NONE'), - (301, 301, 0, 9, 'kioskMode', 4, 0, 3, 3, 'NONE'), - (302, 302, 0, 9, null, 0, 5, 4, 1, 'NONE'), - (303, 303, 0, 9, null, 0, 6, 4, 1, 'NONE'), - (304, 304, 0, 9, null, 4, 5, 3, 1, 'NONE'), - (305, 305, 0, 9, 'logging', 0, 8, 6, 1, 'NONE'), - (306, 306, 0, 9, 'logging', 3, 9, 4, 1, 'LEFT_SPAN'), - (307, 307, 0, 9, 'logging', 3, 10, 4, 1, 'LEFT_SPAN'), - (308, 308, 0, 9, 'macSettings', 7, 1, 5, 1, 'TOP'), - (309, 309, 0, 9, 'macSettings', 7, 2, 5, 1, 'NONE'), - (310, 310, 0, 9, 'macSettings', 7, 3, 5, 1, 'NONE'), - (311, 311, 0, 9, 'macSettings', 7, 4, 5, 1, 'NONE'), - (312, 312, 0, 9, 'macSettings', 7, 5, 5, 1, 'NONE'), - (313, 313, 0, 9, 'macSettings', 7, 6, 5, 1, 'NONE'), - (314, 314, 0, 9, 'macSettings', 7, 7, 5, 1, 'NONE'), - (315, 315, 0, 9, 'macSettings', 7, 9, 5, 1, 'TOP'), - (316, 316, 0, 9, 'macSettings', 7, 10, 5, 1, 'NONE'), - (317, 317, 0, 9, 'logging', 3, 11, 4, 1, 'LEFT_SPAN'), - - (400, 400, 0, 10, 'registry', 0, 1, 4, 1, 'NONE'), - (401, 401, 0, 10, 'registry', 0, 2, 4, 1, 'NONE'), - (402, 402, 0, 10, 'registry', 0, 3, 4, 1, 'NONE'), - (403, 403, 0, 10, 'registry', 0, 4, 4, 1, 'NONE'), - (404, 404, 0, 10, 'registry', 0, 5, 4, 1, 'NONE'), - (405, 405, 0, 10, 'registry', 0, 6, 4, 1, 'NONE'), - (406, 406, 0, 10, 'registry', 0, 7, 4, 1, 'NONE'), - (407, 407, 0, 10, 'registry', 0, 8, 4, 1, 'NONE'), - (408, 408, 0, 10, 'registry', 0, 9, 4, 1, 'NONE'), - - (500, 500, 0, 11, 'specialKeys', 0, 1, 3, 1, 'NONE'), - (501, 501, 0, 11, 'specialKeys', 0, 2, 3, 1, 'NONE'), - (502, 502, 0, 11, 'specialKeys', 0, 3, 3, 1, 'NONE'), - (503, 503, 0, 11, 'specialKeys', 0, 4, 3, 1, 'NONE'), - (504, 504, 0, 11, 'specialKeys', 0, 5, 3, 1, 'NONE'), - (505, 505, 0, 11, 'specialKeys', 0, 6, 3, 1, 'NONE'), - (506, 506, 0, 11, 'specialKeys', 0, 7, 3, 1, 'NONE'), - (507, 507, 0, 11, 'specialKeys', 0, 8, 3, 1, 'NONE'), - (508, 508, 0, 11, 'specialKeys', 0, 9, 3, 1, 'NONE'), - - (509, 509, 0, 11, 'functionKeys', 3, 1, 3, 1, 'NONE'), - (510, 510, 0, 11, 'functionKeys', 3, 2, 3, 1, 'NONE'), - (511, 511, 0, 11, 'functionKeys', 3, 3, 3, 1, 'NONE'), - (512, 512, 0, 11, 'functionKeys', 3, 4, 3, 1, 'NONE'), - (513, 513, 0, 11, 'functionKeys', 3, 5, 3, 1, 'NONE'), - (514, 514, 0, 11, 'functionKeys', 3, 6, 3, 1, 'NONE'), - (515, 515, 0, 11, 'functionKeys', 3, 7, 3, 1, 'NONE'), - (516, 516, 0, 11, 'functionKeys', 3, 8, 3, 1, 'NONE'), - (517, 517, 0, 11, 'functionKeys', 3, 9, 3, 1, 'NONE'), - (518, 518, 0, 11, 'functionKeys', 3, 10, 3, 1, 'NONE'), - (519, 519, 0, 11, 'functionKeys', 3, 11, 3, 1, 'NONE'), - (520, 520, 0, 11, 'functionKeys', 3, 12, 3, 1, 'NONE') - - ; - - - - - - -INSERT IGNORE INTO configuration_node VALUES - (1, 1, 0, 'super-admin', 'test', null, 'EXAM_CONFIG', 'READY_TO_USE') - ; - -INSERT IGNORE INTO configuration VALUES - (1, 1, 1, 'v0', '2019-07-02 12:59:32', 0), - (2, 1, 1, null, null, 1) - ; - -INSERT IGNORE INTO configuration_value VALUES - (1,1,1,1,0,NULL), - (2,1,1,2,0,'true'), - (3,1,1,3,0,'false'), - (4,1,1,4,0,NULL), - (5,1,1,5,0,'2'), - (6,1,1,6,0,'10'), - (7,1,1,7,0,'5'), - (8,1,1,8,0,'0'), - (9,1,1,9,0,'false'), - (10,1,1,10,0,'100%'), - (11,1,1,11,0,'100%'), - (12,1,1,12,0,'1'), - (13,1,1,13,0,'false'), - (14,1,1,14,0,'false'), - (15,1,1,15,0,'false'), - (16,1,1,16,0,'true'), - (17,1,1,17,0,'40'), - (18,1,1,18,0,'true'), - (19,1,1,19,0,'true'), - (20,1,1,20,0,'false'), - (21,1,1,21,0,'true'), - (22,1,1,22,0,'true'), - (23,1,1,23,0,'0'), - (24,1,1,24,0,'false'), - (25,1,1,25,0,'false'), - (26,1,1,26,0,'false'), - (27,1,1,27,0,'25'), - (28,1,1,28,0,'false'), - (29,1,1,29,0,'false'), - (30,1,1,30,0,'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), - (31,1,1,31,0,'2'), - (32,1,1,32,0,'false'), - (33,1,1,33,0,'100%'), - (34,1,1,34,0,'100%'), - (35,1,1,35,0,'2'), - (36,1,1,36,0,'true'), - (37,1,1,37,0,'true'), - (38,1,1,38,0,'false'), - (39,1,1,39,0,'false'), - (40,1,1,40,0,'false'), - (41,1,1,41,0,'false'), - (42,1,1,42,0,'false'), - (43,1,1,43,0,'true'), - (44,1,1,44,0,'true'), - (45,1,1,45,0,'true'), - (46,1,1,46,0,'true'), - (47,1,1,47,0,'false'), - (48,1,1,48,0,'false'), - (49,1,1,49,0,'false'), - (50,1,1,50,0,NULL), - (51,1,1,51,0,'0'), - (52,1,1,52,0,NULL), - (53,1,1,53,0,'0'), - (54,1,1,54,0,NULL), - (55,1,1,55,0,'0'), - (56,1,1,56,0,NULL), - (57,1,1,57,0,'true'), - (58,1,1,58,0,NULL), - (59,1,1,59,0,'true'), - (60,1,1,60,0,NULL), - (61,1,1,61,0,NULL), - (62,1,1,62,0,'false'), - (63,1,1,63,0,'0'), - (64,1,1,64,0,'true'), - (65,1,1,65,0,'true'), - (66,1,1,66,0,'true'), - (67,1,1,67,0,NULL), - (68,1,1,68,0,'true'), - (69,1,1,69,0,'false'), - (70,1,1,70,0,NULL), - (71,1,1,71,0,NULL), - (72,1,1,72,0,'true'), - (73,1,1,73,0,NULL), - (74,1,1,91,0,'false'), - (75,1,1,92,0,'false'), - (76,1,1,93,0,NULL), - (77,1,1,200,0,'false'), - (78,1,1,201,0,'false'), - (79,1,1,202,0,NULL), - (80,1,1,210,0,'0'), - (81,1,1,220,0,NULL), - (82,1,1,221,0,'false'), - (83,1,1,222,0,NULL), - (84,1,1,223,0,'true'), - (85,1,1,231,0,'false'), - (86,1,1,233,0,'false'), - (87,1,1,234,0,NULL), - (88,1,1,235,0,NULL), - (89,1,1,236,0,'false'), - (90,1,1,237,0,NULL), - (91,1,1,238,0,'80'), - (92,1,1,239,0,'false'), - (93,1,1,240,0,NULL), - (94,1,1,241,0,NULL), - (95,1,1,242,0,'false'), - (96,1,1,243,0,NULL), - (97,1,1,244,0,'443'), - (98,1,1,245,0,'false'), - (99,1,1,246,0,NULL), - (100,1,1,247,0,NULL), - (101,1,1,248,0,'false'), - (102,1,1,249,0,NULL), - (103,1,1,250,0,'21'), - (104,1,1,251,0,'false'), - (105,1,1,252,0,NULL), - (106,1,1,253,0,NULL), - (107,1,1,254,0,'false'), - (108,1,1,255,0,NULL), - (109,1,1,256,0,'1080'), - (110,1,1,257,0,'false'), - (111,1,1,258,0,NULL), - (112,1,1,259,0,NULL), - (113,1,1,260,0,'false'), - (114,1,1,261,0,NULL), - (115,1,1,262,0,'1080'), - (116,1,1,263,0,'false'), - (117,1,1,264,0,NULL), - (118,1,1,265,0,NULL), - (119,1,1,300,0,'2'), - (120,1,1,301,0,'0'), - (121,1,1,302,0,'false'), - (122,1,1,303,0,'false'), - (123,1,1,304,0,'true'), - (124,1,1,305,0,'false'), - (125,1,1,306,0,''), - (126,1,1,307,0,'~/Documents'), - (127,1,1,308,0,'0'), - (128,1,1,309,0,'true'), - (129,1,1,310,0,'true'), - (130,1,1,311,0,'false'), - (131,1,1,312,0,'false'), - (132,1,1,313,0,'true'), - (133,1,1,314,0,'false'), - (134,1,1,315,0,'1'), - (135,1,1,316,0,'true'), - (136,1,1,400,0,'false'), - (137,1,1,401,0,'false'), - (138,1,1,402,0,'false'), - (139,1,1,403,0,'false'), - (140,1,1,404,0,'false'), - (141,1,1,405,0,'false'), - (142,1,1,406,0,'false'), - (143,1,1,407,0,'false'), - (144,1,1,408,0,'false'), - (145,1,1,500,0,'false'), - (146,1,1,501,0,'false'), - (147,1,1,502,0,'false'), - (148,1,1,503,0,'false'), - (149,1,1,504,0,'true'), - (150,1,1,505,0,'false'), - (151,1,1,506,0,'false'), - (152,1,1,507,0,'false'), - (153,1,1,508,0,'false'), - (154,1,1,509,0,'false'), - (155,1,1,510,0,'false'), - (156,1,1,511,0,'false'), - (157,1,1,512,0,'false'), - (158,1,1,513,0,'false'), - (159,1,1,514,0,'false'), - (160,1,1,515,0,'false'), - (161,1,1,516,0,'false'), - (162,1,1,517,0,'false'), - (163,1,1,518,0,'false'), - (164,1,1,519,0,'false'), - (165,1,1,520,0,'false'), - (166,1,1,1000,0,'SEB_Server_0.3.0'), - (167,1,1,1001,0,'0'), - (168,1,2,1,0,NULL), - (169,1,2,2,0,'true'), - (170,1,2,3,0,'false'), - (171,1,2,4,0,NULL), - (172,1,2,5,0,'2'), - (173,1,2,6,0,'10'), - (174,1,2,7,0,'5'), - (175,1,2,8,0,'0'), - (176,1,2,9,0,'false'), - (177,1,2,10,0,'100%'), - (178,1,2,11,0,'100%'), - (179,1,2,12,0,'1'), - (180,1,2,13,0,'false'), - (181,1,2,14,0,'false'), - (182,1,2,15,0,'false'), - (183,1,2,16,0,'true'), - (184,1,2,17,0,'40'), - (185,1,2,18,0,'true'), - (186,1,2,19,0,'true'), - (187,1,2,20,0,'false'), - (188,1,2,21,0,'true'), - (189,1,2,22,0,'true'), - (190,1,2,23,0,'0'), - (191,1,2,24,0,'false'), - (192,1,2,25,0,'false'), - (193,1,2,26,0,'false'), - (194,1,2,27,0,'25'), - (195,1,2,28,0,'false'), - (196,1,2,29,0,'false'), - (197,1,2,30,0,'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), - (198,1,2,31,0,'2'), - (199,1,2,32,0,'false'), - (200,1,2,33,0,'100%'), - (201,1,2,34,0,'100%'), - (202,1,2,35,0,'2'), - (203,1,2,36,0,'true'), - (204,1,2,37,0,'true'), - (205,1,2,38,0,'false'), - (206,1,2,39,0,'false'), - (207,1,2,40,0,'false'), - (208,1,2,41,0,'false'), - (209,1,2,42,0,'false'), - (210,1,2,43,0,'true'), - (211,1,2,44,0,'true'), - (212,1,2,45,0,'true'), - (213,1,2,46,0,'true'), - (214,1,2,47,0,'false'), - (215,1,2,48,0,'false'), - (216,1,2,49,0,'false'), - (217,1,2,50,0,NULL), - (218,1,2,51,0,'0'), - (219,1,2,52,0,NULL), - (220,1,2,53,0,'0'), - (221,1,2,54,0,NULL), - (222,1,2,55,0,'0'), - (223,1,2,56,0,NULL), - (224,1,2,57,0,'true'), - (225,1,2,58,0,NULL), - (226,1,2,59,0,'true'), - (227,1,2,60,0,NULL), - (228,1,2,61,0,NULL), - (229,1,2,62,0,'false'), - (230,1,2,63,0,'0'), - (231,1,2,64,0,'true'), - (232,1,2,65,0,'true'), - (233,1,2,66,0,'true'), - (234,1,2,67,0,NULL), - (235,1,2,68,0,'true'), - (236,1,2,69,0,'false'), - (237,1,2,70,0,NULL), - (238,1,2,71,0,NULL), - (239,1,2,72,0,'true'), - (240,1,2,73,0,NULL), - (241,1,2,91,0,'false'), - (242,1,2,92,0,'false'), - (243,1,2,93,0,NULL), - (244,1,2,200,0,'false'), - (245,1,2,201,0,'false'), - (246,1,2,202,0,NULL), - (247,1,2,210,0,'0'), - (248,1,2,220,0,NULL), - (249,1,2,221,0,'false'), - (250,1,2,222,0,NULL), - (251,1,2,223,0,'true'), - (252,1,2,231,0,'false'), - (253,1,2,233,0,'false'), - (254,1,2,234,0,NULL), - (255,1,2,235,0,NULL), - (256,1,2,236,0,'false'), - (257,1,2,237,0,NULL), - (258,1,2,238,0,'80'), - (259,1,2,239,0,'false'), - (260,1,2,240,0,NULL), - (261,1,2,241,0,NULL), - (262,1,2,242,0,'false'), - (263,1,2,243,0,NULL), - (264,1,2,244,0,'443'), - (265,1,2,245,0,'false'), - (266,1,2,246,0,NULL), - (267,1,2,247,0,NULL), - (268,1,2,248,0,'false'), - (269,1,2,249,0,NULL), - (270,1,2,250,0,'21'), - (271,1,2,251,0,'false'), - (272,1,2,252,0,NULL), - (273,1,2,253,0,NULL), - (274,1,2,254,0,'false'), - (275,1,2,255,0,NULL), - (276,1,2,256,0,'1080'), - (277,1,2,257,0,'false'), - (278,1,2,258,0,NULL), - (279,1,2,259,0,NULL), - (280,1,2,260,0,'false'), - (281,1,2,261,0,NULL), - (282,1,2,262,0,'1080'), - (283,1,2,263,0,'false'), - (284,1,2,264,0,NULL), - (285,1,2,265,0,NULL), - (286,1,2,300,0,'2'), - (287,1,2,301,0,'0'), - (288,1,2,302,0,'false'), - (289,1,2,303,0,'false'), - (290,1,2,304,0,'true'), - (291,1,2,305,0,'false'), - (292,1,2,306,0,''), - (293,1,2,307,0,'~/Documents'), - (294,1,2,308,0,'0'), - (295,1,2,309,0,'true'), - (296,1,2,310,0,'true'), - (297,1,2,311,0,'false'), - (298,1,2,312,0,'false'), - (299,1,2,313,0,'true'), - (300,1,2,314,0,'false'), - (301,1,2,315,0,'1'), - (302,1,2,316,0,'true'), - (303,1,2,400,0,'false'), - (304,1,2,401,0,'false'), - (305,1,2,402,0,'false'), - (306,1,2,403,0,'false'), - (307,1,2,404,0,'false'), - (308,1,2,405,0,'false'), - (309,1,2,406,0,'false'), - (310,1,2,407,0,'false'), - (311,1,2,408,0,'false'), - (312,1,2,500,0,'false'), - (313,1,2,501,0,'false'), - (314,1,2,502,0,'false'), - (315,1,2,503,0,'false'), - (316,1,2,504,0,'true'), - (317,1,2,505,0,'false'), - (318,1,2,506,0,'false'), - (319,1,2,507,0,'false'), - (320,1,2,508,0,'false'), - (321,1,2,509,0,'false'), - (322,1,2,510,0,'false'), - (323,1,2,511,0,'false'), - (324,1,2,512,0,'false'), - (325,1,2,513,0,'false'), - (326,1,2,514,0,'false'), - (327,1,2,515,0,'false'), - (328,1,2,516,0,'false'), - (329,1,2,517,0,'false'), - (330,1,2,518,0,'false'), - (331,1,2,519,0,'false'), - (332,1,2,520,0,'false'), - (333,1,2,1000,0,'SEB_Server_0.3.0'), - (334,1,2,1001,0,'0') - ; - -INSERT IGNORE INTO exam_configuration_map VALUES - (1, 1, 2, 1, null, null) - ; - +INSERT IGNORE INTO lms_setup VALUES + (1, 1, 'test', 'MOCKUP', 'http://', 'ccdfa2330533ed6c316a8ffbd64a3197d4a79956ac7ee4c1162f7bdb1a27234fe8793615a51074351e', '8d14b78ecdcbec1d010d414a7208dbe5c411f1fa735c35c7427d840453093a3730d1bc0abe13b9b1a8', null, null, null, null, null, 1) + ; + +INSERT IGNORE INTO seb_client_configuration VALUES + (1, 1, 'test', '2019-07-02 09:22:50', 'test', '98ac3c953abf5948d9d13c81cab580819ee2624c76d6d4147d4896a5b79f49956d382c08c93cb3b9ae350b32', null, 1) + ; + +INSERT IGNORE INTO exam VALUES + (1, 1, 1, 'quiz1', 'super-admin', 'super-admin', 'MANAGED', null, null, 'UP_COMING', 1, 0, null, 1), + (2, 1, 1, 'quiz6', 'super-admin', 'super-admin', 'MANAGED', null, null, 'RUNNING', 1, 0, null, 1) + ; + +INSERT IGNORE INTO indicator VALUES + (1, 2, 'LAST_PING', 'Ping', 'dcdcdc') + ; + +INSERT IGNORE INTO threshold VALUES + (1, 1, 1000.0000, '22b14c'), + (2, 1, 2000.0000, 'ff7e00'), + (3, 1, 5000.0000, 'ed1c24') + ; + +INSERT IGNORE INTO view VALUES + (1, 'general', 4, 1, 0), + (2, 'user_interface', 12, 2, 0), + (3, 'browser', 12, 3, 0), + (4, 'down_upload', 12, 4, 0), + (5, 'exam', 12, 5, 0), + (6, 'applications', 12, 6, 0), + (7, 'resources', 12, 7, 0), + (8, 'network', 12, 8, 0), + (9, 'security', 12, 9, 0), + (10, 'registry', 12, 10, 0), + (11, 'hooked_keys', 12, 11, 0); + +INSERT IGNORE INTO configuration_attribute VALUES + (1, 'hashedAdminPassword', 'PASSWORD_FIELD', null, null, null, null, null), + (2, 'allowQuit', 'CHECKBOX', null, null, null, null, 'true'), + (3, 'ignoreExitKeys', 'CHECKBOX', null, null, null, null, 'false'), + (4, 'hashedQuitPassword', 'PASSWORD_FIELD', null, null, null, null, null), + (5, 'exitKey1', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '2'), + (6, 'exitKey2', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '10'), + (7, 'exitKey3', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7,8,9,10,11', 'ExitKeySequenceValidator', 'resourceLocTextKey=sebserver.examconfig.props.label.exitKey', '5'), + + (8, 'browserViewMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), + (9, 'enableTouchExit', 'CHECKBOX', null, null, null, null, 'false'), + (10, 'mainBrowserWindowWidth', 'COMBO_SELECTION', null, '50%,100%,800,1000', 'WindowsSizeValidator', null, '100%'), + (11, 'mainBrowserWindowHeight', 'COMBO_SELECTION', null, '80%,100%,600,800', 'WindowsSizeValidator', null, '100%'), + (12, 'mainBrowserWindowPositioning', 'SINGLE_SELECTION', null, '0,1,2', null, null, '1'), + (13, 'enableBrowserWindowToolbar', 'CHECKBOX', null, null, null, null, 'false'), + (14, 'hideBrowserWindowToolbar', 'CHECKBOX', null, null, null, null, 'false'), + (15, 'showMenuBar', 'CHECKBOX', null, null, null, null, 'false'), + (16, 'showTaskBar', 'CHECKBOX', null, null, null, null, 'true'), + (17, 'taskBarHeight', 'COMBO_SELECTION', null, '40,60,80', 'IntegerTypeValidator', null, '40'), + (18, 'showReloadButton', 'CHECKBOX', null, null, null, null, 'true'), + (19, 'showTime', 'CHECKBOX', null, null, null, null, 'true'), + (20, 'showInputLanguage', 'CHECKBOX', null, null, null, null, 'false'), + (21, 'enableZoomPage', 'CHECKBOX', null, null, null, null, 'true'), + (22, 'enableZoomText', 'CHECKBOX', null, null, null, null, 'true'), + (23, 'zoomMode', 'RADIO_SELECTION', null, '0,1', null, null, '0'), + (24, 'audioControlEnabled', 'CHECKBOX', null, null, null, null, 'false'), + (25, 'audioMute', 'CHECKBOX', null, null, null, null, 'false'), + (26, 'audioSetVolumeLevel', 'CHECKBOX', null, null, null, null, 'false'), + (27, 'audioVolumeLevel', 'SLIDER', null, '0,100', null, null, '25'), + (28, 'allowSpellCheck', 'CHECKBOX', null, null, null, null, 'false'), + (29, 'allowDictionaryLookup', 'CHECKBOX', null, null, null, null, 'false'), + (30, 'allowSpellCheckDictionary', 'MULTI_CHECKBOX_SELECTION', null, 'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI', null, null, 'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), + + (31, 'newBrowserWindowByLinkPolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '2'), + (32, 'newBrowserWindowByLinkBlockForeign', 'CHECKBOX', null, null, null, null, 'false'), + (33, 'newBrowserWindowByLinkWidth', 'COMBO_SELECTION', null, '50%,100%,800,1000', 'WindowsSizeValidator', null, '100%'), + (34, 'newBrowserWindowByLinkHeight', 'COMBO_SELECTION', null, '80%,100%,600,800', 'WindowsSizeValidator', null, '100%'), + (35, 'newBrowserWindowByLinkPositioning', 'SINGLE_SELECTION', null, '0,1,2', null, null, '2'), + (36, 'enablePlugIns', 'CHECKBOX', null, null, null, null, 'true'), + (37, 'enableJavaScript', 'CHECKBOX', null, null, null, null, 'true'), + (38, 'enableJava', 'CHECKBOX', null, null, null, null, 'false'), + (39, 'blockPopUpWindows', 'CHECKBOX', null, null, null, null, 'false'), + (40, 'allowVideoCapture', 'CHECKBOX', null, null, null, null, 'false'), + (41, 'allowAudioCapture', 'CHECKBOX', null, null, null, null, 'false'), + (42, 'allowBrowsingBackForward', 'CHECKBOX', null, null, null, null, 'false'), + (43, 'newBrowserWindowNavigation', 'CHECKBOX', null, null, null, null, 'true'), + (44, 'browserWindowAllowReload', 'CHECKBOX', null, null, null, null, 'true'), + (45, 'newBrowserWindowAllowReload', 'CHECKBOX', null, null, null, null, 'true'), + (46, 'showReloadWarning', 'CHECKBOX', null, null, null, null, 'true'), + (47, 'newBrowserWindowShowReloadWarning', 'CHECKBOX', null, null, null, null, 'false'), + (48, 'removeBrowserProfile', 'CHECKBOX', null, null, null, null, 'false'), + (49, 'removeLocalStorage', 'CHECKBOX', null, null, null, null, 'false'), + (50, 'browserUserAgent', 'TEXT_FIELD', null, null, null, null, null), + (51, 'browserUserAgentWinDesktopMode', 'RADIO_SELECTION', null, '0,1', null, null, '0'), + (52, 'browserUserAgentWinDesktopModeCustom', 'TEXT_FIELD', null, null, null, null, null), + (53, 'browserUserAgentWinTouchMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), + (54, 'browserUserAgentWinTouchModeCustom', 'TEXT_FIELD', null, null, null, null, null), + (55, 'browserUserAgentMac', 'RADIO_SELECTION', null, '0,1', null, null, '0'), + (56, 'browserUserAgentMacCustom', 'TEXT_FIELD', null, null, null, null, null), + (57, 'enableSebBrowser', 'CHECKBOX', null, null, null, null, 'true'), + (58, 'browserWindowTitleSuffix', 'TEXT_FIELD', null, null, null, null, null), + + (59, 'allowDownUploads', 'CHECKBOX', null, null, null, null, 'true'), + (60, 'downloadDirectoryWin', 'TEXT_FIELD', null, null, null, null, null), + (61, 'downloadDirectoryOSX', 'TEXT_FIELD', null, null, null, null, null), + (62, 'openDownloads', 'CHECKBOX', null, null, null, null, 'false'), + (63, 'chooseFileToUploadPolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), + (64, 'downloadPDFFiles', 'CHECKBOX', null, null, null, null, 'true'), + (65, 'allowPDFPlugIn', 'CHECKBOX', null, null, null, null, 'true'), + (66, 'downloadAndOpenSebConfig', 'CHECKBOX', null, null, null, null, 'true'), + + (67, 'quitURL', 'TEXT_FIELD', null, null, null, null, null), + (68, 'quitURLConfirm', 'CHECKBOX', null, null, null, null, 'true'), + (69, 'restartExamUseStartURL', 'CHECKBOX', null, null, null, null, 'false'), + (70, 'restartExamURL', 'TEXT_FIELD', null, null, null, null, null), + (71, 'restartExamText', 'TEXT_FIELD', null, null, null, null, null), + (72, 'restartExamPasswordProtected', 'CHECKBOX', null, null, null, null, 'true'), + + (73, 'permittedProcesses', 'TABLE', null, null, null, null, null), + (74, 'permittedProcesses.active', 'CHECKBOX', 73, null, null, null, 'true'), + (75, 'permittedProcesses.os', 'SINGLE_SELECTION', 73, '0,1', null, null, '1'), + (76, 'permittedProcesses.title', 'TEXT_FIELD', 73, null, null, null, ''), + (77, 'permittedProcesses.description', 'TEXT_FIELD', 73, null, null, null, ''), + (78, 'permittedProcesses.executable', 'TEXT_FIELD', 73, null, null, null, ''), + (79, 'permittedProcesses.originalName', 'TEXT_FIELD', 73, null, null, null, ''), + (80, 'permittedProcesses.allowedExecutables', 'TEXT_FIELD', 73, null, null, null, ''), + (81, 'permittedProcesses.path', 'TEXT_FIELD', 73, null, null, null, ''), + (82, 'permittedProcesses.arguments', 'INLINE_TABLE', 73, '1:active:CHECKBOX|4:argument:TEXT_FIELD', null, null, null), + (85, 'permittedProcesses.identifier', 'TEXT_FIELD', 73, null, null, null, ''), + (86, 'permittedProcesses.iconInTaskbar', 'CHECKBOX', 73, null, null, null, 'true'), + (87, 'permittedProcesses.autostart', 'CHECKBOX', 73, null, null, null, 'false'), + (88, 'permittedProcesses.runInBackground', 'CHECKBOX', 73, null, null, null, 'false'), + (89, 'permittedProcesses.allowUserToChooseApp', 'CHECKBOX', 73, null, null, null, 'false'), + (90, 'permittedProcesses.strongKill', 'CHECKBOX', 73, null, null, null, 'false'), + (91, 'allowSwitchToApplications', 'CHECKBOX', null, null, null, null, 'false'), + (92, 'allowFlashFullscreen', 'CHECKBOX', null, null, null, null, 'false'), + + (93, 'prohibitedProcesses', 'TABLE', null, null, null, null, null), + (94, 'prohibitedProcesses.active', 'CHECKBOX', 93, null, null, null, 'true'), + (95, 'prohibitedProcesses.os', 'SINGLE_SELECTION', 93, '0,1', null, null, '1'), + (96, 'prohibitedProcesses.executable', 'TEXT_FIELD', 93, null, null, null, ''), + (97, 'prohibitedProcesses.description', 'TEXT_FIELD', 93, null, null, null, ''), + (98, 'prohibitedProcesses.originalName', 'TEXT_FIELD', 93, null, null, null, ''), + (99, 'prohibitedProcesses.identifier', 'TEXT_FIELD', 93, null, null, null, ''), + (100, 'prohibitedProcesses.strongKill', 'CHECKBOX', 93, null, null, null, 'false'), + (101, 'prohibitedProcesses.currentUser', 'CHECKBOX', 93, null, null, null, 'false'), + (102, 'prohibitedProcesses.user', 'TEXT_FIELD', 93, null, null, null, null), + + (200, 'URLFilterEnable', 'CHECKBOX', null, null, null, null, 'false'), + (201, 'URLFilterEnableContentFilter', 'CHECKBOX', null, null, null, null, 'false'), + (202, 'URLFilterRules', 'TABLE', null, null, null, null, null), + (203, 'URLFilterRules.active', 'CHECKBOX', 202, null, null, null, 'true'), + (204, 'URLFilterRules.regex', 'CHECKBOX', 202, null, null, null, 'false'), + (205, 'URLFilterRules.expression', 'TEXT_FIELD', 202, null, null, null, ''), + (206, 'URLFilterRules.action', 'SINGLE_SELECTION', 202, '0,1', null, null, '0'), + + (210, 'proxySettingsPolicy', 'RADIO_SELECTION', null, '0,1', null, null, '0'), + (220, 'proxies', 'COMPOSITE_TABLE', null, 'active,TABLE_ENTRY|autoDiscovery,autoConfiguration,http,https,ftp,socks,rtsp', null, null, null), + (221, 'ExcludeSimpleHostnames', 'CHECKBOX', 220, null, null, 'showInView=true,createDefaultValue=true', 'false'), + (222, 'ExceptionsList', 'TEXT_AREA', 220, null, null, 'showInView=true,createDefaultValue=true', null), + (223, 'FTPPassive', 'CHECKBOX', 220, null, null, 'showInView=true,createDefaultValue=true', 'true'), + (231, 'AutoDiscoveryEnabled', 'CHECKBOX', 220, null, null, 'groupId=autoDiscovery,createDefaultValue=true', 'false'), + (233, 'AutoConfigurationEnabled', 'CHECKBOX', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', 'false'), + (234, 'AutoConfigurationURL', 'TEXT_FIELD', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', null), + (235, 'AutoConfigurationJavaScript', 'TEXT_AREA', 220, null, null, 'groupId=autoConfiguration,createDefaultValue=true', null), + (236, 'HTTPEnable', 'CHECKBOX', 220, null, null, 'groupId=http,createDefaultValue=true', 'false'), + (237, 'HTTPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), + (238, 'HTTPPort', 'INTEGER', 220, null, null, 'groupId=http,createDefaultValue=true', '80'), + (239, 'HTTPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=http,createDefaultValue=true', 'false'), + (240, 'HTTPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), + (241, 'HTTPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=http,createDefaultValue=true', null), + (242, 'HTTPSEnable', 'CHECKBOX', 220, null, null, 'groupId=https,createDefaultValue=true', 'false'), + (243, 'HTTPSProxy', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), + (244, 'HTTPSPort', 'INTEGER', 220, null, null, 'groupId=https,createDefaultValue=true', '443'), + (245, 'HTTPSRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=https,createDefaultValue=true', 'false'), + (246, 'HTTPSUsername', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), + (247, 'HTTPSPassword', 'TEXT_FIELD', 220, null, null, 'groupId=https,createDefaultValue=true', null), + (248, 'FTPEnable', 'CHECKBOX', 220, null, null, 'groupId=ftp,createDefaultValue=true', 'false'), + (249, 'FTPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), + (250, 'FTPPort', 'INTEGER', 220, null, null, 'groupId=ftp,createDefaultValue=true', '21'), + (251, 'FTPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=ftp,createDefaultValue=true', 'false'), + (252, 'FTPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), + (253, 'FTPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=ftp,createDefaultValue=true', null), + (254, 'SOCKSEnable', 'CHECKBOX', 220, null, null, 'groupId=socks,createDefaultValue=true', 'false'), + (255, 'SOCKSProxy', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), + (256, 'SOCKSPort', 'INTEGER', 220, null, null, 'groupId=socks,createDefaultValue=true', '1080'), + (257, 'SOCKSRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=socks,createDefaultValue=true', 'false'), + (258, 'SOCKSUsername', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), + (259, 'SOCKSPassword', 'TEXT_FIELD', 220, null, null, 'groupId=socks,createDefaultValue=true', null), + (260, 'RTSPEnable', 'CHECKBOX', 220, null, null, 'groupId=rtsp,createDefaultValue=true', 'false'), + (261, 'RTSPProxy', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), + (262, 'RTSPPort', 'INTEGER', 220, null, null, 'groupId=rtsp,createDefaultValue=true', '554'), + (263, 'RTSPRequiresPassword', 'CHECKBOX', 220, null, null, 'groupId=rtsp,createDefaultValue=true', 'false'), + (264, 'RTSPUsername', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), + (265, 'RTSPPassword', 'TEXT_FIELD', 220, null, null, 'groupId=rtsp,createDefaultValue=true', null), + + + (300, 'sebServicePolicy', 'RADIO_SELECTION', null, '0,1,2', null, null, '2'), + (301, 'kioskMode', 'RADIO_SELECTION', null, '0,1,2', null, null, '0'), + (302, 'allowVirtualMachine', 'CHECKBOX', null, null, null, null, 'false'), + (303, 'allowScreenSharing', 'CHECKBOX', null, null, null, null, 'false'), + (304, 'enablePrivateClipboard', 'CHECKBOX', null, null, null, null, 'true'), + (305, 'enableLogging', 'CHECKBOX', null, null, null, null, 'false'), + (306, 'logDirectoryWin', 'TEXT_FIELD', null, null, null, null, ''), + (307, 'logDirectoryOSX', 'TEXT_FIELD', null, null, null, null, ''), + (308, 'minMacOSVersion', 'SINGLE_SELECTION', null, '0,1,2,3,4,5,6,7', null, null, '0'), + (309, 'enableAppSwitcherCheck', 'CHECKBOX', null, null, null, null, 'true'), + (310, 'forceAppFolderInstall', 'CHECKBOX', null, null, null, null, 'true'), + (311, 'allowUserAppFolderInstall', 'CHECKBOX', null, null, null, null, 'false'), + (312, 'allowSiri', 'CHECKBOX', null, null, null, null, 'false'), + (313, 'detectStoppedProcess', 'CHECKBOX', null, null, null, null, 'true'), + (314, 'allowDisplayMirroring', 'CHECKBOX', null, null, null, null, 'false'), + (315, 'allowedDisplaysMaxNumber', 'COMBO_SELECTION', null, '1,2,3', null, null, '1'), + (316, 'allowedDisplayBuiltin', 'CHECKBOX', null, null, null, null, 'true'), + (317, 'logLevel', 'SINGLE_SELECTION', null, '0,1,2,3,4', null, null, '1'), + + (400, 'insideSebEnableSwitchUser', 'CHECKBOX', null, null, null, null, 'false'), + (401, 'insideSebEnableLockThisComputer', 'CHECKBOX', null, null, null, null, 'false'), + (402, 'insideSebEnableChangeAPassword', 'CHECKBOX', null, null, null, null, 'false'), + (403, 'insideSebEnableStartTaskManager', 'CHECKBOX', null, null, null, null, 'false'), + (404, 'insideSebEnableLogOff', 'CHECKBOX', null, null, null, null, 'false'), + (405, 'insideSebEnableShutDown', 'CHECKBOX', null, null, null, null, 'false'), + (406, 'insideSebEnableVmWareClientShade', 'CHECKBOX', null, null, null, null, 'false'), + (407, 'insideSebEnableEaseOfAccess', 'CHECKBOX', null, null, null, null, 'false'), + (408, 'insideSebEnableNetworkConnectionSelector', 'CHECKBOX', null, null, null, null, 'false'), + + (500, 'enableEsc', 'CHECKBOX', null, null, null, null, 'false'), + (501, 'enablePrintScreen', 'CHECKBOX', null, null, null, null, 'false'), + (502, 'enableCtrlEsc', 'CHECKBOX', null, null, null, null, 'false'), + (503, 'enableAltEsc', 'CHECKBOX', null, null, null, null, 'false'), + (504, 'enableAltTab', 'CHECKBOX', null, null, null, null, 'true'), + (505, 'enableAltF4', 'CHECKBOX', null, null, null, null, 'false'), + (506, 'enableStartMenu', 'CHECKBOX', null, null, null, null, 'false'), + (507, 'enableRightMouse', 'CHECKBOX', null, null, null, null, 'false'), + (508, 'enableAltMouseWheel', 'CHECKBOX', null, null, null, null, 'false'), + + (509, 'enableF1', 'CHECKBOX', null, null, null, null, 'false'), + (510, 'enableF2', 'CHECKBOX', null, null, null, null, 'false'), + (511, 'enableF3', 'CHECKBOX', null, null, null, null, 'false'), + (512, 'enableF4', 'CHECKBOX', null, null, null, null, 'false'), + (513, 'enableF5', 'CHECKBOX', null, null, null, null, 'false'), + (514, 'enableF6', 'CHECKBOX', null, null, null, null, 'false'), + (515, 'enableF7', 'CHECKBOX', null, null, null, null, 'false'), + (516, 'enableF8', 'CHECKBOX', null, null, null, null, 'false'), + (517, 'enableF9', 'CHECKBOX', null, null, null, null, 'false'), + (518, 'enableF10', 'CHECKBOX', null, null, null, null, 'false'), + (519, 'enableF11', 'CHECKBOX', null, null, null, null, 'false'), + (520, 'enableF12', 'CHECKBOX', null, null, null, null, 'false'), + + (800, 'browserMessagingSocket', 'TEXT_FIELD', null, null, null, null, 'ws://localhost:8706'), + (801, 'browserMessagingPingTime', 'INTEGER', null, null, null, null, '120000'), + (802, 'allowPreferencesWindow', 'CHECKBOX', null, null, null, null, 'true'), + (803, 'useAsymmetricOnlyEncryption', 'CHECKBOX', null, null, null, null, 'false'), + (804, 'touchOptimized', 'CHECKBOX', null, null, null, null, 'false'), + (805, 'browserScreenKeyboard', 'CHECKBOX', null, null, null, null, 'false'), + (806, 'newBrowserWindowByScriptPolicy', 'INTEGER', null, null, null, null, '2'), + (807, 'newBrowserWindowByScriptBlockForeign', 'CHECKBOX', null, null, null, null, 'false'), + (808, 'monitorProcesses', 'CHECKBOX', null, null, null, null, 'false'), + (809, 'blacklistURLFilter', 'TEXT_FIELD', null, null, null, null, ''), + (810, 'whitelistURLFilter', 'TEXT_FIELD', null, null, null, null, ''), + (812, 'allowWlan', 'CHECKBOX', null, null, null, null, 'false'), + (813, 'hookKeys', 'CHECKBOX', null, null, null, null, 'true'), + + (900, 'examSessionClearCookiesOnEnd', 'CHECKBOX', null, null, null, null, 'true'), + (901, 'examSessionClearCookiesOnStart', 'CHECKBOX', null, null, null, null, 'true'), + (902, 'showBackToStartButton', 'CHECKBOX', null, null, null, null, 'true'), + (903, 'showSettingsInApp', 'CHECKBOX', null, null, null, null, 'false'), + (904, 'browserUserAgentWinTouchModeIPad', 'TEXT_FIELD', null, null, null, null, 'Mozilla/5.0 (iPad; CPU OS 12_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/12.1.2 Mobile/15E148 Safari/604.1'), + (905, 'mobileAllowPictureInPictureMediaPlayback', 'CHECKBOX', null, null, null, null, 'false'), + (906, 'lockOnMessageSocketClose', 'CHECKBOX', null, null, null, null, 'false'), + (907, 'enableDrawingEditor', 'CHECKBOX', null, null, null, null, 'false'), + (908, 'URLFilterMessage', 'RADIO_SELECTION', null, '0,1', null, null, '0'), + (909, 'allowDictation', 'CHECKBOX', null, null, null, null, 'false'), + (910, 'allowUserSwitching', 'CHECKBOX', null, null, null, null, 'false'), + (911, 'allowiOSBetaVersionNumber', 'SINGLE_SELECTION', null, '0,12', null, null, '0'), + (912, 'allowiOSVersionNumberMajor', 'SINGLE_SELECTION', null, '9,10,11,12', null, null, '9'), + (913, 'allowiOSVersionNumberMinor', 'INTEGER', null, null, null, null, '3'), + (914, 'allowiOSVersionNumberPatch', 'INTEGER', null, null, null, null, '5'), + (915, 'browserURLSalt', 'CHECKBOX', null, null, null, null, 'true'), + (917, 'browserUserAgentiOS', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '0'), + (918, 'browserUserAgentiOSCustom', 'TEXT_FIELD', null, null, null, null, ''), + (919, 'browserWindowShowURL', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '0'), + (920, 'mobileAllowQRCodeConfig', 'CHECKBOX', null, null, null, null, 'false'), + (921, 'mobileAllowSingleAppMode', 'CHECKBOX', null, null, null, null, 'false'), + (922, 'mobileEnableASAM', 'CHECKBOX', null, null, null, null, 'true'), + (923, 'mobileEnableGuidedAccessLinkTransform', 'CHECKBOX', null, null, null, null, 'false'), + (924, 'mobilePreventAutoLock', 'CHECKBOX', null, null, null, null, 'true'), + (925, 'mobileShowSettings', 'CHECKBOX', null, null, null, null, 'false'), + (926, 'mobileStatusBarAppearance', 'SINGLE_SELECTION', null, '0,1,2', null, null, '1'), + (927, 'mobileStatusBarAppearanceExtended', 'SINGLE_SELECTION', null, '0,1,2,3,4', null, null, '1'), + (928, 'newBrowserWindowShowURL', 'SINGLE_SELECTION', null, '0,1,2,3', null, null, '1'), + (929, 'pinEmbeddedCertificates', 'CHECKBOX', null, null, null, null, 'false'), + (930, 'sendBrowserExamKey', 'CHECKBOX', null, null, null, null, 'true'), + (931, 'showNavigationButtons', 'CHECKBOX', null, null, null, null, 'false'), + (932, 'showScanQRCodeButton', 'CHECKBOX', null, null, null, null, 'false'), + (933, 'startResource', 'TEXT_FIELD', null, null, null, null, ''), + + (1000, 'originatorVersion', 'TEXT_FIELD', null, null, null, null, 'SEB_Server_0.3.0'), + (1001, 'sebConfigPurpose', 'RADIO_SELECTION', null, '0,1', null, null, '0') + + ; + +INSERT IGNORE INTO orientation VALUES + (1, 1, 0, 1, null, 1, 1, 1, 2, 'LEFT'), + (2, 2, 0, 1, null, 1, 3, 1, 1, 'LEFT'), + (3, 3, 0, 1, null, 1, 4, 1, 1, 'LEFT'), + (4, 4, 0, 1, null, 1, 5, 1, 2, 'LEFT'), + (5, 5, 0, 1, 'exitSequence', 2, 1, 1, 1, 'NONE'), + (6, 6, 0, 1, 'exitSequence', 2, 2, 1, 1, 'NONE'), + (7, 7, 0, 1, 'exitSequence', 2, 3, 1, 1, 'NONE'), + + (8, 8, 0, 2, 'browserViewMode', 0, 0, 3, 3, 'NONE'), + (9, 9, 0, 2, 'browserViewMode', 3, 2, 4, 1, 'NONE'), + (10, 10, 0, 2, 'winsize', 1, 4, 2, 1, 'LEFT'), + (11, 11, 0, 2, 'winsize', 1, 5, 2, 1, 'LEFT'), + (12, 12, 0, 2, 'winsize', 5, 4, 2, 1, 'LEFT_SPAN'), + (13, 13, 0, 2, 'wintoolbar', 0, 6, 3, 1, 'NONE'), + (14, 14, 0, 2, 'wintoolbar', 3, 6, 4, 1, 'NONE'), + (15, 15, 0, 2, 'wintoolbar', 0, 7, 3, 1, 'NONE'), + (16, 16, 0, 2, 'taskbar', 0, 9, 3, 1, 'NONE'), + (17, 17, 0, 2, 'taskbar', 5, 9, 2, 1, 'LEFT_SPAN'), + (18, 18, 0, 2, 'taskbar', 0, 10, 3, 1, 'NONE'), + (19, 19, 0, 2, 'taskbar', 0, 11, 3, 1, 'NONE'), + (20, 20, 0, 2, 'taskbar', 0, 12, 3, 1, 'NONE'), + (21, 21, 0, 2, 'zoom', 0, 14, 3, 1, 'NONE'), + (22, 22, 0, 2, 'zoom', 0, 15, 3, 1, 'NONE'), + (23, 23, 0, 2, 'zoomMode', 3, 14, 4, 1, 'NONE'), + (24, 24, 0, 2, 'audio', 7, 0, 5, 1, 'NONE'), + (25, 25, 0, 2, 'audio', 7, 1, 5, 1, 'NONE'), + (26, 26, 0, 2, 'audio', 7, 2, 5, 1, 'NONE'), + (27, 27, 0, 2, 'audio', 7, 3, 5, 1, 'NONE'), + (28, 28, 0, 2, 'spellcheck', 7, 4, 5, 1, 'NONE'), + (29, 29, 0, 2, 'spellcheck', 7, 5, 5, 1, 'NONE'), + (30, 30, 0, 2, 'spellcheck', 7, 7, 5, 9, 'TOP'), + + (31, 31, 0, 3, 'newBrowserWindow', 0, 0, 3, 3, 'NONE'), + (32, 32, 0, 3, 'newBrowserWindow', 4, 0, 3, 1, 'NONE'), + (33, 33, 0, 3, 'newwinsize', 1, 4, 2, 1, 'LEFT'), + (34, 34, 0, 3, 'newwinsize', 1, 5, 2, 1, 'LEFT'), + (35, 35, 0, 3, 'newwinsize', 5, 4, 2, 1, 'LEFT_SPAN'), + (36, 36, 0, 3, 'browserSecurity', 0, 5, 4, 1, 'NONE'), + (37, 37, 0, 3, 'browserSecurity', 4, 5, 3, 1, 'NONE'), + (38, 38, 0, 3, 'browserSecurity', 0, 6, 4, 1, 'NONE'), + (39, 39, 0, 3, 'browserSecurity', 4, 6, 3, 1, 'NONE'), + (40, 40, 0, 3, 'browserSecurity', 0, 7, 4, 1, 'NONE'), + (41, 41, 0, 3, 'browserSecurity', 4, 7, 3, 1, 'NONE'), + (42, 42, 0, 3, 'browserSecurity', 0, 8, 4, 1, 'NONE'), + (43, 43, 0, 3, 'browserSecurity', 4, 8, 3, 1, 'NONE'), + (44, 44, 0, 3, 'browserSecurity', 0, 9, 4, 1, 'NONE'), + (45, 45, 0, 3, 'browserSecurity', 4, 9, 3, 1, 'NONE'), + (46, 46, 0, 3, 'browserSecurity', 0, 10, 4, 1, 'NONE'), + (47, 47, 0, 3, 'browserSecurity', 4, 10, 3, 1, 'NONE'), + (48, 48, 0, 3, 'browserSecurity', 0, 11, 4, 1, 'NONE'), + (49, 49, 0, 3, 'browserSecurity', 4, 11, 3, 1, 'NONE'), + + (50, 50, 0, 3, null, 7, 1, 5, 1, 'TOP'), + (51, 51, 0, 3, 'userAgentDesktop', 7, 2, 5, 2, 'NONE'), + (52, 52, 0, 3, 'userAgentDesktop', 7, 3, 5, 1, 'NONE'), + (53, 53, 0, 3, 'userAgentTouch', 7, 4, 5, 2, 'NONE'), + (54, 54, 0, 3, 'userAgentTouch', 7, 6, 5, 1, 'NONE'), + (55, 55, 0, 3, 'userAgentMac', 7, 8, 5, 2, 'NONE'), + (56, 56, 0, 3, 'userAgentMac', 7, 10, 5, 1, 'NONE'), + (57, 57, 0, 3, null, 0, 13, 6, 1, 'NONE'), + (58, 58, 0, 3, null, 7, 13, 5, 1, 'TOP'), + + (59, 59, 0, 4, null, 0, 0, 8, 1, 'NONE'), + (60, 60, 0, 4, null, 3, 1, 5, 1, 'LEFT_SPAN'), + (61, 61, 0, 4, null, 3, 2, 5, 1, 'LEFT_SPAN'), + (62, 62, 0, 4, null, 0, 3, 8, 1, 'NONE'), + (63, 63, 0, 4, null, 0, 5, 8, 2, 'TOP'), + (64, 64, 0, 4, null, 0, 8, 8, 1, 'NONE'), + (65, 65, 0, 4, null, 0, 9, 8, 1, 'NONE'), + (66, 66, 0, 4, null, 0, 10, 8, 1, 'NONE'), + + (67, 67, 0, 5, 'quitLink', 0, 1, 8, 1, 'TOP'), + (68, 68, 0, 5, 'quitLink', 0, 2, 8, 1, 'NONE'), + (69, 69, 0, 5, 'backToStart', 0, 4, 8, 1, 'NONE'), + (70, 70, 0, 5, 'backToStart', 0, 6, 8, 2, 'TOP'), + (71, 71, 0, 5, 'backToStart', 0, 8, 8, 2, 'TOP'), + (72, 72, 0, 5, 'backToStart', 0, 10, 8, 1, 'NONE'), + + (73, 73, 0, 6, null, 0, 2, 10, 6, 'TOP'), + (74, 74, 0, 6, null, 1, 1, 1, 1, 'LEFT'), + (75, 75, 0, 6, null, 2, 2, 1, 1, 'LEFT'), + (76, 76, 0, 6, null, 4, 4, 2, 1, 'LEFT'), + (77, 77, 0, 6, null, 0, 3, 1, 1, 'LEFT'), + (78, 78, 0, 6, null, 3, 4, 4, 1, 'LEFT'), + (79, 79, 0, 6, null, 0, 5, 1, 1, 'LEFT'), + (80, 80, 0, 6, null, 0, 6, 1, 1, 'LEFT'), + (81, 81, 0, 6, null, 0, 7, 1, 1, 'LEFT'), + + (82, 82, 0, 6, null, 0, 8, 1, 3, 'LEFT'), + + (85, 85, 0, 6, null, 0, 8, 1, 1, 'LEFT'), + (86, 86, 0, 6, null, 0, 7, 1, 1, 'LEFT'), + (87, 87, 0, 6, null, 0, 9, 1, 1, 'LEFT'), + (88, 88, 0, 6, null, 0, 10, 1, 1, 'LEFT'), + (89, 89, 0, 6, null, 0, 11, 1, 1, 'LEFT'), + (90, 90, 0, 6, null, 0, 12, 1, 1, 'LEFT'), + (91, 91, 0, 6, null, 0, 0, 5, 1, 'NONE'), + (92, 92, 0, 6, null, 5, 0, 5, 1, 'NONE'), + (93, 93, 0, 6, null, 0, 10, 10, 6, 'TOP'), + (94, 94, 0, 6, null, 1, 1, 1, 1, 'LEFT'), + (95, 95, 0, 6, null, 2, 2, 1, 1, 'LEFT'), + (96, 96, 0, 6, null, 3, 3, 4, 1, 'LEFT'), + (97, 97, 0, 6, null, 4, 5, 2, 1, 'LEFT'), + (98, 98, 0, 6, null, 0, 4, 1, 1, 'LEFT'), + (99, 99, 0, 6, null, 0, 6, 1, 1, 'LEFT'), + (100, 100, 0, 6, null, 0, 7, 1, 1, 'LEFT'), + + (200, 200, 0, 8, 'urlFilter', 0, 0, 3, 1, 'NONE'), + (201, 201, 0, 8, 'urlFilter', 3, 0, 4, 1, 'NONE'), + (202, 202, 0, 8, 'urlFilter', 0, 1, 12, 6, 'NONE'), + (203, 203, 0, 8, 'urlFilter', 1, 1, 1, 1, 'LEFT'), + (204, 204, 0, 8, 'urlFilter', 2, 2, 1, 1, 'LEFT'), + (205, 205, 0, 8, 'urlFilter', 3, 3, 4, 1, 'LEFT'), + (206, 206, 0, 8, 'urlFilter', 4, 4, 2, 1, 'LEFT'), + + (210, 210, 0, 8, 'proxies', 0, 6, 5, 2, 'NONE'), + (220, 220, 0, 8, 'proxies', 7, 7, 5, 7, 'TOP'), + (221, 221, 0, 8, 'proxies', 0, 8, 6, 1, 'NONE'), + (222, 222, 0, 8, 'proxies', 0, 10, 6, 2, 'TOP'), + (223, 223, 0, 8, 'proxies', 0, 11, 6, 1, 'NONE'), + + (231, 231, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + + (233, 233, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (234, 234, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (235, 235, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (236, 236, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (237, 237, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (238, 238, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (239, 239, 0, 8, null, 0, 3, 1, 1, 'LEFT'), + (240, 240, 0, 8, null, 0, 4, 1, 1, 'LEFT'), + (241, 241, 0, 8, null, 0, 5, 1, 1, 'LEFT'), + (242, 242, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (243, 243, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (244, 244, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (245, 245, 0, 8, null, 0, 3, 1, 1, 'LEFT'), + (246, 246, 0, 8, null, 0, 4, 1, 1, 'LEFT'), + (247, 247, 0, 8, null, 0, 5, 1, 1, 'LEFT'), + (248, 248, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (249, 249, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (250, 250, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (251, 251, 0, 8, null, 0, 3, 1, 1, 'LEFT'), + (252, 252, 0, 8, null, 0, 4, 1, 1, 'LEFT'), + (253, 253, 0, 8, null, 0, 5, 1, 1, 'LEFT'), + (254, 254, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (255, 255, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (256, 256, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (257, 257, 0, 8, null, 0, 3, 1, 1, 'LEFT'), + (258, 258, 0, 8, null, 0, 4, 1, 1, 'LEFT'), + (259, 259, 0, 8, null, 0, 5, 1, 1, 'LEFT'), + (260, 260, 0, 8, 'active', 0, 0, 1, 1, 'LEFT'), + (261, 261, 0, 8, null, 0, 1, 1, 1, 'LEFT'), + (262, 262, 0, 8, null, 0, 2, 1, 1, 'LEFT'), + (263, 263, 0, 8, null, 0, 3, 1, 1, 'LEFT'), + (264, 264, 0, 8, null, 0, 4, 1, 1, 'LEFT'), + (265, 265, 0, 8, null, 0, 5, 1, 1, 'LEFT'), + + + (300, 300, 0, 9, 'servicePolicy', 0, 0, 4, 3, 'NONE'), + (301, 301, 0, 9, 'kioskMode', 4, 0, 3, 3, 'NONE'), + (302, 302, 0, 9, null, 0, 5, 4, 1, 'NONE'), + (303, 303, 0, 9, null, 0, 6, 4, 1, 'NONE'), + (304, 304, 0, 9, null, 4, 5, 3, 1, 'NONE'), + (305, 305, 0, 9, 'logging', 0, 8, 6, 1, 'NONE'), + (306, 306, 0, 9, 'logging', 3, 9, 4, 1, 'LEFT_SPAN'), + (307, 307, 0, 9, 'logging', 3, 10, 4, 1, 'LEFT_SPAN'), + (308, 308, 0, 9, 'macSettings', 7, 1, 5, 1, 'TOP'), + (309, 309, 0, 9, 'macSettings', 7, 2, 5, 1, 'NONE'), + (310, 310, 0, 9, 'macSettings', 7, 3, 5, 1, 'NONE'), + (311, 311, 0, 9, 'macSettings', 7, 4, 5, 1, 'NONE'), + (312, 312, 0, 9, 'macSettings', 7, 5, 5, 1, 'NONE'), + (313, 313, 0, 9, 'macSettings', 7, 6, 5, 1, 'NONE'), + (314, 314, 0, 9, 'macSettings', 7, 7, 5, 1, 'NONE'), + (315, 315, 0, 9, 'macSettings', 7, 9, 5, 1, 'TOP'), + (316, 316, 0, 9, 'macSettings', 7, 10, 5, 1, 'NONE'), + (317, 317, 0, 9, 'logging', 3, 11, 4, 1, 'LEFT_SPAN'), + + (400, 400, 0, 10, 'registry', 0, 1, 4, 1, 'NONE'), + (401, 401, 0, 10, 'registry', 0, 2, 4, 1, 'NONE'), + (402, 402, 0, 10, 'registry', 0, 3, 4, 1, 'NONE'), + (403, 403, 0, 10, 'registry', 0, 4, 4, 1, 'NONE'), + (404, 404, 0, 10, 'registry', 0, 5, 4, 1, 'NONE'), + (405, 405, 0, 10, 'registry', 0, 6, 4, 1, 'NONE'), + (406, 406, 0, 10, 'registry', 0, 7, 4, 1, 'NONE'), + (407, 407, 0, 10, 'registry', 0, 8, 4, 1, 'NONE'), + (408, 408, 0, 10, 'registry', 0, 9, 4, 1, 'NONE'), + + (500, 500, 0, 11, 'specialKeys', 0, 1, 3, 1, 'NONE'), + (501, 501, 0, 11, 'specialKeys', 0, 2, 3, 1, 'NONE'), + (502, 502, 0, 11, 'specialKeys', 0, 3, 3, 1, 'NONE'), + (503, 503, 0, 11, 'specialKeys', 0, 4, 3, 1, 'NONE'), + (504, 504, 0, 11, 'specialKeys', 0, 5, 3, 1, 'NONE'), + (505, 505, 0, 11, 'specialKeys', 0, 6, 3, 1, 'NONE'), + (506, 506, 0, 11, 'specialKeys', 0, 7, 3, 1, 'NONE'), + (507, 507, 0, 11, 'specialKeys', 0, 8, 3, 1, 'NONE'), + (508, 508, 0, 11, 'specialKeys', 0, 9, 3, 1, 'NONE'), + + (509, 509, 0, 11, 'functionKeys', 3, 1, 3, 1, 'NONE'), + (510, 510, 0, 11, 'functionKeys', 3, 2, 3, 1, 'NONE'), + (511, 511, 0, 11, 'functionKeys', 3, 3, 3, 1, 'NONE'), + (512, 512, 0, 11, 'functionKeys', 3, 4, 3, 1, 'NONE'), + (513, 513, 0, 11, 'functionKeys', 3, 5, 3, 1, 'NONE'), + (514, 514, 0, 11, 'functionKeys', 3, 6, 3, 1, 'NONE'), + (515, 515, 0, 11, 'functionKeys', 3, 7, 3, 1, 'NONE'), + (516, 516, 0, 11, 'functionKeys', 3, 8, 3, 1, 'NONE'), + (517, 517, 0, 11, 'functionKeys', 3, 9, 3, 1, 'NONE'), + (518, 518, 0, 11, 'functionKeys', 3, 10, 3, 1, 'NONE'), + (519, 519, 0, 11, 'functionKeys', 3, 11, 3, 1, 'NONE'), + (520, 520, 0, 11, 'functionKeys', 3, 12, 3, 1, 'NONE') + + ; + + + + + + +INSERT IGNORE INTO configuration_node VALUES + (1, 1, 0, 'super-admin', 'test', null, 'EXAM_CONFIG', 'READY_TO_USE') + ; + +INSERT IGNORE INTO configuration VALUES + (1, 1, 1, 'v0', '2019-07-02 12:59:32', 0), + (2, 1, 1, null, null, 1) + ; + +INSERT IGNORE INTO configuration_value VALUES + (1,1,1,1,0,NULL), + (2,1,1,2,0,'true'), + (3,1,1,3,0,'false'), + (4,1,1,4,0,NULL), + (5,1,1,5,0,'2'), + (6,1,1,6,0,'10'), + (7,1,1,7,0,'5'), + (8,1,1,8,0,'0'), + (9,1,1,9,0,'false'), + (10,1,1,10,0,'100%'), + (11,1,1,11,0,'100%'), + (12,1,1,12,0,'1'), + (13,1,1,13,0,'false'), + (14,1,1,14,0,'false'), + (15,1,1,15,0,'false'), + (16,1,1,16,0,'true'), + (17,1,1,17,0,'40'), + (18,1,1,18,0,'true'), + (19,1,1,19,0,'true'), + (20,1,1,20,0,'false'), + (21,1,1,21,0,'true'), + (22,1,1,22,0,'true'), + (23,1,1,23,0,'0'), + (24,1,1,24,0,'false'), + (25,1,1,25,0,'false'), + (26,1,1,26,0,'false'), + (27,1,1,27,0,'25'), + (28,1,1,28,0,'false'), + (29,1,1,29,0,'false'), + (30,1,1,30,0,'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), + (31,1,1,31,0,'2'), + (32,1,1,32,0,'false'), + (33,1,1,33,0,'100%'), + (34,1,1,34,0,'100%'), + (35,1,1,35,0,'2'), + (36,1,1,36,0,'true'), + (37,1,1,37,0,'true'), + (38,1,1,38,0,'false'), + (39,1,1,39,0,'false'), + (40,1,1,40,0,'false'), + (41,1,1,41,0,'false'), + (42,1,1,42,0,'false'), + (43,1,1,43,0,'true'), + (44,1,1,44,0,'true'), + (45,1,1,45,0,'true'), + (46,1,1,46,0,'true'), + (47,1,1,47,0,'false'), + (48,1,1,48,0,'false'), + (49,1,1,49,0,'false'), + (50,1,1,50,0,NULL), + (51,1,1,51,0,'0'), + (52,1,1,52,0,NULL), + (53,1,1,53,0,'0'), + (54,1,1,54,0,NULL), + (55,1,1,55,0,'0'), + (56,1,1,56,0,NULL), + (57,1,1,57,0,'true'), + (58,1,1,58,0,NULL), + (59,1,1,59,0,'true'), + (60,1,1,60,0,NULL), + (61,1,1,61,0,NULL), + (62,1,1,62,0,'false'), + (63,1,1,63,0,'0'), + (64,1,1,64,0,'true'), + (65,1,1,65,0,'true'), + (66,1,1,66,0,'true'), + (67,1,1,67,0,NULL), + (68,1,1,68,0,'true'), + (69,1,1,69,0,'false'), + (70,1,1,70,0,NULL), + (71,1,1,71,0,NULL), + (72,1,1,72,0,'true'), + (73,1,1,73,0,NULL), + (74,1,1,91,0,'false'), + (75,1,1,92,0,'false'), + (76,1,1,93,0,NULL), + (77,1,1,200,0,'false'), + (78,1,1,201,0,'false'), + (79,1,1,202,0,NULL), + (80,1,1,210,0,'0'), + (81,1,1,220,0,NULL), + (82,1,1,221,0,'false'), + (83,1,1,222,0,NULL), + (84,1,1,223,0,'true'), + (85,1,1,231,0,'false'), + (86,1,1,233,0,'false'), + (87,1,1,234,0,NULL), + (88,1,1,235,0,NULL), + (89,1,1,236,0,'false'), + (90,1,1,237,0,NULL), + (91,1,1,238,0,'80'), + (92,1,1,239,0,'false'), + (93,1,1,240,0,NULL), + (94,1,1,241,0,NULL), + (95,1,1,242,0,'false'), + (96,1,1,243,0,NULL), + (97,1,1,244,0,'443'), + (98,1,1,245,0,'false'), + (99,1,1,246,0,NULL), + (100,1,1,247,0,NULL), + (101,1,1,248,0,'false'), + (102,1,1,249,0,NULL), + (103,1,1,250,0,'21'), + (104,1,1,251,0,'false'), + (105,1,1,252,0,NULL), + (106,1,1,253,0,NULL), + (107,1,1,254,0,'false'), + (108,1,1,255,0,NULL), + (109,1,1,256,0,'1080'), + (110,1,1,257,0,'false'), + (111,1,1,258,0,NULL), + (112,1,1,259,0,NULL), + (113,1,1,260,0,'false'), + (114,1,1,261,0,NULL), + (115,1,1,262,0,'1080'), + (116,1,1,263,0,'false'), + (117,1,1,264,0,NULL), + (118,1,1,265,0,NULL), + (119,1,1,300,0,'2'), + (120,1,1,301,0,'0'), + (121,1,1,302,0,'false'), + (122,1,1,303,0,'false'), + (123,1,1,304,0,'true'), + (124,1,1,305,0,'false'), + (125,1,1,306,0,''), + (126,1,1,307,0,'~/Documents'), + (127,1,1,308,0,'0'), + (128,1,1,309,0,'true'), + (129,1,1,310,0,'true'), + (130,1,1,311,0,'false'), + (131,1,1,312,0,'false'), + (132,1,1,313,0,'true'), + (133,1,1,314,0,'false'), + (134,1,1,315,0,'1'), + (135,1,1,316,0,'true'), + (136,1,1,400,0,'false'), + (137,1,1,401,0,'false'), + (138,1,1,402,0,'false'), + (139,1,1,403,0,'false'), + (140,1,1,404,0,'false'), + (141,1,1,405,0,'false'), + (142,1,1,406,0,'false'), + (143,1,1,407,0,'false'), + (144,1,1,408,0,'false'), + (145,1,1,500,0,'false'), + (146,1,1,501,0,'false'), + (147,1,1,502,0,'false'), + (148,1,1,503,0,'false'), + (149,1,1,504,0,'true'), + (150,1,1,505,0,'false'), + (151,1,1,506,0,'false'), + (152,1,1,507,0,'false'), + (153,1,1,508,0,'false'), + (154,1,1,509,0,'false'), + (155,1,1,510,0,'false'), + (156,1,1,511,0,'false'), + (157,1,1,512,0,'false'), + (158,1,1,513,0,'false'), + (159,1,1,514,0,'false'), + (160,1,1,515,0,'false'), + (161,1,1,516,0,'false'), + (162,1,1,517,0,'false'), + (163,1,1,518,0,'false'), + (164,1,1,519,0,'false'), + (165,1,1,520,0,'false'), + (166,1,1,1000,0,'SEB_Server_0.3.0'), + (167,1,1,1001,0,'0'), + (168,1,2,1,0,NULL), + (169,1,2,2,0,'true'), + (170,1,2,3,0,'false'), + (171,1,2,4,0,NULL), + (172,1,2,5,0,'2'), + (173,1,2,6,0,'10'), + (174,1,2,7,0,'5'), + (175,1,2,8,0,'0'), + (176,1,2,9,0,'false'), + (177,1,2,10,0,'100%'), + (178,1,2,11,0,'100%'), + (179,1,2,12,0,'1'), + (180,1,2,13,0,'false'), + (181,1,2,14,0,'false'), + (182,1,2,15,0,'false'), + (183,1,2,16,0,'true'), + (184,1,2,17,0,'40'), + (185,1,2,18,0,'true'), + (186,1,2,19,0,'true'), + (187,1,2,20,0,'false'), + (188,1,2,21,0,'true'), + (189,1,2,22,0,'true'), + (190,1,2,23,0,'0'), + (191,1,2,24,0,'false'), + (192,1,2,25,0,'false'), + (193,1,2,26,0,'false'), + (194,1,2,27,0,'25'), + (195,1,2,28,0,'false'), + (196,1,2,29,0,'false'), + (197,1,2,30,0,'da-DK,en-AU,en-GB,en-US,es-ES,fr-FR,pt-PT,sv-SE,sv-FI'), + (198,1,2,31,0,'2'), + (199,1,2,32,0,'false'), + (200,1,2,33,0,'100%'), + (201,1,2,34,0,'100%'), + (202,1,2,35,0,'2'), + (203,1,2,36,0,'true'), + (204,1,2,37,0,'true'), + (205,1,2,38,0,'false'), + (206,1,2,39,0,'false'), + (207,1,2,40,0,'false'), + (208,1,2,41,0,'false'), + (209,1,2,42,0,'false'), + (210,1,2,43,0,'true'), + (211,1,2,44,0,'true'), + (212,1,2,45,0,'true'), + (213,1,2,46,0,'true'), + (214,1,2,47,0,'false'), + (215,1,2,48,0,'false'), + (216,1,2,49,0,'false'), + (217,1,2,50,0,NULL), + (218,1,2,51,0,'0'), + (219,1,2,52,0,NULL), + (220,1,2,53,0,'0'), + (221,1,2,54,0,NULL), + (222,1,2,55,0,'0'), + (223,1,2,56,0,NULL), + (224,1,2,57,0,'true'), + (225,1,2,58,0,NULL), + (226,1,2,59,0,'true'), + (227,1,2,60,0,NULL), + (228,1,2,61,0,NULL), + (229,1,2,62,0,'false'), + (230,1,2,63,0,'0'), + (231,1,2,64,0,'true'), + (232,1,2,65,0,'true'), + (233,1,2,66,0,'true'), + (234,1,2,67,0,NULL), + (235,1,2,68,0,'true'), + (236,1,2,69,0,'false'), + (237,1,2,70,0,NULL), + (238,1,2,71,0,NULL), + (239,1,2,72,0,'true'), + (240,1,2,73,0,NULL), + (241,1,2,91,0,'false'), + (242,1,2,92,0,'false'), + (243,1,2,93,0,NULL), + (244,1,2,200,0,'false'), + (245,1,2,201,0,'false'), + (246,1,2,202,0,NULL), + (247,1,2,210,0,'0'), + (248,1,2,220,0,NULL), + (249,1,2,221,0,'false'), + (250,1,2,222,0,NULL), + (251,1,2,223,0,'true'), + (252,1,2,231,0,'false'), + (253,1,2,233,0,'false'), + (254,1,2,234,0,NULL), + (255,1,2,235,0,NULL), + (256,1,2,236,0,'false'), + (257,1,2,237,0,NULL), + (258,1,2,238,0,'80'), + (259,1,2,239,0,'false'), + (260,1,2,240,0,NULL), + (261,1,2,241,0,NULL), + (262,1,2,242,0,'false'), + (263,1,2,243,0,NULL), + (264,1,2,244,0,'443'), + (265,1,2,245,0,'false'), + (266,1,2,246,0,NULL), + (267,1,2,247,0,NULL), + (268,1,2,248,0,'false'), + (269,1,2,249,0,NULL), + (270,1,2,250,0,'21'), + (271,1,2,251,0,'false'), + (272,1,2,252,0,NULL), + (273,1,2,253,0,NULL), + (274,1,2,254,0,'false'), + (275,1,2,255,0,NULL), + (276,1,2,256,0,'1080'), + (277,1,2,257,0,'false'), + (278,1,2,258,0,NULL), + (279,1,2,259,0,NULL), + (280,1,2,260,0,'false'), + (281,1,2,261,0,NULL), + (282,1,2,262,0,'1080'), + (283,1,2,263,0,'false'), + (284,1,2,264,0,NULL), + (285,1,2,265,0,NULL), + (286,1,2,300,0,'2'), + (287,1,2,301,0,'0'), + (288,1,2,302,0,'false'), + (289,1,2,303,0,'false'), + (290,1,2,304,0,'true'), + (291,1,2,305,0,'false'), + (292,1,2,306,0,''), + (293,1,2,307,0,'~/Documents'), + (294,1,2,308,0,'0'), + (295,1,2,309,0,'true'), + (296,1,2,310,0,'true'), + (297,1,2,311,0,'false'), + (298,1,2,312,0,'false'), + (299,1,2,313,0,'true'), + (300,1,2,314,0,'false'), + (301,1,2,315,0,'1'), + (302,1,2,316,0,'true'), + (303,1,2,400,0,'false'), + (304,1,2,401,0,'false'), + (305,1,2,402,0,'false'), + (306,1,2,403,0,'false'), + (307,1,2,404,0,'false'), + (308,1,2,405,0,'false'), + (309,1,2,406,0,'false'), + (310,1,2,407,0,'false'), + (311,1,2,408,0,'false'), + (312,1,2,500,0,'false'), + (313,1,2,501,0,'false'), + (314,1,2,502,0,'false'), + (315,1,2,503,0,'false'), + (316,1,2,504,0,'true'), + (317,1,2,505,0,'false'), + (318,1,2,506,0,'false'), + (319,1,2,507,0,'false'), + (320,1,2,508,0,'false'), + (321,1,2,509,0,'false'), + (322,1,2,510,0,'false'), + (323,1,2,511,0,'false'), + (324,1,2,512,0,'false'), + (325,1,2,513,0,'false'), + (326,1,2,514,0,'false'), + (327,1,2,515,0,'false'), + (328,1,2,516,0,'false'), + (329,1,2,517,0,'false'), + (330,1,2,518,0,'false'), + (331,1,2,519,0,'false'), + (332,1,2,520,0,'false'), + (333,1,2,1000,0,'SEB_Server_0.3.0'), + (334,1,2,1001,0,'0') + ; + +INSERT IGNORE INTO exam_configuration_map VALUES + (1, 1, 2, 1, null, null) + ; +