SEBSERV-163 front-end implementation

This commit is contained in:
anhefti 2022-09-07 15:47:36 +02:00
parent 4f2586e799
commit 5703f1cb43
13 changed files with 276 additions and 61 deletions

View file

@ -28,7 +28,7 @@ public interface ClientGroupData extends Entity {
MAC_OS("TODO"), MAC_OS("TODO"),
I_OS("TODO"); I_OS("TODO");
final String queryString; public final String queryString;
private ClientOS(final String queryString) { private ClientOS(final String queryString) {
this.queryString = queryString; this.queryString = queryString;

View file

@ -332,8 +332,10 @@ public final class ClientConnection implements GrantEntity {
return false; return false;
} else if (!this.connectionToken.equals(other.connectionToken)) } else if (!this.connectionToken.equals(other.connectionToken))
return false; return false;
if (this.status != other.status) if (this.status != other.status)
return false; return false;
if (this.userSessionId == null) { if (this.userSessionId == null) {
if (other.userSessionId != null) if (other.userSessionId != null)
return false; return false;

View file

@ -12,6 +12,7 @@ import java.util.Collection;
import java.util.HashSet; import java.util.HashSet;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Objects;
import java.util.Set; import java.util.Set;
import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonCreator;
@ -156,6 +157,14 @@ public class ClientConnectionData implements GrantEntity {
return false; return false;
} }
if (!Objects.equals(this.groups, other.groups)) {
return false;
}
return true;
}
public boolean indicatorValuesEquals(final ClientConnectionData other) {
if (this.indicatorValues.size() != other.indicatorValues.size()) { if (this.indicatorValues.size() != other.indicatorValues.size()) {
return false; return false;
} }

View file

@ -0,0 +1,45 @@
/*
* Copyright (c) 2022 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.gbl.monitoring;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroupData.ClientGroupType;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroupData.ClientOS;
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
@Lazy
@Component
public class ClientOSGroupMatcher implements ClientGroupConnectionMatcher {
@Override
public ClientGroupType matcherType() {
return ClientGroupType.CLIENT_OS;
}
@Override
public boolean isInGroup(final ClientConnection clientConnection, final ClientGroup group) {
if (group == null
|| group.type != ClientGroupType.CLIENT_OS
|| clientConnection == null
|| clientConnection.info == null) {
return false;
}
try {
final ClientOS osType = ClientOS.valueOf(group.getData());
return clientConnection.info.contains(osType.queryString);
} catch (final Exception e) {
return false;
}
}
}

View file

@ -26,6 +26,7 @@ import ch.ethz.seb.sebserver.gbl.api.API;
import ch.ethz.seb.sebserver.gbl.api.EntityType; import ch.ethz.seb.sebserver.gbl.api.EntityType;
import ch.ethz.seb.sebserver.gbl.model.Domain; import ch.ethz.seb.sebserver.gbl.model.Domain;
import ch.ethz.seb.sebserver.gbl.model.EntityKey; import ch.ethz.seb.sebserver.gbl.model.EntityKey;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings;
@ -55,6 +56,7 @@ import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExam; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExam;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExamProctoringSettings; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExamProctoringSettings;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.clientgroup.GetClientGroups;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.indicator.GetIndicators; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.indicator.GetIndicators;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.logs.GetExtendedClientEventPage; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.logs.GetExtendedClientEventPage;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.ConfirmPendingClientNotification; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.session.ConfirmPendingClientNotification;
@ -191,6 +193,11 @@ public class MonitoringClientConnection implements TemplateComposer {
.call() .call()
.getOrThrow(); .getOrThrow();
final Collection<ClientGroup> clientGroups = restService.getBuilder(GetClientGroups.class)
.withQueryParam(Indicator.FILTER_ATTR_EXAM_ID, parentEntityKey.modelId)
.call()
.getOrThrow();
final RestCall<ClientConnectionData>.RestCallBuilder getConnectionData = final RestCall<ClientConnectionData>.RestCallBuilder getConnectionData =
restService.getBuilder(GetClientConnectionData.class) restService.getBuilder(GetClientConnectionData.class)
.withURIVariable(API.PARAM_PARENT_MODEL_ID, exam.getModelId()) .withURIVariable(API.PARAM_PARENT_MODEL_ID, exam.getModelId())
@ -205,7 +212,8 @@ public class MonitoringClientConnection implements TemplateComposer {
pageContext.copyOf(content), pageContext.copyOf(content),
exam, exam,
getConnectionData, getConnectionData,
indicators); indicators,
clientGroups);
// NOTIFICATIONS // NOTIFICATIONS
Supplier<EntityTable<ClientNotification>> _notificationTableSupplier = () -> null; Supplier<EntityTable<ClientNotification>> _notificationTableSupplier = () -> null;

View file

@ -30,6 +30,7 @@ import ch.ethz.seb.sebserver.gbl.api.EntityType;
import ch.ethz.seb.sebserver.gbl.async.AsyncRunner; import ch.ethz.seb.sebserver.gbl.async.AsyncRunner;
import ch.ethz.seb.sebserver.gbl.model.Domain; import ch.ethz.seb.sebserver.gbl.model.Domain;
import ch.ethz.seb.sebserver.gbl.model.EntityKey; import ch.ethz.seb.sebserver.gbl.model.EntityKey;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings;
@ -56,6 +57,7 @@ import ch.ethz.seb.sebserver.gui.service.push.ServerPushService;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExam; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExam;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExamProctoringSettings; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.GetExamProctoringSettings;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.clientgroup.GetClientGroups;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.indicator.GetIndicators; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.exam.indicator.GetIndicators;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.CurrentUser; import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.CurrentUser;
import ch.ethz.seb.sebserver.gui.service.session.ClientConnectionTable; import ch.ethz.seb.sebserver.gui.service.session.ClientConnectionTable;
@ -140,6 +142,11 @@ public class MonitoringRunningExam implements TemplateComposer {
.call() .call()
.getOrThrow(); .getOrThrow();
final Collection<ClientGroup> clientGroups = this.restService.getBuilder(GetClientGroups.class)
.withQueryParam(Indicator.FILTER_ATTR_EXAM_ID, entityKey.modelId)
.call()
.getOrThrow();
final Composite content = this.pageService.getWidgetFactory().defaultPageLayout( final Composite content = this.pageService.getWidgetFactory().defaultPageLayout(
pageContext.getParent(), pageContext.getParent(),
new LocTextKey("sebserver.monitoring.exam", exam.name)); new LocTextKey("sebserver.monitoring.exam", exam.name));
@ -166,6 +173,7 @@ public class MonitoringRunningExam implements TemplateComposer {
tablePane, tablePane,
exam, exam,
indicators, indicators,
clientGroups,
this.distributedSetup); this.distributedSetup);
guiUpdates.add(clientTable); guiUpdates.add(clientTable);

View file

@ -128,6 +128,11 @@ public final class Form implements FormBinding {
return this; return this;
} }
Form putReadonlyField(final String name, final Control label, final Label field) {
this.formFields.add(name, createReadonlyAccessor(label, field));
return this;
}
Form putReadonlyField(final String name, final Control label, final Browser field) { Form putReadonlyField(final String name, final Control label, final Browser field) {
this.formFields.add(name, createReadonlyAccessor(label, field)); this.formFields.add(name, createReadonlyAccessor(label, field));
return this; return this;
@ -308,7 +313,13 @@ public final class Form implements FormBinding {
//@formatter:off //@formatter:off
private FormFieldAccessor createReadonlyAccessor(final Control label, final Text field) { private FormFieldAccessor createReadonlyAccessor(final Control label, final Text field) {
return new FormFieldAccessor(label, field, null) { return new FormFieldAccessor(label, field, null) {
@Override public String getStringValue() { return null; } @Override public String getStringValue() { return field.getText(); }
@Override public void setStringValue(final String value) { field.setText( (value == null) ? StringUtils.EMPTY : value); }
};
}
private FormFieldAccessor createReadonlyAccessor(final Control label, final Label field) {
return new FormFieldAccessor(label, field, null) {
@Override public String getStringValue() { return field.getText(); }
@Override public void setStringValue(final String value) { field.setText( (value == null) ? StringUtils.EMPTY : value); } @Override public void setStringValue(final String value) { field.setText( (value == null) ? StringUtils.EMPTY : value); }
}; };
} }

View file

@ -39,6 +39,7 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
int areaMinHeight = WidgetFactory.TEXT_AREA_INPUT_MIN_HEIGHT; int areaMinHeight = WidgetFactory.TEXT_AREA_INPUT_MIN_HEIGHT;
boolean isColorBox = false; boolean isColorBox = false;
boolean isHTML = false; boolean isHTML = false;
boolean isMarkup = false;
TextFieldBuilder(final String name, final LocTextKey label, final String value) { TextFieldBuilder(final String name, final LocTextKey label, final String value) {
super(name, label, value); super(name, label, value);
@ -76,6 +77,11 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
return this; return this;
} }
public TextFieldBuilder asMarkup() {
this.isMarkup = true;
return this;
}
public TextFieldBuilder asHTML(final int minHeight) { public TextFieldBuilder asHTML(final int minHeight) {
this.isHTML = true; this.isHTML = true;
this.areaMinHeight = minHeight; this.areaMinHeight = minHeight;
@ -118,6 +124,15 @@ public final class TextFieldBuilder extends FieldBuilder<String> {
return; return;
} }
if (readonly && this.isMarkup) {
final Label label = new Label(fieldGrid, SWT.NONE);
label.setData(RWT.MARKUP_ENABLED, Boolean.TRUE);
label.setText(this.value);
label.setLayoutData(new GridData(SWT.FILL, SWT.TOP, true, true));
builder.form.putReadonlyField(this.name, titleLabel, label);
return;
}
final String testKey = (this.label != null) ? this.label.name : this.name; final String testKey = (this.label != null) ? this.label.name : this.name;
final LocTextKey label = getARIALabel(builder); final LocTextKey label = getARIALabel(builder);
final Text textInput = (this.isNumber) final Text textInput = (this.isNumber)

View file

@ -13,6 +13,7 @@ import java.util.Map;
import java.util.function.Consumer; import java.util.function.Consumer;
import java.util.function.Function; import java.util.function.Function;
import java.util.function.Supplier; import java.util.function.Supplier;
import java.util.stream.Collectors;
import org.apache.commons.lang3.BooleanUtils; import org.apache.commons.lang3.BooleanUtils;
import org.eclipse.swt.graphics.Color; import org.eclipse.swt.graphics.Color;
@ -22,6 +23,7 @@ import org.slf4j.LoggerFactory;
import ch.ethz.seb.sebserver.gbl.Constants; import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.model.Domain; import ch.ethz.seb.sebserver.gbl.model.Domain;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
import ch.ethz.seb.sebserver.gbl.model.exam.QuizData; import ch.ethz.seb.sebserver.gbl.model.exam.QuizData;
@ -43,6 +45,7 @@ import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.DisposedOAuth2RestTemplateException; import ch.ethz.seb.sebserver.gui.service.remote.webservice.auth.DisposedOAuth2RestTemplateException;
import ch.ethz.seb.sebserver.gui.service.session.IndicatorData.ThresholdColor; import ch.ethz.seb.sebserver.gui.service.session.IndicatorData.ThresholdColor;
import ch.ethz.seb.sebserver.gui.table.EntityTable; import ch.ethz.seb.sebserver.gui.table.EntityTable;
import ch.ethz.seb.sebserver.gui.widget.WidgetFactory;
public class ClientConnectionDetails { public class ClientConnectionDetails {
@ -52,6 +55,8 @@ public class ClientConnectionDetails {
new LocTextKey("sebserver.monitoring.connection.form.exam"); new LocTextKey("sebserver.monitoring.connection.form.exam");
private final static LocTextKey CONNECTION_ID_TEXT_KEY = private final static LocTextKey CONNECTION_ID_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.form.id"); new LocTextKey("sebserver.monitoring.connection.form.id");
private final static LocTextKey CONNECTION_GROUP_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.form.group");
private final static LocTextKey CONNECTION_INFO_TEXT_KEY = private final static LocTextKey CONNECTION_INFO_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.form.info"); new LocTextKey("sebserver.monitoring.connection.form.info");
private final static LocTextKey CONNECTION_STATUS_TEXT_KEY = private final static LocTextKey CONNECTION_STATUS_TEXT_KEY =
@ -62,6 +67,8 @@ public class ClientConnectionDetails {
private final PageService pageService; private final PageService pageService;
private final ResourceService resourceService; private final ResourceService resourceService;
private final Map<Long, IndicatorData> indicatorMapping; private final Map<Long, IndicatorData> indicatorMapping;
private final Map<Long, ClientGroup> clientGroupMapping;
private final boolean hasClientGroups;
private final RestCall<ClientConnectionData>.RestCallBuilder restCallBuilder; private final RestCall<ClientConnectionData>.RestCallBuilder restCallBuilder;
private final FormHandle<?> formHandle; private final FormHandle<?> formHandle;
private final ColorData colorData; private final ColorData colorData;
@ -78,10 +85,11 @@ public class ClientConnectionDetails {
final PageContext pageContext, final PageContext pageContext,
final Exam exam, final Exam exam,
final RestCall<ClientConnectionData>.RestCallBuilder restCallBuilder, final RestCall<ClientConnectionData>.RestCallBuilder restCallBuilder,
final Collection<Indicator> indicators) { final Collection<Indicator> indicators,
final Collection<ClientGroup> clientGroups) {
final Display display = pageContext.getRoot().getDisplay(); final Display display = pageContext.getRoot().getDisplay();
this.hasClientGroups = clientGroups != null && !clientGroups.isEmpty();
this.pageService = pageService; this.pageService = pageService;
this.resourceService = pageService.getResourceService(); this.resourceService = pageService.getResourceService();
this.restCallBuilder = restCallBuilder; this.restCallBuilder = restCallBuilder;
@ -92,6 +100,12 @@ public class ClientConnectionDetails {
this.colorData, this.colorData,
NUMBER_OF_NONE_INDICATOR_ROWS); NUMBER_OF_NONE_INDICATOR_ROWS);
this.clientGroupMapping = clientGroups == null || clientGroups.isEmpty()
? null
: clientGroups
.stream()
.collect(Collectors.toMap(cg -> cg.id, Function.identity()));
final FormBuilder formBuilder = pageService.formBuilder(pageContext) final FormBuilder formBuilder = pageService.formBuilder(pageContext)
.readonly(true) .readonly(true)
.addField(FormBuilder.text( .addField(FormBuilder.text(
@ -102,6 +116,12 @@ public class ClientConnectionDetails {
Domain.CLIENT_CONNECTION.ATTR_EXAM_USER_SESSION_ID, Domain.CLIENT_CONNECTION.ATTR_EXAM_USER_SESSION_ID,
CONNECTION_ID_TEXT_KEY, CONNECTION_ID_TEXT_KEY,
Constants.EMPTY_NOTE)) Constants.EMPTY_NOTE))
.addFieldIf(() -> this.hasClientGroups,
() -> FormBuilder.text(
ClientConnectionData.ATTR_CLIENT_GROUPS,
CONNECTION_GROUP_TEXT_KEY,
Constants.EMPTY_NOTE)
.asMarkup())
.addField(FormBuilder.text( .addField(FormBuilder.text(
ClientConnection.ATTR_INFO, ClientConnection.ATTR_INFO,
CONNECTION_INFO_TEXT_KEY, CONNECTION_INFO_TEXT_KEY,
@ -179,6 +199,13 @@ public class ClientConnectionDetails {
ClientConnection.ATTR_INFO, ClientConnection.ATTR_INFO,
this.connectionData.clientConnection.info); this.connectionData.clientConnection.info);
if (this.hasClientGroups
&& Constants.EMPTY_NOTE.equals(form.getFieldValue(ClientConnectionData.ATTR_CLIENT_GROUPS))) {
form.setFieldValue(
ClientConnectionData.ATTR_CLIENT_GROUPS,
getGroupInfo());
}
if (this.missingChanged) { if (this.missingChanged) {
// update status // update status
form.setFieldValue( form.setFieldValue(
@ -252,4 +279,20 @@ public class ClientConnectionDetails {
} }
} }
private String getGroupInfo() {
final StringBuilder sb = new StringBuilder();
this.clientGroupMapping.keySet().stream().forEach(key -> {
if (this.connectionData.groups.contains(key)) {
final ClientGroup clientGroup = this.clientGroupMapping.get(key);
sb.append(WidgetFactory.getTextWithBackgroundHTML(clientGroup.name, clientGroup.color));
}
});
if (sb.length() <= 0) {
return Constants.EMPTY_NOTE;
} else {
return sb.toString();
}
}
} }

View file

@ -26,6 +26,7 @@ import java.util.stream.Collectors;
import org.apache.commons.lang3.BooleanUtils; import org.apache.commons.lang3.BooleanUtils;
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.StringUtils;
import org.eclipse.rap.rwt.RWT;
import org.eclipse.swt.SWT; import org.eclipse.swt.SWT;
import org.eclipse.swt.graphics.Color; import org.eclipse.swt.graphics.Color;
import org.eclipse.swt.graphics.Rectangle; import org.eclipse.swt.graphics.Rectangle;
@ -44,12 +45,13 @@ import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.api.EntityType; import ch.ethz.seb.sebserver.gbl.api.EntityType;
import ch.ethz.seb.sebserver.gbl.model.Domain; import ch.ethz.seb.sebserver.gbl.model.Domain;
import ch.ethz.seb.sebserver.gbl.model.EntityKey; import ch.ethz.seb.sebserver.gbl.model.EntityKey;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.Indicator; import ch.ethz.seb.sebserver.gbl.model.exam.Indicator;
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus; import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection.ConnectionStatus;
import ch.ethz.seb.sebserver.gbl.monitoring.IndicatorValue;
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnectionData; import ch.ethz.seb.sebserver.gbl.model.session.ClientConnectionData;
import ch.ethz.seb.sebserver.gbl.monitoring.IndicatorValue;
import ch.ethz.seb.sebserver.gbl.util.Tuple; import ch.ethz.seb.sebserver.gbl.util.Tuple;
import ch.ethz.seb.sebserver.gui.service.ResourceService; import ch.ethz.seb.sebserver.gui.service.ResourceService;
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey; import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
@ -60,31 +62,35 @@ import ch.ethz.seb.sebserver.gui.widget.WidgetFactory;
public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate { public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate {
private static final int[] TABLE_PROPORTIONS = new int[] { 3, 3, 2, 1 };
private static final int BOTTOM_PADDING = 20; private static final int BOTTOM_PADDING = 20;
private static final int NUMBER_OF_NONE_INDICATOR_COLUMNS = 3; //private static final int[] TABLE_PROPORTIONS = new int[] { 3, 3, 2, 1 };
//private static final int NUMBER_OF_NONE_INDICATOR_COLUMNS = 3;
private static final String INDICATOR_NAME_TEXT_KEY_PREFIX = private static final String INDICATOR_NAME_TEXT_KEY_PREFIX =
"sebserver.exam.indicator.type.description."; "sebserver.exam.indicator.type.description.";
private final static LocTextKey CONNECTION_ID_TEXT_KEY = private final static LocTextKey CONNECTION_ID_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.id"); new LocTextKey("sebserver.monitoring.connection.list.column.id");
private final static LocTextKey CONNECTION_ID_TOOLTIP_TEXT_KEY = private final static LocTextKey CONNECTION_ID_TOOLTIP_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.id" + Constants.TOOLTIP_TEXT_KEY_SUFFIX); new LocTextKey(CONNECTION_ID_TEXT_KEY + Constants.TOOLTIP_TEXT_KEY_SUFFIX);
private final static LocTextKey CONNECTION_GROUP_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.group");
private final static LocTextKey CONNECTION_GROUP_TOOLTIP_TEXT_KEY =
new LocTextKey(CONNECTION_GROUP_TEXT_KEY + Constants.TOOLTIP_TEXT_KEY_SUFFIX);
private final static LocTextKey CONNECTION_INFO_TEXT_KEY = private final static LocTextKey CONNECTION_INFO_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.info"); new LocTextKey("sebserver.monitoring.connection.list.column.info");
private final static LocTextKey CONNECTION_INFO_TOOLTIP_TEXT_KEY = private final static LocTextKey CONNECTION_INFO_TOOLTIP_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.info" + Constants.TOOLTIP_TEXT_KEY_SUFFIX); new LocTextKey(CONNECTION_INFO_TEXT_KEY + Constants.TOOLTIP_TEXT_KEY_SUFFIX);
private final static LocTextKey CONNECTION_STATUS_TEXT_KEY = private final static LocTextKey CONNECTION_STATUS_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.status"); new LocTextKey("sebserver.monitoring.connection.list.column.status");
private final static LocTextKey CONNECTION_STATUS_TOOLTIP_TEXT_KEY = private final static LocTextKey CONNECTION_STATUS_TOOLTIP_TEXT_KEY =
new LocTextKey("sebserver.monitoring.connection.list.column.status" + Constants.TOOLTIP_TEXT_KEY_SUFFIX); new LocTextKey(CONNECTION_STATUS_TEXT_KEY + Constants.TOOLTIP_TEXT_KEY_SUFFIX);
private final PageService pageService; private final PageService pageService;
private final Exam exam; private final Exam exam;
private final boolean distributedSetup; private final boolean distributedSetup;
private final Map<Long, IndicatorData> indicatorMapping; private final Map<Long, IndicatorData> indicatorMapping;
private final Map<Long, ClientGroup> clientGroupMapping;
private final Table table; private final Table table;
private final ColorData colorData; private final ColorData colorData;
private final Function<ClientConnectionData, String> localizedClientConnectionStatusNameFunction; private final Function<ClientConnectionData, String> localizedClientConnectionStatusNameFunction;
@ -99,6 +105,10 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
private final Color darkFontColor; private final Color darkFontColor;
private final Color lightFontColor; private final Color lightFontColor;
private final boolean hasClientGroups;
private final int numberOfNoneIndicatorColumns;
private final int[] tableProportions;
private boolean forceUpdateAll = false; private boolean forceUpdateAll = false;
public ClientConnectionTable( public ClientConnectionTable(
@ -106,6 +116,7 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
final Composite tableRoot, final Composite tableRoot,
final Exam exam, final Exam exam,
final Collection<Indicator> indicators, final Collection<Indicator> indicators,
final Collection<ClientGroup> clientGroups,
final boolean distributedSetup) { final boolean distributedSetup) {
this.pageService = pageService; this.pageService = pageService;
@ -121,11 +132,23 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
this.darkFontColor = new Color(display, Constants.BLACK_RGB); this.darkFontColor = new Color(display, Constants.BLACK_RGB);
this.lightFontColor = new Color(display, Constants.WHITE_RGB); this.lightFontColor = new Color(display, Constants.WHITE_RGB);
this.hasClientGroups = clientGroups != null && !clientGroups.isEmpty();
this.numberOfNoneIndicatorColumns = this.hasClientGroups ? 4 : 3;
this.tableProportions = this.hasClientGroups
? new int[] { 3, 2, 3, 2, 1 }
: new int[] { 3, 3, 2, 1 };
this.indicatorMapping = IndicatorData.createFormIndicators( this.indicatorMapping = IndicatorData.createFormIndicators(
indicators, indicators,
display, display,
this.colorData, this.colorData,
NUMBER_OF_NONE_INDICATOR_COLUMNS); this.numberOfNoneIndicatorColumns);
this.clientGroupMapping = clientGroups == null || clientGroups.isEmpty()
? null
: clientGroups
.stream()
.collect(Collectors.toMap(cg -> cg.id, Function.identity()));
this.localizedClientConnectionStatusNameFunction = this.localizedClientConnectionStatusNameFunction =
resourceService.localizedClientConnectionStatusNameFunction(); resourceService.localizedClientConnectionStatusNameFunction();
@ -136,11 +159,11 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
gridLayout.marginWidth = 100; gridLayout.marginWidth = 100;
gridLayout.marginRight = 100; gridLayout.marginRight = 100;
this.table.setLayout(gridLayout); this.table.setLayout(gridLayout);
final GridData gridData = new GridData(SWT.FILL, SWT.TOP, true, false); final GridData gridData = new GridData(SWT.FILL, SWT.FILL, true, true);
this.table.setLayoutData(gridData); this.table.setLayoutData(gridData);
this.table.setHeaderVisible(true); this.table.setHeaderVisible(true);
this.table.setLinesVisible(true); this.table.setLinesVisible(true);
this.table.setData(RWT.MARKUP_ENABLED, Boolean.TRUE);
this.table.addListener(SWT.Selection, event -> this.notifySelectionChange()); this.table.addListener(SWT.Selection, event -> this.notifySelectionChange());
this.table.addListener(SWT.MouseUp, this::notifyTableInfoClick); this.table.addListener(SWT.MouseUp, this::notifyTableInfoClick);
@ -148,6 +171,12 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
this.table, this.table,
CONNECTION_ID_TEXT_KEY, CONNECTION_ID_TEXT_KEY,
CONNECTION_ID_TOOLTIP_TEXT_KEY); CONNECTION_ID_TOOLTIP_TEXT_KEY);
if (this.clientGroupMapping != null && !this.clientGroupMapping.isEmpty()) {
widgetFactory.tableColumnLocalized(
this.table,
CONNECTION_GROUP_TEXT_KEY,
CONNECTION_GROUP_TOOLTIP_TEXT_KEY);
}
widgetFactory.tableColumnLocalized( widgetFactory.tableColumnLocalized(
this.table, this.table,
CONNECTION_INFO_TEXT_KEY, CONNECTION_INFO_TEXT_KEY,
@ -285,10 +314,10 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
@Override @Override
public void update(final MonitoringStatus monitoringStatus) { public void update(final MonitoringStatus monitoringStatus) {
final Collection<ClientConnectionData> connectionData = monitoringStatus.getConnectionData(); final Collection<ClientConnectionData> connectionData = monitoringStatus.getConnectionData();
final boolean sizeChanged = connectionData.size() != this.table.getItemCount();
final boolean needsSync = monitoringStatus.statusFilterChanged() || final boolean needsSync = monitoringStatus.statusFilterChanged() ||
this.forceUpdateAll || this.forceUpdateAll ||
connectionData.size() != this.table.getItemCount() || sizeChanged ||
(this.tableMapping != null && (this.tableMapping != null &&
this.table != null && this.table != null &&
this.tableMapping.size() != this.table.getItemCount()) this.tableMapping.size() != this.table.getItemCount())
@ -326,6 +355,7 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
} }
this.forceUpdateAll = false; this.forceUpdateAll = false;
this.needsSort = sizeChanged;
updateGUI(); updateGUI();
} }
@ -352,19 +382,21 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
if (this.tableWidth != area.width) { if (this.tableWidth != area.width) {
// proportions size // proportions size
final int pSize = TABLE_PROPORTIONS[0] + final int pSize = this.tableProportions[0] +
TABLE_PROPORTIONS[1] + this.tableProportions[1] +
TABLE_PROPORTIONS[2] + this.tableProportions[2] +
TABLE_PROPORTIONS[3] * this.indicatorMapping.size(); this.tableProportions[3] +
(this.hasClientGroups ? this.tableProportions[4] : 0)
* this.indicatorMapping.size();
final int columnUnitSize = (pSize > 0) final int columnUnitSize = (pSize > 0)
? area.width / pSize ? area.width / pSize
: area.width / TABLE_PROPORTIONS.length - 1 + this.indicatorMapping.size(); : area.width / this.tableProportions.length - 1 + this.indicatorMapping.size();
final TableColumn[] columns = this.table.getColumns(); final TableColumn[] columns = this.table.getColumns();
for (int i = 0; i < columns.length; i++) { for (int i = 0; i < columns.length; i++) {
final int proportionFactor = (i < TABLE_PROPORTIONS.length) final int proportionFactor = (i < this.tableProportions.length)
? TABLE_PROPORTIONS[i] ? this.tableProportions[i]
: TABLE_PROPORTIONS[TABLE_PROPORTIONS.length - 1]; : this.tableProportions[this.tableProportions.length - 1];
columns[i].setWidth(proportionFactor * columnUnitSize); columns[i].setWidth(proportionFactor * columnUnitSize);
} }
this.tableWidth = area.width; this.tableWidth = area.width;
@ -412,7 +444,8 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
private final class UpdatableTableItem implements Comparable<UpdatableTableItem> { private final class UpdatableTableItem implements Comparable<UpdatableTableItem> {
final Long connectionId; final Long connectionId;
private boolean changed = false; private boolean dataChanged = false;
private boolean indicatorValueChanged = false;
private ClientConnectionData connectionData; private ClientConnectionData connectionData;
private int thresholdsWeight; private int thresholdsWeight;
private int[] indicatorWeights = null; private int[] indicatorWeights = null;
@ -424,17 +457,34 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
} }
private void update(final TableItem tableItem, final boolean force) { private void update(final TableItem tableItem, final boolean force) {
if (force || this.changed) { if (force || this.dataChanged) {
update(tableItem); updateData(tableItem);
} }
this.changed = false; if (force || this.indicatorValueChanged) {
updateIndicatorValues(tableItem);
}
this.dataChanged = false;
this.indicatorValueChanged = false;
} }
private void update(final TableItem tableItem) { private void updateData(final TableItem tableItem) {
updateData(tableItem); tableItem.setText(0, getConnectionIdentifier());
if (ClientConnectionTable.this.hasClientGroups) {
tableItem.setText(1, getGroupInfo());
tableItem.setText(2, getConnectionInfo());
tableItem.setText(
3,
ClientConnectionTable.this.localizedClientConnectionStatusNameFunction
.apply(this.connectionData));
} else {
tableItem.setText(1, getConnectionInfo());
tableItem.setText(
2,
ClientConnectionTable.this.localizedClientConnectionStatusNameFunction
.apply(this.connectionData));
}
if (this.connectionData != null) { if (this.connectionData != null) {
updateConnectionStatusColor(tableItem); updateConnectionStatusColor(tableItem);
updateIndicatorValues(tableItem);
updateDuplicateColor(tableItem); updateDuplicateColor(tableItem);
updateNotifications(tableItem); updateNotifications(tableItem);
} }
@ -451,19 +501,12 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
} }
} }
private void updateData(final TableItem tableItem) {
tableItem.setText(0, getConnectionIdentifier());
tableItem.setText(1, getConnectionInfo());
tableItem.setText(
2,
ClientConnectionTable.this.localizedClientConnectionStatusNameFunction.apply(this.connectionData));
}
private void updateConnectionStatusColor(final TableItem tableItem) { private void updateConnectionStatusColor(final TableItem tableItem) {
final Color statusColor = ClientConnectionTable.this.colorData.getStatusColor(this.connectionData); final Color statusColor = ClientConnectionTable.this.colorData.getStatusColor(this.connectionData);
final Color statusTextColor = ClientConnectionTable.this.colorData.getStatusTextColor(statusColor); final Color statusTextColor = ClientConnectionTable.this.colorData.getStatusTextColor(statusColor);
tableItem.setBackground(2, statusColor); final int index = ClientConnectionTable.this.hasClientGroups ? 3 : 2;
tableItem.setForeground(2, statusTextColor); tableItem.setBackground(index, statusColor);
tableItem.setForeground(index, statusTextColor);
} }
private void updateDuplicateColor(final TableItem tableItem) { private void updateDuplicateColor(final TableItem tableItem) {
@ -578,6 +621,22 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
return Constants.EMPTY_NOTE; return Constants.EMPTY_NOTE;
} }
private String getGroupInfo() {
final StringBuilder sb = new StringBuilder();
ClientConnectionTable.this.clientGroupMapping.keySet().stream().forEach(key -> {
if (this.connectionData.groups.contains(key)) {
final ClientGroup clientGroup = ClientConnectionTable.this.clientGroupMapping.get(key);
sb.append(WidgetFactory.getTextWithBackgroundHTML(clientGroup.name, clientGroup.color));
}
});
if (sb.length() <= 0) {
return Constants.EMPTY_NOTE;
} else {
return sb.toString();
}
}
String getConnectionIdentifier() { String getConnectionIdentifier() {
if (this.connectionData != null && this.connectionData.clientConnection.userSessionId != null) { if (this.connectionData != null && this.connectionData.clientConnection.userSessionId != null) {
return this.connectionData.clientConnection.userSessionId; return this.connectionData.clientConnection.userSessionId;
@ -587,15 +646,16 @@ public final class ClientConnectionTable implements FullPageMonitoringGUIUpdate
} }
void push(final ClientConnectionData connectionData) { void push(final ClientConnectionData connectionData) {
this.changed = this.connectionData == null || this.dataChanged = this.connectionData == null ||
!this.connectionData.dataEquals(connectionData); !this.connectionData.dataEquals(connectionData);
final boolean statusChanged = this.connectionData == null || this.indicatorValueChanged = this.connectionData == null ||
this.connectionData.clientConnection.status != connectionData.clientConnection.status; (this.connectionData.clientConnection.status.clientActiveStatus
&& !this.connectionData.indicatorValuesEquals(connectionData));
final boolean notificationChanged = this.connectionData == null || final boolean notificationChanged = this.connectionData == null ||
BooleanUtils.toBoolean(this.connectionData.pendingNotification) != BooleanUtils BooleanUtils.toBoolean(this.connectionData.pendingNotification) != BooleanUtils
.toBoolean(connectionData.pendingNotification); .toBoolean(connectionData.pendingNotification);
if (statusChanged || notificationChanged) { if (this.dataChanged || notificationChanged) {
ClientConnectionTable.this.needsSort = true; ClientConnectionTable.this.needsSort = true;
} }

View file

@ -1053,25 +1053,27 @@ public class WidgetFactory {
} }
} }
public String getColorValueHTML(final ClientGroupData data) { public static String getTextWithBackgroundHTML(final String text, final String color) {
final String color = data.getColor(); return new StringBuilder().append("<span style='padding: 5px 5px 5px 5px; background-color: #")
if (StringUtils.isBlank(color)) {
return Constants.EMPTY_NOTE;
}
return new StringBuilder().append("<span style='padding: 2px 5px 2px 5px; background-color: #")
.append(color) .append(color)
.append(";") .append(";")
.append((Utils.darkColorContrast(Utils.parseRGB(color))) .append((Utils.darkColorContrast(Utils.parseRGB(color)))
? "color: #4a4a4a; " ? "color: #4a4a4a; "
: "color: #FFFFFF;") : "color: #FFFFFF;")
.append("'>&nbsp;&nbsp;&nbsp;") .append("'>")
.append(" (#") .append(text)
.append(color) // .append("&nbsp;")
.append(")&nbsp;&nbsp;&nbsp;")
.append("</span>") .append("</span>")
.toString(); .toString();
}
public static String getColorValueHTML(final ClientGroupData data) {
final String color = data.getColor();
if (StringUtils.isBlank(color)) {
return Constants.EMPTY_NOTE;
}
return getTextWithBackgroundHTML(color, color);
} }
} }

View file

@ -1918,6 +1918,8 @@ sebserver.monitoring.exam=Monitoring Exam: {0}
sebserver.monitoring.connection.list.column.id=User Name or Session sebserver.monitoring.connection.list.column.id=User Name or Session
sebserver.monitoring.connection.list.column.id.tooltip=The user session identifier or username sent by the SEB client after LMS login sebserver.monitoring.connection.list.column.id.tooltip=The user session identifier or username sent by the SEB client after LMS login
sebserver.monitoring.connection.list.column.group=Client Group(s)
sebserver.monitoring.connection.list.column.group.tooltip=The client group(s) the connection belongs to
sebserver.monitoring.connection.list.column.info=Connection Info sebserver.monitoring.connection.list.column.info=Connection Info
sebserver.monitoring.connection.list.column.info.tooltip=Format: IP Address,SEB Version, OSName sebserver.monitoring.connection.list.column.info.tooltip=Format: IP Address,SEB Version, OSName
sebserver.monitoring.connection.list.column.status=Status sebserver.monitoring.connection.list.column.status=Status
@ -1925,6 +1927,8 @@ sebserver.monitoring.connection.list.column.status.tooltip=The current connectio
sebserver.monitoring.connection.form.id=User Name or Session sebserver.monitoring.connection.form.id=User Name or Session
sebserver.monitoring.connection.form.id.tooltip=The user session identifier or username sent by the SEB client after LMS login sebserver.monitoring.connection.form.id.tooltip=The user session identifier or username sent by the SEB client after LMS login
sebserver.monitoring.connection.form.group=Client Group(s)
sebserver.monitoring.connection.form.group.tooltip=The client groups this SEB client belongs to
sebserver.monitoring.connection.form.info=Connection Info sebserver.monitoring.connection.form.info=Connection Info
sebserver.monitoring.connection.form.info.tooltip=Format: IP Address,SEB Version, OSName sebserver.monitoring.connection.form.info.tooltip=Format: IP Address,SEB Version, OSName
sebserver.monitoring.connection.form.status=Status sebserver.monitoring.connection.form.status=Status

View file

@ -25,6 +25,7 @@ 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.JSONMapper; import ch.ethz.seb.sebserver.gbl.api.JSONMapper;
import ch.ethz.seb.sebserver.gbl.model.exam.Chapters; import ch.ethz.seb.sebserver.gbl.model.exam.Chapters;
import ch.ethz.seb.sebserver.gbl.model.exam.ClientGroup;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam.ExamStatus; import ch.ethz.seb.sebserver.gbl.model.exam.Exam.ExamStatus;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam.ExamType; import ch.ethz.seb.sebserver.gbl.model.exam.Exam.ExamType;
@ -279,6 +280,13 @@ public class ModelObjectJSONGenerator {
System.out.println(domainObject.getClass().getSimpleName() + ":"); System.out.println(domainObject.getClass().getSimpleName() + ":");
System.out.println(writerWithDefaultPrettyPrinter.writeValueAsString(domainObject)); System.out.println(writerWithDefaultPrettyPrinter.writeValueAsString(domainObject));
((ClientConnectionData) domainObject)
.addToClientGroup(new ClientGroup(1L, 1L, "group1", null, null, null, null));
((ClientConnectionData) domainObject)
.addToClientGroup(new ClientGroup(2L, 1L, "group2", null, null, null, null));
System.out.println(domainObject.getClass().getSimpleName() + ":");
System.out.println(writerWithDefaultPrettyPrinter.writeValueAsString(domainObject));
domainObject = new ClientEvent(1L, 1L, EventType.WARN_LOG, domainObject = new ClientEvent(1L, 1L, EventType.WARN_LOG,
System.currentTimeMillis(), System.currentTimeMillis(), 123.0, "text"); System.currentTimeMillis(), System.currentTimeMillis(), 123.0, "text");
System.out.println(domainObject.getClass().getSimpleName() + ":"); System.out.println(domainObject.getClass().getSimpleName() + ":");