Merge remote-tracking branch 'origin/dev-1.2' into development
Conflicts: src/main/java/ch/ethz/seb/sebserver/gbl/util/Utils.java src/main/java/ch/ethz/seb/sebserver/gui/content/ProctorRoomConnectionsPopup.java src/main/java/ch/ethz/seb/sebserver/gui/content/SEBClientEvents.java src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/SEBClientConnectionServiceImpl.java src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java
This commit is contained in:
commit
9896258afc
45 changed files with 1161 additions and 155 deletions
|
@ -24,6 +24,8 @@ import ch.ethz.seb.sebserver.gbl.api.authorization.Privilege;
|
||||||
/** Global Constants used in SEB Server web-service as well as in web-gui component */
|
/** Global Constants used in SEB Server web-service as well as in web-gui component */
|
||||||
public final class Constants {
|
public final class Constants {
|
||||||
|
|
||||||
|
public static final String FILE_EXT_CSV = ".csv";
|
||||||
|
|
||||||
public static final String DEFAULT_LANG_CODE = "en";
|
public static final String DEFAULT_LANG_CODE = "en";
|
||||||
public static final String DEFAULT_TIME_ZONE_CODE = "UTC";
|
public static final String DEFAULT_TIME_ZONE_CODE = "UTC";
|
||||||
public static final String TOOLTIP_TEXT_KEY_SUFFIX = ".tooltip";
|
public static final String TOOLTIP_TEXT_KEY_SUFFIX = ".tooltip";
|
||||||
|
|
|
@ -208,6 +208,10 @@ public final class API {
|
||||||
|
|
||||||
public static final String SEB_CLIENT_EVENT_ENDPOINT = "/seb-client-event";
|
public static final String SEB_CLIENT_EVENT_ENDPOINT = "/seb-client-event";
|
||||||
public static final String SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT = "/search";
|
public static final String SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT = "/search";
|
||||||
|
public static final String SEB_CLIENT_EVENT_EXPORT_PATH_SEGMENT = "/export";
|
||||||
|
public static final String SEB_CLIENT_EVENT_EXPORT_TYPE = "exportType";
|
||||||
|
public static final String SEB_CLIENT_EVENT_EXPORT_INCLUDE_CONNECTIONS = "includeConnectionDetails";
|
||||||
|
public static final String SEB_CLIENT_EVENT_EXPORT_INCLUDE_EXAMS = "includeExamDetails";
|
||||||
public static final String SEB_CLIENT_EVENT_EXTENDED_PAGE_ENDPOINT = SEB_CLIENT_EVENT_ENDPOINT
|
public static final String SEB_CLIENT_EVENT_EXTENDED_PAGE_ENDPOINT = SEB_CLIENT_EVENT_ENDPOINT
|
||||||
+ SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT;
|
+ SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT;
|
||||||
|
|
||||||
|
|
|
@ -128,9 +128,9 @@ public final class Indicator implements Entity {
|
||||||
this.thresholds = Utils.immutableListOf(thresholds);
|
this.thresholds = Utils.immutableListOf(thresholds);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Indicator(final Exam exam, final POSTMapper postParams) {
|
public Indicator(final Long examId, final POSTMapper postParams) {
|
||||||
this.id = null;
|
this.id = null;
|
||||||
this.examId = exam.id;
|
this.examId = examId;
|
||||||
this.name = postParams.getString(Domain.INDICATOR.ATTR_NAME);
|
this.name = postParams.getString(Domain.INDICATOR.ATTR_NAME);
|
||||||
this.type = postParams.getEnum(Domain.INDICATOR.ATTR_TYPE, IndicatorType.class);
|
this.type = postParams.getEnum(Domain.INDICATOR.ATTR_TYPE, IndicatorType.class);
|
||||||
this.defaultColor = postParams.getString(Domain.INDICATOR.ATTR_COLOR);
|
this.defaultColor = postParams.getString(Domain.INDICATOR.ATTR_COLOR);
|
||||||
|
|
|
@ -68,6 +68,10 @@ public class ClientEvent implements Entity, IndicatorValueHolder {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public enum ExportType {
|
||||||
|
CSV
|
||||||
|
}
|
||||||
|
|
||||||
@JsonProperty(Domain.CLIENT_EVENT.ATTR_ID)
|
@JsonProperty(Domain.CLIENT_EVENT.ATTR_ID)
|
||||||
public final Long id;
|
public final Long id;
|
||||||
|
|
||||||
|
|
|
@ -262,6 +262,13 @@ public final class Result<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Result<T> onSuccess(final Consumer<T> handler) {
|
||||||
|
if (this.error == null) {
|
||||||
|
handler.accept(this.value);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
/** Uses a given error handler to apply an error if there is one and returning itself again
|
/** Uses a given error handler to apply an error if there is one and returning itself again
|
||||||
* for further processing.
|
* for further processing.
|
||||||
*
|
*
|
||||||
|
|
|
@ -673,4 +673,20 @@ public final class Utils {
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static CharSequence trim(final CharSequence sequence) {
|
||||||
|
if (sequence == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return StringUtils.trim(sequence.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String toCSVString(final String text) {
|
||||||
|
if (StringUtils.isBlank(text)) {
|
||||||
|
return StringUtils.EMPTY;
|
||||||
|
}
|
||||||
|
return Constants.DOUBLE_QUOTE + text.replace("\"", "\"\"") + Constants.DOUBLE_QUOTE;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -835,6 +835,10 @@ public enum ActionDefinition {
|
||||||
ImageIcon.DELETE,
|
ImageIcon.DELETE,
|
||||||
PageStateDefinitionImpl.SEB_CLIENT_LOGS,
|
PageStateDefinitionImpl.SEB_CLIENT_LOGS,
|
||||||
ActionCategory.LOGS_SEB_CLIENT_LIST),
|
ActionCategory.LOGS_SEB_CLIENT_LIST),
|
||||||
|
LOGS_SEB_CLIENT_EXPORT_CSV(
|
||||||
|
new LocTextKey("sebserver.seblogs.action.export.csv"),
|
||||||
|
ImageIcon.EXPORT,
|
||||||
|
ActionCategory.LOGS_SEB_CLIENT_LIST),
|
||||||
|
|
||||||
;
|
;
|
||||||
|
|
||||||
|
|
|
@ -96,7 +96,7 @@ public class SEBClientEventDetailsPopup {
|
||||||
this.i18nSupport = pageService.getI18nSupport();
|
this.i18nSupport = pageService.getI18nSupport();
|
||||||
}
|
}
|
||||||
|
|
||||||
PageAction showDetails(final PageAction action, final ExtendedClientEvent clientEvent) {
|
public PageAction showDetails(final PageAction action, final ExtendedClientEvent clientEvent) {
|
||||||
action.getSingleSelection();
|
action.getSingleSelection();
|
||||||
|
|
||||||
final ModalInputDialog<Void> dialog = new ModalInputDialog<>(
|
final ModalInputDialog<Void> dialog = new ModalInputDialog<>(
|
||||||
|
|
|
@ -40,17 +40,8 @@ public abstract class AbstractDownloadServiceHandler implements DownloadServiceH
|
||||||
log.debug("download requested... trying to get needed parameter from request");
|
log.debug("download requested... trying to get needed parameter from request");
|
||||||
|
|
||||||
final String modelId = request.getParameter(API.PARAM_MODEL_ID);
|
final String modelId = request.getParameter(API.PARAM_MODEL_ID);
|
||||||
if (StringUtils.isBlank(modelId)) {
|
|
||||||
log.error(
|
|
||||||
"Mandatory modelId parameter not found within HttpServletRequest. Download request is ignored");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug(
|
log.debug("Found modelId: {} for {} download.", modelId);
|
||||||
"Found modelId: {} for {} download. Trying to request webservice...",
|
|
||||||
modelId,
|
|
||||||
downloadFileName);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
final String parentModelId = request.getParameter(API.PARAM_PARENT_MODEL_ID);
|
final String parentModelId = request.getParameter(API.PARAM_PARENT_MODEL_ID);
|
||||||
|
@ -67,7 +58,7 @@ public abstract class AbstractDownloadServiceHandler implements DownloadServiceH
|
||||||
response.setHeader(HttpHeaders.CONTENT_DISPOSITION, header);
|
response.setHeader(HttpHeaders.CONTENT_DISPOSITION, header);
|
||||||
response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
|
response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
|
||||||
|
|
||||||
webserviceCall(modelId, parentModelId, response.getOutputStream());
|
webserviceCall(modelId, parentModelId, response.getOutputStream(), request);
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error(
|
log.error(
|
||||||
|
@ -76,6 +67,10 @@ public abstract class AbstractDownloadServiceHandler implements DownloadServiceH
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract void webserviceCall(String modelId, String parentModelId, OutputStream downloadOut);
|
protected abstract void webserviceCall(
|
||||||
|
String modelId,
|
||||||
|
String parentModelId,
|
||||||
|
OutputStream downloadOut,
|
||||||
|
HttpServletRequest request);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,9 +8,14 @@
|
||||||
|
|
||||||
package ch.ethz.seb.sebserver.gui.service.remote.download;
|
package ch.ethz.seb.sebserver.gui.service.remote.download;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.Constants;
|
import java.util.Collection;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.API;
|
import java.util.Map;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
import java.util.function.Function;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.eclipse.rap.rwt.RWT;
|
import org.eclipse.rap.rwt.RWT;
|
||||||
import org.eclipse.rap.rwt.service.ServiceHandler;
|
import org.eclipse.rap.rwt.service.ServiceHandler;
|
||||||
|
@ -19,12 +24,10 @@ import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.context.annotation.Lazy;
|
import org.springframework.context.annotation.Lazy;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import ch.ethz.seb.sebserver.gbl.Constants;
|
||||||
import javax.servlet.http.HttpServletResponse;
|
import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
import java.util.Collection;
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
import java.util.Map;
|
import ch.ethz.seb.sebserver.gbl.util.Utils;
|
||||||
import java.util.function.Function;
|
|
||||||
import java.util.stream.Collectors;
|
|
||||||
|
|
||||||
/** Implements a eclipse RAP ServiceHandler to handle downloads */
|
/** Implements a eclipse RAP ServiceHandler to handle downloads */
|
||||||
@Lazy
|
@Lazy
|
||||||
|
@ -73,6 +76,33 @@ public class DownloadService implements ServiceHandler {
|
||||||
.processDownload(request, response);
|
.processDownload(request, response);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String createDownloadURL(
|
||||||
|
final Class<? extends DownloadServiceHandler> handlerClass,
|
||||||
|
final String downloadFileName,
|
||||||
|
final Map<String, String> queryAttrs) {
|
||||||
|
|
||||||
|
final StringBuilder url = new StringBuilder()
|
||||||
|
.append(RWT.getServiceManager()
|
||||||
|
.getServiceHandlerUrl(DownloadService.DOWNLOAD_SERVICE_NAME))
|
||||||
|
.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);
|
||||||
|
|
||||||
|
queryAttrs.forEach((name, value) -> {
|
||||||
|
url.append(Constants.FORM_URL_ENCODED_SEPARATOR)
|
||||||
|
.append(name)
|
||||||
|
.append(Constants.FORM_URL_ENCODED_NAME_VALUE_SEPARATOR)
|
||||||
|
.append(Utils.encodeFormURL_UTF_8(value));
|
||||||
|
});
|
||||||
|
|
||||||
|
return url.toString();
|
||||||
|
}
|
||||||
|
|
||||||
public String createDownloadURL(
|
public String createDownloadURL(
|
||||||
final String modelId,
|
final String modelId,
|
||||||
final Class<? extends DownloadServiceHandler> handlerClass,
|
final Class<? extends DownloadServiceHandler> handlerClass,
|
||||||
|
|
|
@ -12,6 +12,8 @@ import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -35,14 +37,16 @@ public class SEBClientConfigDownload extends AbstractDownloadServiceHandler {
|
||||||
|
|
||||||
private final RestService restService;
|
private final RestService restService;
|
||||||
|
|
||||||
protected SEBClientConfigDownload(
|
protected SEBClientConfigDownload(final RestService restService) {
|
||||||
final RestService restService) {
|
|
||||||
|
|
||||||
this.restService = restService;
|
this.restService = restService;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) {
|
protected void webserviceCall(
|
||||||
|
final String modelId,
|
||||||
|
final String parentModelId,
|
||||||
|
final OutputStream downloadOut,
|
||||||
|
final HttpServletRequest request) {
|
||||||
|
|
||||||
final RestCall<InputStream>.RestCallBuilder restCallBuilder = this.restService
|
final RestCall<InputStream>.RestCallBuilder restCallBuilder = this.restService
|
||||||
.getBuilder(ExportClientConfig.class)
|
.getBuilder(ExportClientConfig.class)
|
||||||
|
|
|
@ -0,0 +1,80 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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 java.util.Enumeration;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
|
||||||
|
import org.apache.commons.io.IOUtils;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.context.annotation.Lazy;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
import org.springframework.util.LinkedMultiValueMap;
|
||||||
|
import org.springframework.util.MultiValueMap;
|
||||||
|
|
||||||
|
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.ExportSEBClientLogs;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Component
|
||||||
|
@GuiProfile
|
||||||
|
public class SEBClientLogExport extends AbstractDownloadServiceHandler {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(SEBClientLogExport.class);
|
||||||
|
|
||||||
|
private final RestService restService;
|
||||||
|
|
||||||
|
protected SEBClientLogExport(final RestService restService) {
|
||||||
|
this.restService = restService;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void webserviceCall(
|
||||||
|
final String modelId,
|
||||||
|
final String parentModelId,
|
||||||
|
final OutputStream downloadOut,
|
||||||
|
final HttpServletRequest request) {
|
||||||
|
|
||||||
|
final MultiValueMap<String, String> queryParams = new LinkedMultiValueMap<>();
|
||||||
|
|
||||||
|
final Enumeration<String> paramNames = request.getParameterNames();
|
||||||
|
while (paramNames.hasMoreElements()) {
|
||||||
|
final String param = paramNames.nextElement();
|
||||||
|
queryParams.add(param, String.valueOf(request.getParameter(param)));
|
||||||
|
}
|
||||||
|
|
||||||
|
final InputStream input = this.restService
|
||||||
|
.getBuilder(ExportSEBClientLogs.class)
|
||||||
|
.withQueryParams(queryParams)
|
||||||
|
.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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -12,6 +12,8 @@ import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
@ -37,7 +39,11 @@ public class SEBExamConfigDownload extends AbstractDownloadServiceHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) {
|
protected void webserviceCall(
|
||||||
|
final String modelId,
|
||||||
|
final String parentModelId,
|
||||||
|
final OutputStream downloadOut,
|
||||||
|
final HttpServletRequest request) {
|
||||||
|
|
||||||
final InputStream input = this.restService.getBuilder(ExportExamConfig.class)
|
final InputStream input = this.restService.getBuilder(ExportExamConfig.class)
|
||||||
.withURIVariable(API.PARAM_MODEL_ID, modelId)
|
.withURIVariable(API.PARAM_MODEL_ID, modelId)
|
||||||
|
|
|
@ -12,6 +12,8 @@ import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
@ -37,7 +39,11 @@ public class SEBExamConfigPlaintextDownload extends AbstractDownloadServiceHandl
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void webserviceCall(final String modelId, final String parentModelId, final OutputStream downloadOut) {
|
protected void webserviceCall(
|
||||||
|
final String modelId,
|
||||||
|
final String parentModelId,
|
||||||
|
final OutputStream downloadOut,
|
||||||
|
final HttpServletRequest request) {
|
||||||
|
|
||||||
final InputStream input = this.restService.getBuilder(ExportPlainXML.class)
|
final InputStream input = this.restService.getBuilder(ExportPlainXML.class)
|
||||||
.withURIVariable(API.PARAM_MODEL_ID, modelId)
|
.withURIVariable(API.PARAM_MODEL_ID, modelId)
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
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.profile.GuiProfile;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.AbstractExportCall;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Component
|
||||||
|
@GuiProfile
|
||||||
|
public class ExportSEBClientLogs extends AbstractExportCall {
|
||||||
|
|
||||||
|
public ExportSEBClientLogs() {
|
||||||
|
super(new TypeKey<>(
|
||||||
|
CallType.UNDEFINED,
|
||||||
|
EntityType.CLIENT_EVENT,
|
||||||
|
new TypeReference<InputStream>() {
|
||||||
|
}),
|
||||||
|
HttpMethod.GET,
|
||||||
|
MediaType.APPLICATION_FORM_URLENCODED,
|
||||||
|
API.SEB_CLIENT_EVENT_ENDPOINT
|
||||||
|
+ API.SEB_CLIENT_EVENT_EXPORT_PATH_SEGMENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -243,6 +243,14 @@ public class EntityTable<ROW> {
|
||||||
return this.name;
|
return this.name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getSortColumn() {
|
||||||
|
return this.sortColumn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PageSortOrder getSortOrder() {
|
||||||
|
return this.sortOrder;
|
||||||
|
}
|
||||||
|
|
||||||
public EntityType getEntityType() {
|
public EntityType getEntityType() {
|
||||||
if (this.pageSupplier != null) {
|
if (this.pageSupplier != null) {
|
||||||
return this.pageSupplier.getEntityType();
|
return this.pageSupplier.getEntityType();
|
||||||
|
|
|
@ -106,8 +106,17 @@ public class StaticListPageSupplier<T> implements PageSupplier<T> {
|
||||||
if (numOfPages <= 0) {
|
if (numOfPages <= 0) {
|
||||||
return new Page<>(1, 1, this.column, this.list);
|
return new Page<>(1, 1, this.column, this.list);
|
||||||
}
|
}
|
||||||
final List<T> subList = this.list.subList(this.pageNumber * this.pageSize,
|
|
||||||
this.pageNumber * this.pageSize + this.pageSize);
|
int from = (this.pageNumber - 1) * this.pageSize;
|
||||||
|
if (from < 0) {
|
||||||
|
from = 0;
|
||||||
|
}
|
||||||
|
int to = (this.pageNumber - 1) * this.pageSize + this.pageSize;
|
||||||
|
if (to >= this.list.size()) {
|
||||||
|
to = this.list.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<T> subList = this.list.subList(from, to);
|
||||||
return new Page<>(numOfPages, this.pageNumber, this.column, subList);
|
return new Page<>(numOfPages, this.pageNumber, this.column, subList);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,6 +77,30 @@ public interface PaginationService {
|
||||||
final String tableName,
|
final String tableName,
|
||||||
final Supplier<Result<Collection<T>>> delegate);
|
final Supplier<Result<Collection<T>>> delegate);
|
||||||
|
|
||||||
|
/** Fetches a paged batch of objects
|
||||||
|
*
|
||||||
|
* NOTE: Paging always depends on SQL level. It depends on the collection given by the SQL select statement
|
||||||
|
* that is executed within MyBatis by using the MyBatis page service.
|
||||||
|
* Be aware that if the delegate that is given here applies an additional filter to the filtering done
|
||||||
|
* on SQL level, this will lead to paging with not fully filled pages or even to empty pages if the filter
|
||||||
|
* filters a lot of the entries given by the SQL statement away.
|
||||||
|
* So we recommend to apply as much of the filtering as possible on the SQL level and only if necessary and
|
||||||
|
* not avoidable, apply a additional filter on software-level that eventually filter one or two entities
|
||||||
|
* for a page.
|
||||||
|
*
|
||||||
|
* @param pageNumber the current page number
|
||||||
|
* @param pageSize the (full) size of the page
|
||||||
|
* @param sort the name of the sort column with a leading '-' for descending sort order
|
||||||
|
* @param tableName the name of the SQL table on which the pagination is applying to
|
||||||
|
* @param delegate a collection supplier the does the underling SQL query with specified pagination attributes
|
||||||
|
* @return Result refers to a Collection of specified type of objects or to an exception on error case */
|
||||||
|
<T> Result<Page<T>> getPageOf(
|
||||||
|
final Integer pageNumber,
|
||||||
|
final Integer pageSize,
|
||||||
|
final String sort,
|
||||||
|
final String tableName,
|
||||||
|
final Supplier<Result<Collection<T>>> delegate);
|
||||||
|
|
||||||
/** Use this to build a current Page from a given list of objects.
|
/** Use this to build a current Page from a given list of objects.
|
||||||
*
|
*
|
||||||
* @param <T> the Type if list entities
|
* @param <T> the Type if list entities
|
||||||
|
|
|
@ -147,6 +147,29 @@ public class PaginationServiceImpl implements PaginationService {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T> Result<Page<T>> getPageOf(
|
||||||
|
final Integer pageNumber,
|
||||||
|
final Integer pageSize,
|
||||||
|
final String sort,
|
||||||
|
final String tableName,
|
||||||
|
final Supplier<Result<Collection<T>>> delegate) {
|
||||||
|
|
||||||
|
return Result.tryCatch(() -> {
|
||||||
|
//final SqlTable table = SqlTable.of(tableName);
|
||||||
|
final com.github.pagehelper.Page<Object> page =
|
||||||
|
setPagination(pageNumber, pageSize, sort, tableName);
|
||||||
|
|
||||||
|
final Collection<T> list = delegate.get().getOrThrow();
|
||||||
|
|
||||||
|
return new Page<>(
|
||||||
|
page.getPages(),
|
||||||
|
page.getPageNum(),
|
||||||
|
sort,
|
||||||
|
list);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
private String verifySortColumnName(final String sort, final String columnName) {
|
private String verifySortColumnName(final String sort, final String columnName) {
|
||||||
|
|
||||||
if (StringUtils.isBlank(sort)) {
|
if (StringUtils.isBlank(sort)) {
|
||||||
|
|
|
@ -132,6 +132,8 @@ public interface ClientConnectionDAO extends
|
||||||
key = "#connectionToken")
|
key = "#connectionToken")
|
||||||
Result<Void> removeFromProctoringRoom(Long connectionId, String connectionToken);
|
Result<Void> removeFromProctoringRoom(Long connectionId, String connectionToken);
|
||||||
|
|
||||||
|
Result<Void> markForProctoringUpdate(Long id);
|
||||||
|
|
||||||
/** Deletes the given ClientConnection data.
|
/** Deletes the given ClientConnection data.
|
||||||
*
|
*
|
||||||
* This evicts all entries from the CONNECTION_TOKENS_CACHE.
|
* This evicts all entries from the CONNECTION_TOKENS_CACHE.
|
||||||
|
|
|
@ -389,6 +389,19 @@ public class ClientConnectionDAOImpl implements ClientConnectionDAO {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Transactional
|
||||||
|
public Result<Void> markForProctoringUpdate(final Long id) {
|
||||||
|
return Result.tryCatch(() -> {
|
||||||
|
this.clientConnectionRecordMapper.updateByPrimaryKeySelective(new ClientConnectionRecord(
|
||||||
|
id,
|
||||||
|
null, null, null, null, null,
|
||||||
|
null, null, null, null, null, null,
|
||||||
|
null,
|
||||||
|
1));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@Transactional
|
@Transactional
|
||||||
public Result<Void> removeFromProctoringRoom(final Long connectionId, final String connectionToken) {
|
public Result<Void> removeFromProctoringRoom(final Long connectionId, final String connectionToken) {
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam;
|
||||||
|
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.Collection;
|
||||||
|
|
||||||
|
import org.springframework.scheduling.annotation.Async;
|
||||||
|
|
||||||
|
import ch.ethz.seb.sebserver.gbl.async.AsyncServiceSpringConfig;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.util.Result;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.impl.SEBServerUser;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap;
|
||||||
|
|
||||||
|
public interface SEBClientEventAdminService {
|
||||||
|
|
||||||
|
Result<EntityProcessingReport> deleteAllClientEvents(Collection<String> ids);
|
||||||
|
|
||||||
|
@Async(AsyncServiceSpringConfig.EXECUTOR_BEAN_NAME)
|
||||||
|
void exportSEBClientLogs(
|
||||||
|
OutputStream output,
|
||||||
|
FilterMap filterMap,
|
||||||
|
String sort,
|
||||||
|
ExportType exportType,
|
||||||
|
boolean includeConnectionDetails,
|
||||||
|
boolean includeExamDetails,
|
||||||
|
final SEBServerUser currentUser);
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,32 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam;
|
||||||
|
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientConnectionRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientEventRecord;
|
||||||
|
|
||||||
|
public interface SEBClientEventExporter {
|
||||||
|
|
||||||
|
ExportType exportType();
|
||||||
|
|
||||||
|
void streamHeader(
|
||||||
|
OutputStream output,
|
||||||
|
boolean includeConnectionDetails,
|
||||||
|
boolean includeExamDetails);
|
||||||
|
|
||||||
|
void streamData(
|
||||||
|
OutputStream output,
|
||||||
|
ClientEventRecord eventData,
|
||||||
|
ClientConnectionRecord connectionData,
|
||||||
|
Exam examData);
|
||||||
|
}
|
|
@ -37,6 +37,7 @@ import ch.ethz.seb.sebserver.gbl.model.institution.LmsSetup.LmsType;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
||||||
import ch.ethz.seb.sebserver.gbl.util.Cryptor;
|
import ch.ethz.seb.sebserver.gbl.util.Cryptor;
|
||||||
import ch.ethz.seb.sebserver.gbl.util.Result;
|
import ch.ethz.seb.sebserver.gbl.util.Result;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.util.Utils;
|
||||||
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.AdditionalAttributeRecord;
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.AdditionalAttributeRecord;
|
||||||
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.AdditionalAttributesDAO;
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.AdditionalAttributesDAO;
|
||||||
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ExamDAO;
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ExamDAO;
|
||||||
|
@ -180,7 +181,7 @@ public class ExamAdminServiceImpl implements ExamAdminService {
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_SERVER_URL,
|
ProctoringServiceSettings.ATTR_SERVER_URL,
|
||||||
proctoringServiceSettings.serverURL);
|
StringUtils.trim(proctoringServiceSettings.serverURL));
|
||||||
|
|
||||||
this.additionalAttributesDAO.saveAdditionalAttribute(
|
this.additionalAttributesDAO.saveAdditionalAttribute(
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
|
@ -192,13 +193,13 @@ public class ExamAdminServiceImpl implements ExamAdminService {
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_APP_KEY,
|
ProctoringServiceSettings.ATTR_APP_KEY,
|
||||||
proctoringServiceSettings.appKey);
|
StringUtils.trim(proctoringServiceSettings.appKey));
|
||||||
|
|
||||||
this.additionalAttributesDAO.saveAdditionalAttribute(
|
this.additionalAttributesDAO.saveAdditionalAttribute(
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_APP_SECRET,
|
ProctoringServiceSettings.ATTR_APP_SECRET,
|
||||||
this.cryptor.encrypt(proctoringServiceSettings.appSecret)
|
this.cryptor.encrypt(Utils.trim(proctoringServiceSettings.appSecret))
|
||||||
.getOrThrow()
|
.getOrThrow()
|
||||||
.toString());
|
.toString());
|
||||||
|
|
||||||
|
@ -207,13 +208,13 @@ public class ExamAdminServiceImpl implements ExamAdminService {
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_SDK_KEY,
|
ProctoringServiceSettings.ATTR_SDK_KEY,
|
||||||
proctoringServiceSettings.sdkKey);
|
StringUtils.trim(proctoringServiceSettings.sdkKey));
|
||||||
|
|
||||||
this.additionalAttributesDAO.saveAdditionalAttribute(
|
this.additionalAttributesDAO.saveAdditionalAttribute(
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_SDK_SECRET,
|
ProctoringServiceSettings.ATTR_SDK_SECRET,
|
||||||
this.cryptor.encrypt(proctoringServiceSettings.sdkSecret)
|
this.cryptor.encrypt(Utils.trim(proctoringServiceSettings.sdkSecret))
|
||||||
.getOrThrow()
|
.getOrThrow()
|
||||||
.toString());
|
.toString());
|
||||||
}
|
}
|
||||||
|
@ -240,7 +241,8 @@ public class ExamAdminServiceImpl implements ExamAdminService {
|
||||||
EntityType.EXAM,
|
EntityType.EXAM,
|
||||||
examId,
|
examId,
|
||||||
ProctoringServiceSettings.ATTR_ENABLE_PROCTORING)
|
ProctoringServiceSettings.ATTR_ENABLE_PROCTORING)
|
||||||
.map(rec -> BooleanUtils.toBoolean(rec.getValue()));
|
.map(rec -> BooleanUtils.toBoolean(rec.getValue()))
|
||||||
|
.onError(error -> log.error("Failed to verify proctoring enabled for exam: {}", examId, error));
|
||||||
if (result.hasError()) {
|
if (result.hasError()) {
|
||||||
return Result.of(false);
|
return Result.of(false);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,284 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam.impl;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.EnumMap;
|
||||||
|
import java.util.EnumSet;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
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.api.API.BulkActionType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.api.EntityType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport.ErrorEntry;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.Page;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.user.UserRole;
|
||||||
|
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.datalayer.batis.mapper.ClientEventRecordDynamicSqlSupport;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientConnectionRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientEventRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.PaginationService;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.impl.SEBServerUser;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ClientEventDAO;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.exam.SEBClientEventAdminService;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.exam.SEBClientEventExporter;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Service
|
||||||
|
@WebServiceProfile
|
||||||
|
public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminService {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(SEBClientEventAdminServiceImpl.class);
|
||||||
|
|
||||||
|
private final PaginationService paginationService;
|
||||||
|
private final ClientEventDAO clientEventDAO;
|
||||||
|
private final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler;
|
||||||
|
private final EnumMap<ExportType, SEBClientEventExporter> exporter;
|
||||||
|
|
||||||
|
public SEBClientEventAdminServiceImpl(
|
||||||
|
final PaginationService paginationService,
|
||||||
|
final ClientEventDAO clientEventDAO,
|
||||||
|
final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler,
|
||||||
|
final Collection<SEBClientEventExporter> exporter) {
|
||||||
|
|
||||||
|
this.paginationService = paginationService;
|
||||||
|
this.clientEventDAO = clientEventDAO;
|
||||||
|
this.sebClientEventExportTransactionHandler = sebClientEventExportTransactionHandler;
|
||||||
|
|
||||||
|
this.exporter = new EnumMap<>(ExportType.class);
|
||||||
|
exporter.forEach(exp -> this.exporter.putIfAbsent(exp.exportType(), exp));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Result<EntityProcessingReport> deleteAllClientEvents(final Collection<String> ids) {
|
||||||
|
return Result.tryCatch(() -> {
|
||||||
|
|
||||||
|
if (ids == null || ids.isEmpty()) {
|
||||||
|
return EntityProcessingReport.ofEmptyError();
|
||||||
|
}
|
||||||
|
|
||||||
|
final Set<EntityKey> sources = ids.stream()
|
||||||
|
.map(id -> new EntityKey(id, EntityType.CLIENT_EVENT))
|
||||||
|
.collect(Collectors.toSet());
|
||||||
|
|
||||||
|
final Result<Collection<EntityKey>> delete = this.clientEventDAO.delete(sources);
|
||||||
|
if (delete.hasError()) {
|
||||||
|
return new EntityProcessingReport(
|
||||||
|
Collections.emptyList(),
|
||||||
|
Collections.emptyList(),
|
||||||
|
Arrays.asList(new ErrorEntry(null, APIMessage.ErrorMessage.UNEXPECTED.of(delete.getError()))),
|
||||||
|
BulkActionType.HARD_DELETE);
|
||||||
|
} else {
|
||||||
|
return new EntityProcessingReport(
|
||||||
|
sources,
|
||||||
|
delete.get(),
|
||||||
|
Collections.emptyList(),
|
||||||
|
BulkActionType.HARD_DELETE);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void exportSEBClientLogs(
|
||||||
|
final OutputStream output,
|
||||||
|
final FilterMap filterMap,
|
||||||
|
final String sort,
|
||||||
|
final ExportType exportType,
|
||||||
|
final boolean includeConnectionDetails,
|
||||||
|
final boolean includeExamDetails,
|
||||||
|
final SEBServerUser currentUser) {
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
new exportRunner(
|
||||||
|
this.exporter.get(exportType),
|
||||||
|
includeConnectionDetails,
|
||||||
|
includeExamDetails,
|
||||||
|
new Pager(filterMap, sort),
|
||||||
|
output)
|
||||||
|
.run(currentUser);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.error("Unexpected error during export SEB logs: ", e);
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
output.flush();
|
||||||
|
} catch (final IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
output.close();
|
||||||
|
} catch (final IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private class exportRunner {
|
||||||
|
|
||||||
|
private final SEBClientEventExporter exporter;
|
||||||
|
private final boolean includeConnectionDetails;
|
||||||
|
private final boolean includeExamDetails;
|
||||||
|
private final Iterator<Collection<ClientEventRecord>> pager;
|
||||||
|
private final OutputStream output;
|
||||||
|
|
||||||
|
private final Map<Long, Exam> examCache;
|
||||||
|
private final Map<Long, ClientConnectionRecord> connectionCache;
|
||||||
|
|
||||||
|
public exportRunner(
|
||||||
|
final SEBClientEventExporter exporter,
|
||||||
|
final boolean includeConnectionDetails,
|
||||||
|
final boolean includeExamDetails,
|
||||||
|
final Iterator<Collection<ClientEventRecord>> pager,
|
||||||
|
final OutputStream output) {
|
||||||
|
|
||||||
|
this.exporter = exporter;
|
||||||
|
this.includeConnectionDetails = includeConnectionDetails;
|
||||||
|
this.includeExamDetails = includeExamDetails;
|
||||||
|
this.pager = pager;
|
||||||
|
this.output = output;
|
||||||
|
|
||||||
|
this.connectionCache = new HashMap<>();
|
||||||
|
this.examCache = new HashMap<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void run(final SEBServerUser currentUser) {
|
||||||
|
|
||||||
|
final EnumSet<UserRole> userRoles = currentUser.getUserRoles();
|
||||||
|
final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER);
|
||||||
|
|
||||||
|
// first stream header line
|
||||||
|
this.exporter.streamHeader(this.output, this.includeConnectionDetails, this.includeExamDetails);
|
||||||
|
|
||||||
|
// then batch with the pager and stream line per line
|
||||||
|
while (this.pager.hasNext()) {
|
||||||
|
this.pager.next().forEach(rec -> {
|
||||||
|
|
||||||
|
final Exam exam = getExam(rec.getClientConnectionId());
|
||||||
|
|
||||||
|
if (!isSupporterOnly || exam.isOwner(currentUser.uuid())) {
|
||||||
|
this.exporter.streamData(
|
||||||
|
this.output,
|
||||||
|
rec,
|
||||||
|
this.includeConnectionDetails ? getConnection(rec.getClientConnectionId()) : null,
|
||||||
|
this.includeExamDetails ? getExam(rec.getClientConnectionId()) : null);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ClientConnectionRecord getConnection(final Long connectionId) {
|
||||||
|
if (!this.connectionCache.containsKey(connectionId)) {
|
||||||
|
SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler
|
||||||
|
.clientConnectionById(connectionId)
|
||||||
|
.onSuccess(rec -> this.connectionCache.put(rec.getId(), rec))
|
||||||
|
.onError(error -> log.error("Failed to get ClientConnectionRecord for id: {}",
|
||||||
|
connectionId,
|
||||||
|
error));
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.connectionCache.get(connectionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Exam getExam(final Long connectionId) {
|
||||||
|
final ClientConnectionRecord connection = getConnection(connectionId);
|
||||||
|
final Long examId = connection.getExamId();
|
||||||
|
if (!this.examCache.containsKey(examId)) {
|
||||||
|
SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler
|
||||||
|
.examById(examId)
|
||||||
|
.onSuccess(e -> this.examCache.put(examId, e))
|
||||||
|
.onError(error -> log.error("Failed to get Exam for id: {}",
|
||||||
|
examId,
|
||||||
|
error));
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.examCache.get(examId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Pager implements Iterator<Collection<ClientEventRecord>> {
|
||||||
|
|
||||||
|
private final FilterMap filterMap;
|
||||||
|
private final String sort;
|
||||||
|
|
||||||
|
private int pageNumber = 1;
|
||||||
|
private final int pageSize = 100;
|
||||||
|
|
||||||
|
private Collection<ClientEventRecord> nextRecords;
|
||||||
|
|
||||||
|
public Pager(
|
||||||
|
final FilterMap filterMap,
|
||||||
|
final String sort) {
|
||||||
|
|
||||||
|
this.filterMap = filterMap;
|
||||||
|
this.sort = sort;
|
||||||
|
|
||||||
|
fetchNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return this.nextRecords != null && !this.nextRecords.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Collection<ClientEventRecord> next() {
|
||||||
|
final Collection<ClientEventRecord> result = this.nextRecords;
|
||||||
|
fetchNext();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void fetchNext() {
|
||||||
|
try {
|
||||||
|
|
||||||
|
final Page<ClientEventRecord> nextPage = SEBClientEventAdminServiceImpl.this.paginationService
|
||||||
|
.getPageOf(
|
||||||
|
this.pageNumber,
|
||||||
|
this.pageSize,
|
||||||
|
this.sort,
|
||||||
|
ClientEventRecordDynamicSqlSupport.clientEventRecord.name(),
|
||||||
|
() -> SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler
|
||||||
|
.allMatching(this.filterMap, Utils.truePredicate()))
|
||||||
|
.getOrThrow();
|
||||||
|
|
||||||
|
if (nextPage.getPageNumber() == this.pageNumber) {
|
||||||
|
this.nextRecords = nextPage.content;
|
||||||
|
this.pageNumber++;
|
||||||
|
} else {
|
||||||
|
this.nextRecords = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.error("Failed to fetch next batch: ", e);
|
||||||
|
this.nextRecords = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,132 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam.impl;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
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.Constants;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.EventType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.util.Utils;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientConnectionRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientEventRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.exam.SEBClientEventExporter;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Component
|
||||||
|
@WebServiceProfile
|
||||||
|
public class SEBClientEventCSVExporter implements SEBClientEventExporter {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(SEBClientEventCSVExporter.class);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ExportType exportType() {
|
||||||
|
return ExportType.CSV;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void streamHeader(
|
||||||
|
final OutputStream output,
|
||||||
|
final boolean includeConnectionDetails,
|
||||||
|
final boolean includeExamDetails) {
|
||||||
|
|
||||||
|
final StringBuilder builder = new StringBuilder();
|
||||||
|
builder.append("Event Type,Message,Value,Client Time (UTC),Server Time (UTC)");
|
||||||
|
|
||||||
|
if (includeConnectionDetails) {
|
||||||
|
builder.append(",User Session-ID,Client Machine,Connection Status,Connection Token");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (includeExamDetails) {
|
||||||
|
builder.append("Exam Name,Exam Description,Exam Type,Start Time (LMS),End Time (LMS)");
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.append(Constants.CARRIAGE_RETURN);
|
||||||
|
|
||||||
|
try {
|
||||||
|
output.write(Utils.toByteArray(builder));
|
||||||
|
} catch (final IOException e) {
|
||||||
|
log.error("Failed to stream header: ", e);
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
output.flush();
|
||||||
|
} catch (final IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void streamData(
|
||||||
|
final OutputStream output,
|
||||||
|
final ClientEventRecord eventData,
|
||||||
|
final ClientConnectionRecord connectionData,
|
||||||
|
final Exam examData) {
|
||||||
|
|
||||||
|
final StringBuilder builder = new StringBuilder();
|
||||||
|
final EventType type = EventType.byId(eventData.getType());
|
||||||
|
|
||||||
|
builder.append(type.name());
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toCSVString(eventData.getText()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(eventData.getNumericValue() != null ? eventData.getNumericValue() : "");
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toDateTimeUTC(eventData.getClientTime()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toDateTimeUTC(eventData.getServerTime()));
|
||||||
|
|
||||||
|
if (connectionData != null) {
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toCSVString(connectionData.getExamUserSessionId()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toCSVString(connectionData.getClientAddress()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(connectionData.getStatus());
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(connectionData.getConnectionToken());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (examData != null) {
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toCSVString(examData.getName()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(Utils.toCSVString(examData.getDescription()));
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(examData.getType().name());
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(examData.getStartTime());
|
||||||
|
builder.append(Constants.COMMA);
|
||||||
|
builder.append(examData.getEndTime());
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.append(Constants.CARRIAGE_RETURN);
|
||||||
|
|
||||||
|
try {
|
||||||
|
output.write(Utils.toByteArray(builder));
|
||||||
|
} catch (final IOException e) {
|
||||||
|
log.error("Failed to stream header: ", e);
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
output.flush();
|
||||||
|
} catch (final IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,113 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021 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.exam.impl;
|
||||||
|
|
||||||
|
import static org.mybatis.dynamic.sql.SqlBuilder.equalTo;
|
||||||
|
import static org.mybatis.dynamic.sql.SqlBuilder.isEqualToWhenPresent;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
|
import org.mybatis.dynamic.sql.SqlBuilder;
|
||||||
|
import org.springframework.context.annotation.Lazy;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.EventType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.util.Result;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientConnectionRecordDynamicSqlSupport;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientConnectionRecordMapper;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecordDynamicSqlSupport;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecordMapper;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientConnectionRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.ClientEventRecord;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ExamDAO;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Component
|
||||||
|
@WebServiceProfile
|
||||||
|
public class SEBClientEventExportTransactionHandler {
|
||||||
|
|
||||||
|
private final ClientEventRecordMapper clientEventRecordMapper;
|
||||||
|
private final ClientConnectionRecordMapper clientConnectionRecordMapper;
|
||||||
|
private final ExamDAO examDAO;
|
||||||
|
|
||||||
|
public SEBClientEventExportTransactionHandler(
|
||||||
|
final ClientEventRecordMapper clientEventRecordMapper,
|
||||||
|
final ClientConnectionRecordMapper clientConnectionRecordMapper,
|
||||||
|
final ExamDAO examDAO) {
|
||||||
|
|
||||||
|
this.clientEventRecordMapper = clientEventRecordMapper;
|
||||||
|
this.clientConnectionRecordMapper = clientConnectionRecordMapper;
|
||||||
|
this.examDAO = examDAO;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Transactional(readOnly = true)
|
||||||
|
public Result<Collection<ClientEventRecord>> allMatching(
|
||||||
|
final FilterMap filterMap,
|
||||||
|
final Predicate<ClientEvent> predicate) {
|
||||||
|
|
||||||
|
return Result.tryCatch(() -> this.clientEventRecordMapper
|
||||||
|
.selectByExample()
|
||||||
|
.leftJoin(ClientConnectionRecordDynamicSqlSupport.clientConnectionRecord)
|
||||||
|
.on(
|
||||||
|
ClientConnectionRecordDynamicSqlSupport.id,
|
||||||
|
equalTo(ClientEventRecordDynamicSqlSupport.clientConnectionId))
|
||||||
|
.where(
|
||||||
|
ClientConnectionRecordDynamicSqlSupport.institutionId,
|
||||||
|
isEqualToWhenPresent(filterMap.getInstitutionId()))
|
||||||
|
.and(
|
||||||
|
ClientConnectionRecordDynamicSqlSupport.examId,
|
||||||
|
isEqualToWhenPresent(filterMap.getClientEventExamId()))
|
||||||
|
.and(
|
||||||
|
ClientConnectionRecordDynamicSqlSupport.examUserSessionId,
|
||||||
|
SqlBuilder.isLikeWhenPresent(filterMap.getSQLWildcard(ClientConnection.FILTER_ATTR_SESSION_ID)))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.clientConnectionId,
|
||||||
|
isEqualToWhenPresent(filterMap.getClientEventConnectionId()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.type,
|
||||||
|
isEqualToWhenPresent(filterMap.getClientEventTypeId()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.type,
|
||||||
|
SqlBuilder.isNotEqualTo(EventType.LAST_PING.id))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.clientTime,
|
||||||
|
SqlBuilder.isGreaterThanOrEqualToWhenPresent(filterMap.getClientEventClientTimeFrom()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.clientTime,
|
||||||
|
SqlBuilder.isLessThanOrEqualToWhenPresent(filterMap.getClientEventClientTimeTo()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.serverTime,
|
||||||
|
SqlBuilder.isGreaterThanOrEqualToWhenPresent(filterMap.getClientEventServerTimeFrom()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.serverTime,
|
||||||
|
SqlBuilder.isLessThanOrEqualToWhenPresent(filterMap.getClientEventServerTimeTo()))
|
||||||
|
.and(
|
||||||
|
ClientEventRecordDynamicSqlSupport.text,
|
||||||
|
SqlBuilder.isLikeWhenPresent(filterMap.getClientEventText()))
|
||||||
|
.build()
|
||||||
|
.execute());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Transactional(readOnly = true)
|
||||||
|
public Result<ClientConnectionRecord> clientConnectionById(final Long id) {
|
||||||
|
return Result.tryCatch(() -> this.clientConnectionRecordMapper.selectByPrimaryKey(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
public Result<Exam> examById(final Long id) {
|
||||||
|
return this.examDAO.byPK(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -202,7 +202,8 @@ public class SEBRestrictionServiceImpl implements SEBRestrictionService {
|
||||||
.map(sebRestrictionData -> {
|
.map(sebRestrictionData -> {
|
||||||
|
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug("Applying SEB Client restriction on LMS with: {}", sebRestrictionData);
|
log.debug(" *** SEB Restriction *** Applying SEB Client restriction on LMS with: {}",
|
||||||
|
sebRestrictionData);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.lmsAPIService
|
return this.lmsAPIService
|
||||||
|
@ -221,7 +222,7 @@ public class SEBRestrictionServiceImpl implements SEBRestrictionService {
|
||||||
public Result<Exam> releaseSEBClientRestriction(final Exam exam) {
|
public Result<Exam> releaseSEBClientRestriction(final Exam exam) {
|
||||||
|
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug("Release SEB Client restrictions for exam: {}", exam);
|
log.debug(" *** SEB Restriction *** Release SEB Client restrictions from LMS for exam: {}", exam);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.lmsAPIService
|
return this.lmsAPIService
|
||||||
|
|
|
@ -236,10 +236,6 @@ public class MockupLmsAPITemplate implements LmsAPITemplate {
|
||||||
this.webserviceInfo.getHttpScheme() +
|
this.webserviceInfo.getHttpScheme() +
|
||||||
"://" + externalAddressAlias + "/api/";
|
"://" + externalAddressAlias + "/api/";
|
||||||
|
|
||||||
if (log.isTraceEnabled()) {
|
|
||||||
log.trace("Use external address for course access: {}", _externalStartURI);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new QuizData(
|
return new QuizData(
|
||||||
quizData.id, quizData.institutionId, quizData.lmsSetupId, quizData.lmsType,
|
quizData.id, quizData.institutionId, quizData.lmsSetupId, quizData.lmsType,
|
||||||
quizData.name, quizData.description, quizData.startTime,
|
quizData.name, quizData.description, quizData.startTime,
|
||||||
|
|
|
@ -186,9 +186,7 @@ public class SEBClientConnectionServiceImpl implements SEBClientConnectionServic
|
||||||
|
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
|
|
||||||
|
|
||||||
final ClientConnection clientConnection = getClientConnection(connectionToken);
|
final ClientConnection clientConnection = getClientConnection(connectionToken);
|
||||||
|
|
||||||
checkInstitutionalIntegrity(institutionId, clientConnection);
|
checkInstitutionalIntegrity(institutionId, clientConnection);
|
||||||
checkExamIntegrity(examId, clientConnection);
|
checkExamIntegrity(examId, clientConnection);
|
||||||
|
|
||||||
|
@ -271,7 +269,8 @@ public class SEBClientConnectionServiceImpl implements SEBClientConnectionServic
|
||||||
|
|
||||||
// connection integrity check
|
// connection integrity check
|
||||||
if (clientConnection.status == ConnectionStatus.ACTIVE) {
|
if (clientConnection.status == ConnectionStatus.ACTIVE) {
|
||||||
if (clientConnection.clientAddress != null && clientConnection.clientAddress.equals(clientAddress)) {
|
if (clientConnection.clientAddress != null &&
|
||||||
|
(StringUtils.isBlank(clientAddress) || clientConnection.clientAddress.equals(clientAddress))) {
|
||||||
// It seems that this is the same SEB that tries to establish the connection once again.
|
// It seems that this is the same SEB that tries to establish the connection once again.
|
||||||
// Just log this and return already established connection
|
// Just log this and return already established connection
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
|
|
|
@ -215,7 +215,7 @@ public class SEBClientInstructionServiceImpl implements SEBClientInstructionServ
|
||||||
.toString();
|
.toString();
|
||||||
|
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug("Send SEB client instruction: {} to: {} ", connectionToken, instructionJSON);
|
log.debug("Send SEB client instruction: {} to: {} ", instructionJSON, connectionToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
return instructionJSON;
|
return instructionJSON;
|
||||||
|
|
|
@ -77,7 +77,9 @@ public abstract class AbstractPingIndicator extends AbstractClientIndicator {
|
||||||
// Update last ping time on persistent storage
|
// Update last ping time on persistent storage
|
||||||
final long millisecondsNow = DateTimeUtils.currentTimeMillis();
|
final long millisecondsNow = DateTimeUtils.currentTimeMillis();
|
||||||
if (millisecondsNow - this.lastUpdate > INTERVAL_FOR_PERSISTENT_UPDATE) {
|
if (millisecondsNow - this.lastUpdate > INTERVAL_FOR_PERSISTENT_UPDATE) {
|
||||||
this.distributedPingCache.updatePing(this.pingRecord, millisecondsNow);
|
synchronized (this) {
|
||||||
|
this.distributedPingCache.updatePing(this.pingRecord, millisecondsNow);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,9 +21,11 @@ import org.joda.time.DateTimeUtils;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.DisposableBean;
|
import org.springframework.beans.factory.DisposableBean;
|
||||||
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
import org.springframework.context.annotation.Lazy;
|
import org.springframework.context.annotation.Lazy;
|
||||||
import org.springframework.scheduling.TaskScheduler;
|
import org.springframework.scheduling.TaskScheduler;
|
||||||
import org.springframework.stereotype.Component;
|
import org.springframework.stereotype.Component;
|
||||||
|
import org.springframework.transaction.annotation.Isolation;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.EventType;
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.EventType;
|
||||||
|
@ -51,13 +53,14 @@ public class DistributedPingCache implements DisposableBean {
|
||||||
final ClientEventLastPingMapper clientEventLastPingMapper,
|
final ClientEventLastPingMapper clientEventLastPingMapper,
|
||||||
final ClientEventRecordMapper clientEventRecordMapper,
|
final ClientEventRecordMapper clientEventRecordMapper,
|
||||||
final WebserviceInfo webserviceInfo,
|
final WebserviceInfo webserviceInfo,
|
||||||
final TaskScheduler taskScheduler) {
|
final TaskScheduler taskScheduler,
|
||||||
|
@Value("${sebserver.webservice.distributed.pingUpdate:3000}") final long pingUpdate) {
|
||||||
|
|
||||||
this.clientEventLastPingMapper = clientEventLastPingMapper;
|
this.clientEventLastPingMapper = clientEventLastPingMapper;
|
||||||
this.clientEventRecordMapper = clientEventRecordMapper;
|
this.clientEventRecordMapper = clientEventRecordMapper;
|
||||||
if (webserviceInfo.isDistributed()) {
|
if (webserviceInfo.isDistributed()) {
|
||||||
try {
|
try {
|
||||||
this.taskRef = taskScheduler.scheduleAtFixedRate(this::updateCache, 1000);
|
this.taskRef = taskScheduler.scheduleAtFixedRate(this::updateCache, pingUpdate);
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Failed to initialize distributed ping cache update task");
|
log.error("Failed to initialize distributed ping cache update task");
|
||||||
this.taskRef = null;
|
this.taskRef = null;
|
||||||
|
@ -126,7 +129,6 @@ public class DistributedPingCache implements DisposableBean {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Transactional
|
|
||||||
public void updatePing(final Long pingRecordId, final Long pingTime) {
|
public void updatePing(final Long pingRecordId, final Long pingTime) {
|
||||||
try {
|
try {
|
||||||
|
|
||||||
|
@ -155,6 +157,8 @@ public class DistributedPingCache implements DisposableBean {
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Failed to delete ping for connection -> {}", connectionId, e);
|
log.error("Failed to delete ping for connection -> {}", connectionId, e);
|
||||||
|
} finally {
|
||||||
|
this.pingCache.remove(connectionId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -180,7 +184,7 @@ public class DistributedPingCache implements DisposableBean {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Transactional
|
@Transactional(readOnly = true, isolation = Isolation.READ_UNCOMMITTED)
|
||||||
public void updateCache() {
|
public void updateCache() {
|
||||||
|
|
||||||
if (this.pingCache.isEmpty()) {
|
if (this.pingCache.isEmpty()) {
|
||||||
|
|
|
@ -282,22 +282,29 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
private void assignToCollectingRoom(final ClientConnectionRecord cc) {
|
private void assignToCollectingRoom(final ClientConnectionRecord cc) {
|
||||||
try {
|
try {
|
||||||
|
|
||||||
final RemoteProctoringRoom proctoringRoom = getProctoringRoom(
|
if (cc.getRemoteProctoringRoomId() == null) {
|
||||||
cc.getExamId(),
|
|
||||||
cc.getConnectionToken());
|
|
||||||
|
|
||||||
this.clientConnectionDAO
|
final RemoteProctoringRoom proctoringRoom = getProctoringRoom(
|
||||||
.assignToProctoringRoom(
|
cc.getExamId(),
|
||||||
cc.getId(),
|
cc.getConnectionToken());
|
||||||
cc.getConnectionToken(),
|
|
||||||
proctoringRoom.id)
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Assigning new SEB client to proctoring room: {}, connection: {}",
|
||||||
|
proctoringRoom.id,
|
||||||
|
cc);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.clientConnectionDAO
|
||||||
|
.assignToProctoringRoom(
|
||||||
|
cc.getId(),
|
||||||
|
cc.getConnectionToken(),
|
||||||
|
proctoringRoom.id)
|
||||||
|
.getOrThrow();
|
||||||
|
}
|
||||||
|
|
||||||
|
applyProcotringInstruction(cc)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
|
||||||
applyProcotringInstruction(
|
|
||||||
cc.getExamId(),
|
|
||||||
cc.getConnectionToken())
|
|
||||||
.getOrThrow();
|
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Failed to assign connection to collecting room: {}", cc, e);
|
log.error("Failed to assign connection to collecting room: {}", cc, e);
|
||||||
}
|
}
|
||||||
|
@ -598,11 +605,11 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private Result<Void> applyProcotringInstruction(
|
private Result<Void> applyProcotringInstruction(final ClientConnectionRecord cc) {
|
||||||
final Long examId,
|
|
||||||
final String connectionToken) {
|
|
||||||
|
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
|
final Long examId = cc.getExamId();
|
||||||
|
final String connectionToken = cc.getConnectionToken();
|
||||||
final ProctoringServiceSettings settings = this.examAdminService
|
final ProctoringServiceSettings settings = this.examAdminService
|
||||||
.getProctoringServiceSettings(examId)
|
.getProctoringServiceSettings(examId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
@ -624,17 +631,32 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
townhallRoom.subject)
|
townhallRoom.subject)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
|
||||||
sendJoinInstruction(
|
try {
|
||||||
examId,
|
sendJoinInstruction(
|
||||||
connectionToken,
|
examId,
|
||||||
roomConnection,
|
connectionToken,
|
||||||
examProctoringService);
|
roomConnection,
|
||||||
|
examProctoringService);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.error("Failed to send join for town-hall room assignment to connection: {}", cc);
|
||||||
|
this.clientConnectionDAO
|
||||||
|
.markForProctoringUpdate(cc.getId())
|
||||||
|
.onError(error -> log.error("Failed to mark connection for proctoring update: ", error));
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
|
try {
|
||||||
|
|
||||||
sendJoinCollectingRoomInstructions(
|
sendJoinCollectingRoomInstruction(
|
||||||
settings,
|
settings,
|
||||||
Arrays.asList(connectionToken),
|
examProctoringService,
|
||||||
examProctoringService);
|
connectionToken);
|
||||||
|
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.error("Failed to send join for collecting room assignment to connection: {}", cc);
|
||||||
|
this.clientConnectionDAO
|
||||||
|
.markForProctoringUpdate(cc.getId())
|
||||||
|
.onError(error -> log.error("Failed to mark connection for proctoring update: ", error));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -653,23 +675,30 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
clientConnectionTokens
|
clientConnectionTokens
|
||||||
.stream()
|
.stream()
|
||||||
.forEach(connectionToken -> {
|
.forEach(connectionToken -> {
|
||||||
final ProctoringRoomConnection proctoringConnection = examProctoringService
|
try {
|
||||||
.getClientRoomConnection(
|
final ProctoringRoomConnection proctoringConnection = examProctoringService
|
||||||
proctoringSettings,
|
.getClientRoomConnection(
|
||||||
|
proctoringSettings,
|
||||||
|
connectionToken,
|
||||||
|
roomName,
|
||||||
|
(StringUtils.isNotBlank(subject)) ? subject : roomName)
|
||||||
|
.onError(error -> log.error(
|
||||||
|
"Failed to get client room connection data for {} cause: {}",
|
||||||
|
connectionToken,
|
||||||
|
error.getMessage()))
|
||||||
|
.get();
|
||||||
|
if (proctoringConnection != null) {
|
||||||
|
sendJoinInstruction(
|
||||||
|
proctoringSettings.examId,
|
||||||
connectionToken,
|
connectionToken,
|
||||||
roomName,
|
proctoringConnection,
|
||||||
(StringUtils.isNotBlank(subject)) ? subject : roomName)
|
examProctoringService);
|
||||||
.onError(error -> log.error(
|
}
|
||||||
"Failed to get client room connection data for {} cause: {}",
|
} catch (final Exception e) {
|
||||||
connectionToken,
|
log.error("Failed to send join to break-out room: {} connection: {}",
|
||||||
error.getMessage()))
|
roomName,
|
||||||
.get();
|
roomName,
|
||||||
if (proctoringConnection != null) {
|
e);
|
||||||
sendJoinInstruction(
|
|
||||||
proctoringSettings.examId,
|
|
||||||
connectionToken,
|
|
||||||
proctoringConnection,
|
|
||||||
examProctoringService);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -688,10 +717,18 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
|
|
||||||
clientConnectionTokens
|
clientConnectionTokens
|
||||||
.stream()
|
.stream()
|
||||||
.forEach(connectionToken -> sendJoinCollectingRoomInstruction(
|
.forEach(connectionToken -> {
|
||||||
proctoringSettings,
|
try {
|
||||||
examProctoringService,
|
sendJoinCollectingRoomInstruction(
|
||||||
connectionToken));
|
proctoringSettings,
|
||||||
|
examProctoringService,
|
||||||
|
connectionToken);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.error(
|
||||||
|
"Failed to send proctoring room join instruction to single client. Skip and proceed with other clients. {}",
|
||||||
|
e.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendJoinCollectingRoomInstruction(
|
private void sendJoinCollectingRoomInstruction(
|
||||||
|
@ -721,8 +758,10 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
clientConnection.clientConnection.connectionToken,
|
clientConnection.clientConnection.connectionToken,
|
||||||
proctoringConnection,
|
proctoringConnection,
|
||||||
examProctoringService);
|
examProctoringService);
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Failed to send proctoring room join instruction to client: {}", connectionToken, e);
|
log.error("Failed to send proctoring room join instruction to client: {}", connectionToken, e);
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -732,6 +771,12 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
final ProctoringRoomConnection proctoringConnection,
|
final ProctoringRoomConnection proctoringConnection,
|
||||||
final ExamProctoringService examProctoringService) {
|
final ExamProctoringService examProctoringService) {
|
||||||
|
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Send proctoring join instruction to connection: {}, room: {}",
|
||||||
|
connectionToken,
|
||||||
|
proctoringConnection.roomName);
|
||||||
|
}
|
||||||
|
|
||||||
final Map<String, String> attributes = examProctoringService
|
final Map<String, String> attributes = examProctoringService
|
||||||
.createJoinInstructionAttributes(proctoringConnection);
|
.createJoinInstructionAttributes(proctoringConnection);
|
||||||
|
|
||||||
|
@ -742,7 +787,8 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
|
||||||
attributes,
|
attributes,
|
||||||
connectionToken,
|
connectionToken,
|
||||||
true)
|
true)
|
||||||
.onError(error -> log.error("Failed to send join instruction: {}", connectionToken, error));
|
.onError(error -> log.error("Failed to send join instruction: {}", connectionToken, error))
|
||||||
|
.getOrThrow();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -623,6 +623,10 @@ public class ZoomProctoringService implements ExamProctoringService {
|
||||||
credentials.clientIdAsString(),
|
credentials.clientIdAsString(),
|
||||||
expTime);
|
expTime);
|
||||||
|
|
||||||
|
if (log.isTraceEnabled()) {
|
||||||
|
log.trace("Zoom API Token payload: {}", jwtPayload);
|
||||||
|
}
|
||||||
|
|
||||||
final String jwtPayloadPart = urlEncoder
|
final String jwtPayloadPart = urlEncoder
|
||||||
.encodeToString(jwtPayload.getBytes(StandardCharsets.UTF_8));
|
.encodeToString(jwtPayload.getBytes(StandardCharsets.UTF_8));
|
||||||
|
|
||||||
|
@ -675,6 +679,10 @@ public class ZoomProctoringService implements ExamProctoringService {
|
||||||
expTime,
|
expTime,
|
||||||
expTime);
|
expTime);
|
||||||
|
|
||||||
|
if (log.isTraceEnabled()) {
|
||||||
|
log.trace("Zoom SDK Token payload: {}", jwtPayload);
|
||||||
|
}
|
||||||
|
|
||||||
final String jwtPayloadPart = urlEncoder
|
final String jwtPayloadPart = urlEncoder
|
||||||
.encodeToString(jwtPayload.getBytes(StandardCharsets.UTF_8));
|
.encodeToString(jwtPayload.getBytes(StandardCharsets.UTF_8));
|
||||||
|
|
||||||
|
@ -724,6 +732,10 @@ public class ZoomProctoringService implements ExamProctoringService {
|
||||||
final String tmpString = String.format("%s.%s.%s.%d.%s", apiKey, meetingId, ts, status, hashBase64Str);
|
final String tmpString = String.format("%s.%s.%s.%d.%s", apiKey, meetingId, ts, status, hashBase64Str);
|
||||||
final String encodedString = Base64.getEncoder().encodeToString(tmpString.getBytes());
|
final String encodedString = Base64.getEncoder().encodeToString(tmpString.getBytes());
|
||||||
|
|
||||||
|
if (log.isTraceEnabled()) {
|
||||||
|
log.trace("Zoom Meeting signature payload: {}", tmpString);
|
||||||
|
}
|
||||||
|
|
||||||
return encodedString.replaceAll("\\=+$", "");
|
return encodedString.replaceAll("\\=+$", "");
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
|
@ -732,12 +744,16 @@ public class ZoomProctoringService implements ExamProctoringService {
|
||||||
}
|
}
|
||||||
|
|
||||||
private long forExam(final ProctoringServiceSettings examProctoring) {
|
private long forExam(final ProctoringServiceSettings examProctoring) {
|
||||||
if (examProctoring.examId == null) {
|
|
||||||
throw new IllegalStateException("Missing exam identifier from ExamProctoring data");
|
|
||||||
}
|
|
||||||
|
|
||||||
long expTime = Utils.toSeconds(System.currentTimeMillis() + Constants.DAY_IN_MILLIS);
|
// TODO
|
||||||
if (this.examSessionService.isExamRunning(examProctoring.examId)) {
|
// NOTE: following is the original code that includes the exam end time but seems to make trouble for OLAT
|
||||||
|
final long nowInSeconds = Utils.getSecondsNow();
|
||||||
|
final long nowPlus30MinInSeconds = nowInSeconds + Utils.toSeconds(30 * Constants.MINUTE_IN_MILLIS);
|
||||||
|
final long nowPlusOneDayInSeconds = nowInSeconds + Utils.toSeconds(Constants.DAY_IN_MILLIS);
|
||||||
|
final long nowPlusTwoDayInSeconds = nowInSeconds + Utils.toSeconds(2 * Constants.DAY_IN_MILLIS);
|
||||||
|
|
||||||
|
long expTime = nowPlusOneDayInSeconds;
|
||||||
|
if (examProctoring.examId == null && this.examSessionService.isExamRunning(examProctoring.examId)) {
|
||||||
final Exam exam = this.examSessionService.getRunningExam(examProctoring.examId)
|
final Exam exam = this.examSessionService.getRunningExam(examProctoring.examId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
if (exam.endTime != null) {
|
if (exam.endTime != null) {
|
||||||
|
@ -746,10 +762,16 @@ public class ZoomProctoringService implements ExamProctoringService {
|
||||||
}
|
}
|
||||||
// refer to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
|
// refer to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
|
||||||
// "exp": 0, //JWT expiration date (Min:1800 seconds greater than iat value, Max: 48 hours greater than iat value) in epoch format.
|
// "exp": 0, //JWT expiration date (Min:1800 seconds greater than iat value, Max: 48 hours greater than iat value) in epoch format.
|
||||||
if (expTime - Utils.getSecondsNow() > Utils.toSeconds(2 * Constants.DAY_IN_MILLIS)) {
|
if (expTime > nowPlusTwoDayInSeconds) {
|
||||||
expTime = Utils.toSeconds(System.currentTimeMillis() + Constants.DAY_IN_MILLIS);
|
expTime = nowPlusTwoDayInSeconds - 10; // Do not set to max because it is not well defined if max is included or not
|
||||||
|
} else if (expTime < nowPlus30MinInSeconds) {
|
||||||
|
expTime = nowPlusOneDayInSeconds;
|
||||||
}
|
}
|
||||||
return expTime;
|
|
||||||
|
log.debug("**** SDK Token exp time with exam-end-time inclusion would be: {}", expTime);
|
||||||
|
|
||||||
|
// NOTE: Set this to the maximum according to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
|
||||||
|
return nowPlusTwoDayInSeconds - 10; // Do not set to max because it is not well defined if max is included or not;
|
||||||
}
|
}
|
||||||
|
|
||||||
private final static class ZoomRestTemplate {
|
private final static class ZoomRestTemplate {
|
||||||
|
|
|
@ -8,17 +8,20 @@
|
||||||
|
|
||||||
package ch.ethz.seb.sebserver.webservice.weblayer.api;
|
package ch.ethz.seb.sebserver.webservice.weblayer.api;
|
||||||
|
|
||||||
import java.util.Arrays;
|
import java.io.IOException;
|
||||||
|
import java.io.PipedInputStream;
|
||||||
|
import java.io.PipedOutputStream;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
|
||||||
import java.util.stream.Collectors;
|
|
||||||
|
|
||||||
|
import javax.servlet.ServletOutputStream;
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.mybatis.dynamic.sql.SqlTable;
|
import org.mybatis.dynamic.sql.SqlTable;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.http.MediaType;
|
import org.springframework.http.MediaType;
|
||||||
import org.springframework.util.MultiValueMap;
|
import org.springframework.util.MultiValueMap;
|
||||||
import org.springframework.web.bind.annotation.RequestMapping;
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
@ -28,16 +31,14 @@ import org.springframework.web.bind.annotation.RestController;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.api.API;
|
import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType;
|
import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
|
||||||
import ch.ethz.seb.sebserver.gbl.api.EntityType;
|
import ch.ethz.seb.sebserver.gbl.api.EntityType;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType;
|
import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityDependency;
|
import ch.ethz.seb.sebserver.gbl.model.EntityDependency;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport;
|
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport.ErrorEntry;
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.GrantEntity;
|
import ch.ethz.seb.sebserver.gbl.model.GrantEntity;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.Page;
|
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.ClientEvent;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent;
|
import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.user.UserRole;
|
import ch.ethz.seb.sebserver.gbl.model.user.UserRole;
|
||||||
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
|
||||||
|
@ -53,6 +54,7 @@ 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.ExamDAO;
|
||||||
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap;
|
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.dao.UserActivityLogDAO;
|
||||||
|
import ch.ethz.seb.sebserver.webservice.servicelayer.exam.SEBClientEventAdminService;
|
||||||
import ch.ethz.seb.sebserver.webservice.servicelayer.validation.BeanValidationService;
|
import ch.ethz.seb.sebserver.webservice.servicelayer.validation.BeanValidationService;
|
||||||
|
|
||||||
@WebServiceProfile
|
@WebServiceProfile
|
||||||
|
@ -62,6 +64,7 @@ public class ClientEventController extends ReadonlyEntityController<ClientEvent,
|
||||||
|
|
||||||
private final ExamDAO examDao;
|
private final ExamDAO examDao;
|
||||||
private final ClientEventDAO clientEventDAO;
|
private final ClientEventDAO clientEventDAO;
|
||||||
|
private final SEBClientEventAdminService sebClientEventAdminService;
|
||||||
|
|
||||||
protected ClientEventController(
|
protected ClientEventController(
|
||||||
final AuthorizationService authorization,
|
final AuthorizationService authorization,
|
||||||
|
@ -70,7 +73,8 @@ public class ClientEventController extends ReadonlyEntityController<ClientEvent,
|
||||||
final UserActivityLogDAO userActivityLogDAO,
|
final UserActivityLogDAO userActivityLogDAO,
|
||||||
final PaginationService paginationService,
|
final PaginationService paginationService,
|
||||||
final BeanValidationService beanValidationService,
|
final BeanValidationService beanValidationService,
|
||||||
final ExamDAO examDao) {
|
final ExamDAO examDao,
|
||||||
|
final SEBClientEventAdminService sebClientEventAdminService) {
|
||||||
|
|
||||||
super(authorization,
|
super(authorization,
|
||||||
bulkActionService,
|
bulkActionService,
|
||||||
|
@ -81,6 +85,7 @@ public class ClientEventController extends ReadonlyEntityController<ClientEvent,
|
||||||
|
|
||||||
this.examDao = examDao;
|
this.examDao = examDao;
|
||||||
this.clientEventDAO = entityDAO;
|
this.clientEventDAO = entityDAO;
|
||||||
|
this.sebClientEventAdminService = sebClientEventAdminService;
|
||||||
}
|
}
|
||||||
|
|
||||||
@RequestMapping(
|
@RequestMapping(
|
||||||
|
@ -136,28 +141,69 @@ public class ClientEventController extends ReadonlyEntityController<ClientEvent,
|
||||||
|
|
||||||
this.checkWritePrivilege(institutionId);
|
this.checkWritePrivilege(institutionId);
|
||||||
|
|
||||||
if (ids == null || ids.isEmpty()) {
|
return this.sebClientEventAdminService
|
||||||
return EntityProcessingReport.ofEmptyError();
|
.deleteAllClientEvents(ids)
|
||||||
}
|
.getOrThrow();
|
||||||
|
}
|
||||||
|
|
||||||
final Set<EntityKey> sources = ids.stream()
|
@RequestMapping(
|
||||||
.map(id -> new EntityKey(id, EntityType.CLIENT_EVENT))
|
path = API.SEB_CLIENT_EVENT_EXPORT_PATH_SEGMENT,
|
||||||
.collect(Collectors.toSet());
|
method = RequestMethod.GET,
|
||||||
|
produces = MediaType.APPLICATION_OCTET_STREAM_VALUE)
|
||||||
|
public void exportEvents(
|
||||||
|
@RequestParam(
|
||||||
|
name = API.PARAM_INSTITUTION_ID,
|
||||||
|
required = true,
|
||||||
|
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
||||||
|
@RequestParam(name = API.SEB_CLIENT_EVENT_EXPORT_TYPE, required = true) final ExportType type,
|
||||||
|
@RequestParam(
|
||||||
|
name = API.SEB_CLIENT_EVENT_EXPORT_INCLUDE_CONNECTIONS,
|
||||||
|
required = false,
|
||||||
|
defaultValue = "true") final boolean includeConnectionDetails,
|
||||||
|
@RequestParam(
|
||||||
|
name = API.SEB_CLIENT_EVENT_EXPORT_INCLUDE_EXAMS,
|
||||||
|
required = false,
|
||||||
|
defaultValue = "false") final boolean includeExamDetails,
|
||||||
|
@RequestParam(name = Page.ATTR_SORT, required = false) final String sort,
|
||||||
|
@RequestParam final MultiValueMap<String, String> allRequestParams,
|
||||||
|
final HttpServletRequest request,
|
||||||
|
final HttpServletResponse response) throws IOException {
|
||||||
|
|
||||||
final Result<Collection<EntityKey>> delete = this.clientEventDAO.delete(sources);
|
// at least current user must have base read access for specified entity type within its own institution
|
||||||
|
checkReadPrivilege(institutionId);
|
||||||
|
|
||||||
if (delete.hasError()) {
|
final FilterMap filterMap = new FilterMap(allRequestParams, request.getQueryString());
|
||||||
return new EntityProcessingReport(
|
populateFilterMap(filterMap, institutionId, sort);
|
||||||
Collections.emptyList(),
|
|
||||||
Collections.emptyList(),
|
final ServletOutputStream outputStream = response.getOutputStream();
|
||||||
Arrays.asList(new ErrorEntry(null, APIMessage.ErrorMessage.UNEXPECTED.of(delete.getError()))),
|
PipedOutputStream pout;
|
||||||
BulkActionType.HARD_DELETE);
|
PipedInputStream pin;
|
||||||
} else {
|
try {
|
||||||
return new EntityProcessingReport(
|
pout = new PipedOutputStream();
|
||||||
sources,
|
pin = new PipedInputStream(pout);
|
||||||
delete.get(),
|
|
||||||
Collections.emptyList(),
|
final SEBServerUser currentUser = this.authorization
|
||||||
BulkActionType.HARD_DELETE);
|
.getUserService()
|
||||||
|
.getCurrentUser();
|
||||||
|
|
||||||
|
this.sebClientEventAdminService.exportSEBClientLogs(
|
||||||
|
pout,
|
||||||
|
filterMap,
|
||||||
|
sort,
|
||||||
|
type,
|
||||||
|
includeConnectionDetails,
|
||||||
|
includeExamDetails,
|
||||||
|
currentUser);
|
||||||
|
|
||||||
|
IOUtils.copyLarge(pin, outputStream);
|
||||||
|
|
||||||
|
response.setStatus(HttpStatus.OK.value());
|
||||||
|
|
||||||
|
outputStream.flush();
|
||||||
|
|
||||||
|
} finally {
|
||||||
|
outputStream.flush();
|
||||||
|
outputStream.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -177,12 +223,14 @@ public class ClientEventController extends ReadonlyEntityController<ClientEvent,
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Result<ClientEvent> checkReadAccess(final ClientEvent entity) {
|
protected Result<ClientEvent> checkReadAccess(final ClientEvent entity) {
|
||||||
|
final EnumSet<UserRole> userRoles = this.authorization
|
||||||
|
.getUserService()
|
||||||
|
.getCurrentUser()
|
||||||
|
.getUserRoles();
|
||||||
|
final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER);
|
||||||
|
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
final EnumSet<UserRole> userRoles = this.authorization
|
|
||||||
.getUserService()
|
|
||||||
.getCurrentUser()
|
|
||||||
.getUserRoles();
|
|
||||||
final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER);
|
|
||||||
if (isSupporterOnly) {
|
if (isSupporterOnly) {
|
||||||
// check owner grant be getting exam
|
// check owner grant be getting exam
|
||||||
return super.checkReadAccess(entity)
|
return super.checkReadAccess(entity)
|
||||||
|
|
|
@ -131,7 +131,8 @@ public class ExamAPI_V1_Controller {
|
||||||
.map(this::createRunningExamInfo)
|
.map(this::createRunningExamInfo)
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
} else {
|
} else {
|
||||||
final Exam exam = this.examSessionService.getExamDAO().byPK(examId)
|
final Exam exam = this.examSessionService.getExamDAO()
|
||||||
|
.byPK(examId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
|
||||||
result = Arrays.asList(createRunningExamInfo(exam));
|
result = Arrays.asList(createRunningExamInfo(exam));
|
||||||
|
|
|
@ -327,7 +327,7 @@ public class ExamProctoringController {
|
||||||
|
|
||||||
this.authorizationService.checkRead(this.examSessionService
|
this.authorizationService.checkRead(this.examSessionService
|
||||||
.getExamDAO()
|
.getExamDAO()
|
||||||
.byPK(examId)
|
.examGrantEntityByPK(examId)
|
||||||
.getOrThrow());
|
.getOrThrow());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -72,11 +72,10 @@ public class IndicatorController extends EntityController<Indicator, Indicator>
|
||||||
@Override
|
@Override
|
||||||
protected Indicator createNew(final POSTMapper postParams) {
|
protected Indicator createNew(final POSTMapper postParams) {
|
||||||
final Long examId = postParams.getLong(Domain.INDICATOR.ATTR_EXAM_ID);
|
final Long examId = postParams.getLong(Domain.INDICATOR.ATTR_EXAM_ID);
|
||||||
|
if (examId == null) {
|
||||||
return this.examDao
|
throw new RuntimeException("Missing exam model id from request parameter map!");
|
||||||
.byPK(examId)
|
}
|
||||||
.map(exam -> new Indicator(exam, postParams))
|
return new Indicator(examId, postParams);
|
||||||
.getOrThrow();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -120,7 +119,8 @@ public class IndicatorController extends EntityController<Indicator, Indicator>
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.examDao.byPK(entity.examId)
|
return this.examDao
|
||||||
|
.examGrantEntityByPK(entity.examId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,6 +13,7 @@ spring.datasource.hikari.initializationFailTimeout=30000
|
||||||
spring.datasource.hikari.connectionTimeout=30000
|
spring.datasource.hikari.connectionTimeout=30000
|
||||||
spring.datasource.hikari.idleTimeout=600000
|
spring.datasource.hikari.idleTimeout=600000
|
||||||
spring.datasource.hikari.maxLifetime=1800000
|
spring.datasource.hikari.maxLifetime=1800000
|
||||||
|
spring.datasource.hikari.maximumPoolSize=500
|
||||||
|
|
||||||
sebserver.http.client.connect-timeout=15000
|
sebserver.http.client.connect-timeout=15000
|
||||||
sebserver.http.client.connection-request-timeout=10000
|
sebserver.http.client.connection-request-timeout=10000
|
||||||
|
|
|
@ -10,8 +10,9 @@ server.tomcat.uri-encoding=UTF-8
|
||||||
logging.level.ch=INFO
|
logging.level.ch=INFO
|
||||||
logging.level.ch.ethz.seb.sebserver.webservice.datalayer=INFO
|
logging.level.ch.ethz.seb.sebserver.webservice.datalayer=INFO
|
||||||
logging.level.org.springframework.cache=INFO
|
logging.level.org.springframework.cache=INFO
|
||||||
logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl=DEBUG
|
logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl=TRACE
|
||||||
logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session=DEBUG
|
logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session=DEBUG
|
||||||
|
logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring=TRACE
|
||||||
#logging.level.ch.ethz.seb.sebserver.webservice.weblayer.api.ExamAPI_V1_Controller=TRACE
|
#logging.level.ch.ethz.seb.sebserver.webservice.weblayer.api.ExamAPI_V1_Controller=TRACE
|
||||||
|
|
||||||
sebserver.http.client.connect-timeout=150000
|
sebserver.http.client.connect-timeout=150000
|
||||||
|
|
|
@ -25,7 +25,7 @@ sebserver.gui.entrypoint=/gui
|
||||||
|
|
||||||
sebserver.gui.webservice.apipath=${sebserver.webservice.api.admin.endpoint}
|
sebserver.gui.webservice.apipath=${sebserver.webservice.api.admin.endpoint}
|
||||||
# defines the polling interval that is used to poll the webservice for client connection data on a monitored exam page
|
# defines the polling interval that is used to poll the webservice for client connection data on a monitored exam page
|
||||||
sebserver.gui.webservice.poll-interval=1000
|
sebserver.gui.webservice.poll-interval=3000
|
||||||
sebserver.gui.webservice.mock-lms-enabled=true
|
sebserver.gui.webservice.mock-lms-enabled=true
|
||||||
sebserver.gui.webservice.edx-lms-enabled=true
|
sebserver.gui.webservice.edx-lms-enabled=true
|
||||||
sebserver.gui.webservice.moodle-lms-enabled=true
|
sebserver.gui.webservice.moodle-lms-enabled=true
|
||||||
|
|
|
@ -10,6 +10,9 @@ sebserver.init.adminaccount.username=sebserver-admin
|
||||||
sebserver.init.database.integrity.checks=true
|
sebserver.init.database.integrity.checks=true
|
||||||
sebserver.init.database.integrity.try-fix=true
|
sebserver.init.database.integrity.try-fix=true
|
||||||
|
|
||||||
|
sebserver.webservice.distributed=false
|
||||||
|
sebserver.webservice.distributed.pingUpdate=3000
|
||||||
|
|
||||||
### webservice caching
|
### webservice caching
|
||||||
spring.cache.jcache.provider=org.ehcache.jsr107.EhcacheCachingProvider
|
spring.cache.jcache.provider=org.ehcache.jsr107.EhcacheCachingProvider
|
||||||
spring.cache.jcache.config=classpath:config/ehcache.xml
|
spring.cache.jcache.config=classpath:config/ehcache.xml
|
||||||
|
@ -28,6 +31,7 @@ spring.datasource.hikari.initializationFailTimeout=3000
|
||||||
spring.datasource.hikari.connectionTimeout=30000
|
spring.datasource.hikari.connectionTimeout=30000
|
||||||
spring.datasource.hikari.idleTimeout=600000
|
spring.datasource.hikari.idleTimeout=600000
|
||||||
spring.datasource.hikari.maxLifetime=1800000
|
spring.datasource.hikari.maxLifetime=1800000
|
||||||
|
spring.datasource.hikari.maximumPoolSize=500
|
||||||
|
|
||||||
### webservice security
|
### webservice security
|
||||||
spring.datasource.password=${sebserver.mariadb.password}
|
spring.datasource.password=${sebserver.mariadb.password}
|
||||||
|
|
|
@ -1956,6 +1956,7 @@ sebserver.seblogs.form.column.exam.endTime=End Time
|
||||||
sebserver.seblogs.form.column.exam.endTime.tooltip=The end date and time of the exam
|
sebserver.seblogs.form.column.exam.endTime.tooltip=The end date and time of the exam
|
||||||
|
|
||||||
sebserver.seblogs.action.delete=Delete Logs
|
sebserver.seblogs.action.delete=Delete Logs
|
||||||
|
sebserver.seblogs.action.export.csv=Export CSV
|
||||||
sebserver.seblogs.delete.form.title=Delete SEB Logs
|
sebserver.seblogs.delete.form.title=Delete SEB Logs
|
||||||
sebserver.seblogs.delete.form.info=This will delete all SEB client logs from the current filtered list.<br/>Please check carefully if all SEB client logs from the list shall be deleted.<br/><br/>There are currently {0} logs within the list.
|
sebserver.seblogs.delete.form.info=This will delete all SEB client logs from the current filtered list.<br/>Please check carefully if all SEB client logs from the list shall be deleted.<br/><br/>There are currently {0} logs within the list.
|
||||||
sebserver.seblogs.delete.action.delete=Delete All Logs
|
sebserver.seblogs.delete.action.delete=Delete All Logs
|
||||||
|
|
|
@ -51,7 +51,8 @@ public class ExamProctoringRoomServiceTest extends AdministrationAPIIntegrationT
|
||||||
assertFalse(runningExamsForInstitution.hasError());
|
assertFalse(runningExamsForInstitution.hasError());
|
||||||
final Collection<Exam> collection = runningExamsForInstitution.get();
|
final Collection<Exam> collection = runningExamsForInstitution.get();
|
||||||
assertFalse(collection.isEmpty());
|
assertFalse(collection.isEmpty());
|
||||||
final Exam exam = collection.iterator().next();
|
final Exam exam = collection.stream().filter(e -> e.id == 2L).findAny().orElse(null);
|
||||||
|
assertNotNull(exam);
|
||||||
assertEquals("Demo Quiz 6 (MOCKUP)", exam.name);
|
assertEquals("Demo Quiz 6 (MOCKUP)", exam.name);
|
||||||
assertEquals("2", String.valueOf(exam.id));
|
assertEquals("2", String.valueOf(exam.id));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue