SEBSERV-180 fixes

This commit is contained in:
anhefti 2021-05-18 15:16:36 +02:00
parent 7a686c0e6e
commit a172326fa5
10 changed files with 200 additions and 107 deletions

View file

@ -382,7 +382,7 @@ public final class Result<T> {
@Override @Override
public String toString() { public String toString() {
throw new RuntimeException("!!!!!!!!!!!!!"); throw new RuntimeException("Result.toString is probably called by mistake !!!");
//return "Result [value=" + this.value + ", error=" + this.error + "]"; //return "Result [value=" + this.value + ", error=" + this.error + "]";
} }

View file

@ -371,15 +371,18 @@ public class MonitoringClientConnection implements TemplateComposer {
.call() .call()
.onError(error -> log.error("Failed to get ProctoringServiceSettings", error)) .onError(error -> log.error("Failed to get ProctoringServiceSettings", error))
.getOr(null); .getOr(null);
final ProctoringGUIService proctoringGUIService = currentUser.getProctoringGUIService();
if (procotringSettings != null && procotringSettings.enableProctoring) { if (procotringSettings != null && procotringSettings.enableProctoring) {
final ProctoringGUIService proctoringGUIService = this.resourceService
.getCurrentUser()
.getProctoringGUIService();
actionBuilder actionBuilder
.newAction(ActionDefinition.MONITOR_EXAM_CLIENT_CONNECTION_PROCTORING) .newAction(ActionDefinition.MONITOR_EXAM_CLIENT_CONNECTION_PROCTORING)
.withEntityKey(parentEntityKey) .withEntityKey(parentEntityKey)
.withExec(action -> this.monitoringProctoringService.openOneToOneRoom( .withExec(action -> this.monitoringProctoringService.openOneToOneRoom(
action, action,
connectionData, proctoringGUIService)) connectionData,
proctoringGUIService))
.noEventPropagation() .noEventPropagation()
.publish() .publish()

View file

@ -36,6 +36,7 @@ import ch.ethz.seb.sebserver.gui.service.i18n.PolyglotPageService;
import ch.ethz.seb.sebserver.gui.service.page.PageContext; import ch.ethz.seb.sebserver.gui.service.page.PageContext;
import ch.ethz.seb.sebserver.gui.service.page.PageService; import ch.ethz.seb.sebserver.gui.service.page.PageService;
import ch.ethz.seb.sebserver.gui.service.page.TemplateComposer; import ch.ethz.seb.sebserver.gui.service.page.TemplateComposer;
import ch.ethz.seb.sebserver.gui.service.page.event.ActionActivationEvent;
import ch.ethz.seb.sebserver.gui.service.page.event.ActionActivationEventListener; import ch.ethz.seb.sebserver.gui.service.page.event.ActionActivationEventListener;
import ch.ethz.seb.sebserver.gui.service.page.event.ActionPublishEventListener; import ch.ethz.seb.sebserver.gui.service.page.event.ActionPublishEventListener;
import ch.ethz.seb.sebserver.gui.service.page.event.PageEventListener; import ch.ethz.seb.sebserver.gui.service.page.event.PageEventListener;
@ -118,19 +119,7 @@ public class ActionPane implements TemplateComposer {
continue; continue;
} }
final PageAction action = (PageAction) actionItem.getData(ACTION_EVENT_CALL_KEY); de_activate_action_icon(event, parent, actionItem);
final Image image = event.activation
? ad.icon.getImage(parent.getDisplay())
: ad.icon.getGreyedImage(parent.getDisplay());
actionItem.setImage(image);
if (event.activation) {
actionItem.setForeground(null);
} else {
actionItem.setForeground(new Color(parent.getDisplay(), new RGBA(150, 150, 150, 50)));
ActionPane.this.pageService.getPolyglotPageService().injectI18n(
actionItem,
(action != null) ? action.getTitle() : ad.title);
}
} }
if (event.decoration != null) { if (event.decoration != null) {
@ -141,10 +130,33 @@ public class ActionPane implements TemplateComposer {
ActionPane.this.pageService.getPolyglotPageService().injectI18n( ActionPane.this.pageService.getPolyglotPageService().injectI18n(
actionItemToDecorate, event.decoration._2.title); actionItemToDecorate, event.decoration._2.title);
} }
de_activate_action_icon(event, parent, actionItemToDecorate);
} }
}); });
} }
private void de_activate_action_icon(
final ActionActivationEvent event,
final Composite parent,
final TreeItem actionItemToDecorate) {
final PageAction action = (PageAction) actionItemToDecorate.getData(ACTION_EVENT_CALL_KEY);
final Image image = event.activation
? event.decoration._1.icon.getImage(parent.getDisplay())
: event.decoration._1.icon.getGreyedImage(parent.getDisplay());
actionItemToDecorate.setImage(image);
if (event.activation) {
actionItemToDecorate.setForeground(null);
} else {
actionItemToDecorate
.setForeground(new Color(parent.getDisplay(), new RGBA(150, 150, 150, 50)));
ActionPane.this.pageService.getPolyglotPageService().injectI18n(
actionItemToDecorate,
(action != null) ? action.getTitle() : event.decoration._1.title);
}
}
private TreeItem findAction( private TreeItem findAction(
final Map<String, Tree> actionTrees, final Map<String, Tree> actionTrees,
final Composite parent, final Composite parent,

View file

@ -44,6 +44,7 @@ public class PageContextImpl implements PageContext {
private static final Logger log = LoggerFactory.getLogger(PageContextImpl.class); private static final Logger log = LoggerFactory.getLogger(PageContextImpl.class);
private static final LocTextKey UNEXPECTED_ERROR = new LocTextKey("sebserver.error.unexpected");
private static final String ENTITY_LIST_TYPE = null; private static final String ENTITY_LIST_TYPE = null;
private final I18nSupport i18nSupport; private final I18nSupport i18nSupport;
@ -312,6 +313,18 @@ public class PageContextImpl implements PageContext {
@Override @Override
public void notifyError(final LocTextKey message, final Exception error) { public void notifyError(final LocTextKey message, final Exception error) {
if (error == null) {
final MessageBox messageBox = new Message(
getShell(),
this.i18nSupport.getText(UNEXPECTED_ERROR),
this.i18nSupport.getText(message),
SWT.ERROR,
this.i18nSupport);
messageBox.setMarkupEnabled(true);
messageBox.open(null);
return;
}
log.error("Unexpected GUI error notified: {}", error.getMessage()); log.error("Unexpected GUI error notified: {}", error.getMessage());
final String errorMessage = message != null final String errorMessage = message != null

View file

@ -59,6 +59,19 @@ public class MonitoringProctoringService {
private static final Logger log = LoggerFactory.getLogger(MonitoringProctoringService.class); private static final Logger log = LoggerFactory.getLogger(MonitoringProctoringService.class);
public static final LocTextKey OPEN_TOWNHALL_ERROR =
new LocTextKey("sebserver.exam.proctoring.townhall.open.error");
public static final LocTextKey CLOSE_TOWNHALL_ERROR =
new LocTextKey("sebserver.exam.proctoring.townhall.close.error");
public static final LocTextKey OPEN_ONE_ERROR =
new LocTextKey("sebserver.exam.proctoring.one.open.error");
public static final LocTextKey CLOSE_ONE_ERROR =
new LocTextKey("sebserver.exam.proctoring.one.close.error");
public static final LocTextKey OPEN_COLLECTING_ERROR =
new LocTextKey("sebserver.exam.proctoring.collecting.open.error");
public static final LocTextKey CLOSE_COLLECTING_ERROR =
new LocTextKey("sebserver.exam.proctoring.collecting.close.error");
private static final LocTextKey EXAM_ROOM_NAME = private static final LocTextKey EXAM_ROOM_NAME =
new LocTextKey("sebserver.monitoring.exam.proctoring.room.all.name"); new LocTextKey("sebserver.monitoring.exam.proctoring.room.all.name");
@ -109,26 +122,32 @@ public class MonitoringProctoringService {
final PageAction action) { final PageAction action) {
if (isTownhallRoomActive(action.getEntityKey().modelId)) { if (isTownhallRoomActive(action.getEntityKey().modelId)) {
closeTownhallRoom(proctoringGUIService, action); if (closeTownhallRoom(proctoringGUIService, action)) {
this.pageService.firePageEvent( this.pageService.firePageEvent(
new ActionActivationEvent( new ActionActivationEvent(
true, true,
new Tuple<>( new Tuple<>(
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM, ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM,
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM)), ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM)),
action.pageContext()); action.pageContext());
return action; } else {
action.pageContext().notifyError(CLOSE_TOWNHALL_ERROR, null);
}
} else { } else {
openTownhallRoom(proctoringGUIService, action); if (openTownhallRoom(proctoringGUIService, action)) {
this.pageService.firePageEvent( this.pageService.firePageEvent(
new ActionActivationEvent( new ActionActivationEvent(
true, true,
new Tuple<>( new Tuple<>(
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM, ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM,
ActionDefinition.MONITOR_EXAM_CLOSE_TOWNHALL_PROCTOR_ROOM)), ActionDefinition.MONITOR_EXAM_CLOSE_TOWNHALL_PROCTOR_ROOM)),
action.pageContext()); action.pageContext());
return action; } else {
action.pageContext().notifyError(OPEN_TOWNHALL_ERROR, null);
}
} }
return action;
} }
public void initCollectingRoomActions( public void initCollectingRoomActions(
@ -219,50 +238,56 @@ public class MonitoringProctoringService {
final PageAction action, final PageAction action,
final ClientConnectionData connectionData) { final ClientConnectionData connectionData) {
final String examId = action.getEntityKey().modelId; try {
final String examId = action.getEntityKey().modelId;
final ProctoringServiceSettings proctoringSettings = this.pageService.getRestService() final ProctoringServiceSettings proctoringSettings = this.pageService.getRestService()
.getBuilder(GetProctoringSettings.class) .getBuilder(GetProctoringSettings.class)
.withURIVariable(API.PARAM_MODEL_ID, examId) .withURIVariable(API.PARAM_MODEL_ID, examId)
.call()
.getOrThrow();
final Optional<RemoteProctoringRoom> roomOptional =
this.pageService.getRestService().getBuilder(GetCollectingRooms.class)
.withURIVariable(API.PARAM_MODEL_ID, examId)
.call()
.getOrThrow()
.stream()
.filter(room -> room.id.equals(connectionData.clientConnection.remoteProctoringRoomId))
.findFirst();
if (roomOptional.isPresent()) {
final RemoteProctoringRoom room = roomOptional.get();
final ProctoringRoomConnection proctoringConnectionData = this.pageService
.getRestService()
.getBuilder(GetProctorRoomConnection.class)
.withURIVariable(API.PARAM_MODEL_ID, String.valueOf(proctoringSettings.examId))
.withQueryParam(ProctoringRoomConnection.ATTR_ROOM_NAME, room.name)
.withQueryParam(ProctoringRoomConnection.ATTR_SUBJECT, Utils.encodeFormURL_UTF_8(room.subject))
.call() .call()
.getOrThrow(); .getOrThrow();
ProctoringGUIService.setCurrentProctoringWindowData(examId, proctoringConnectionData); final Optional<RemoteProctoringRoom> roomOptional =
final String script = String.format( this.pageService.getRestService().getBuilder(GetCollectingRooms.class)
MonitoringProctoringService.OPEN_ROOM_SCRIPT, .withURIVariable(API.PARAM_MODEL_ID, examId)
room.name, .call()
800, .getOrThrow()
1200, .stream()
this.guiServiceInfo.getExternalServerURIBuilder().toUriString(), .filter(room -> room.id.equals(connectionData.clientConnection.remoteProctoringRoomId))
this.remoteProctoringEndpoint); .findFirst();
RWT.getClient() if (roomOptional.isPresent()) {
.getService(JavaScriptExecutor.class) final RemoteProctoringRoom room = roomOptional.get();
.execute(script); final ProctoringRoomConnection proctoringConnectionData = this.pageService
.getRestService()
.getBuilder(GetProctorRoomConnection.class)
.withURIVariable(API.PARAM_MODEL_ID, String.valueOf(proctoringSettings.examId))
.withQueryParam(ProctoringRoomConnection.ATTR_ROOM_NAME, room.name)
.withQueryParam(ProctoringRoomConnection.ATTR_SUBJECT, Utils.encodeFormURL_UTF_8(room.subject))
.call()
.getOrThrow();
this.pageService.getCurrentUser() ProctoringGUIService.setCurrentProctoringWindowData(examId, proctoringConnectionData);
.getProctoringGUIService() final String script = String.format(
.registerProctoringWindow(examId, room.name, room.name); MonitoringProctoringService.OPEN_ROOM_SCRIPT,
room.name,
800,
1200,
this.guiServiceInfo.getExternalServerURIBuilder().toUriString(),
this.remoteProctoringEndpoint);
RWT.getClient()
.getService(JavaScriptExecutor.class)
.execute(script);
this.pageService.getCurrentUser()
.getProctoringGUIService()
.registerProctoringWindow(examId, room.name, room.name);
}
} catch (final Exception e) {
log.error("Failed to open popup for collecting room: ", e);
action.pageContext().notifyError(CLOSE_COLLECTING_ERROR, e);
} }
return action; return action;
@ -273,41 +298,49 @@ public class MonitoringProctoringService {
final ClientConnectionData connectionData, final ClientConnectionData connectionData,
final ProctoringGUIService proctoringGUIService) { final ProctoringGUIService proctoringGUIService) {
final String connectionToken = connectionData.clientConnection.connectionToken; try {
final String examId = action.getEntityKey().modelId;
if (!proctoringGUIService.hasWindow(connectionToken)) { final String connectionToken = connectionData.clientConnection.connectionToken;
final ProctoringRoomConnection proctoringConnectionData = proctoringGUIService final String examId = action.getEntityKey().modelId;
.openBreakOutRoom(
examId,
connectionToken,
connectionData.clientConnection.userSessionId,
Arrays.asList(connectionToken))
.onError(error -> log.error(
"Failed to open single proctoring room for connection {} {}",
connectionToken,
error.getMessage()))
.getOr(null);
ProctoringGUIService.setCurrentProctoringWindowData( if (!proctoringGUIService.hasWindow(connectionToken)) {
examId, final ProctoringRoomConnection proctoringConnectionData = proctoringGUIService
.openBreakOutRoom(
examId,
connectionToken,
connectionData.clientConnection.userSessionId,
Arrays.asList(connectionToken))
.onError(error -> log.error(
"Failed to open single proctoring room for connection {} {}",
connectionToken,
error.getMessage()))
.getOr(null);
ProctoringGUIService.setCurrentProctoringWindowData(
examId,
connectionToken,
proctoringConnectionData);
}
final JavaScriptExecutor javaScriptExecutor = RWT.getClient().getService(JavaScriptExecutor.class);
final String script = String.format(
MonitoringProctoringService.OPEN_ROOM_SCRIPT,
connectionToken, connectionToken,
proctoringConnectionData); 420,
640,
this.guiServiceInfo.getExternalServerURIBuilder().toUriString(),
this.remoteProctoringEndpoint);
javaScriptExecutor.execute(script);
} catch (final Exception e) {
log.error("Failed to open popup for one to one room: ", e);
action.pageContext().notifyError(OPEN_ONE_ERROR, e);
} }
final JavaScriptExecutor javaScriptExecutor = RWT.getClient().getService(JavaScriptExecutor.class);
final String script = String.format(
MonitoringProctoringService.OPEN_ROOM_SCRIPT,
connectionToken,
420,
640,
this.guiServiceInfo.getExternalServerURIBuilder().toUriString(),
this.remoteProctoringEndpoint);
javaScriptExecutor.execute(script);
return action; return action;
} }
private PageAction openTownhallRoom( private boolean openTownhallRoom(
final ProctoringGUIService proctoringGUIService, final ProctoringGUIService proctoringGUIService,
final PageAction action) { final PageAction action) {
@ -342,11 +375,13 @@ public class MonitoringProctoringService {
} catch (final Exception e) { } catch (final Exception e) {
log.error("Failed to open popup for town-hall room: ", e); log.error("Failed to open popup for town-hall room: ", e);
return false;
} }
return action;
return true;
} }
private PageAction closeTownhallRoom( private boolean closeTownhallRoom(
final ProctoringGUIService proctoringGUIService, final ProctoringGUIService proctoringGUIService,
final PageAction action) { final PageAction action) {
@ -360,8 +395,10 @@ public class MonitoringProctoringService {
} catch (final Exception e) { } catch (final Exception e) {
log.error("Failed to close proctoring town-hall room for exam: {}", examId); log.error("Failed to close proctoring town-hall room for exam: {}", examId);
return false;
} }
return action;
return true;
} }
private void updateTownhallButton( private void updateTownhallButton(
@ -384,16 +421,18 @@ public class MonitoringProctoringService {
this.pageService.firePageEvent( this.pageService.firePageEvent(
new ActionActivationEvent( new ActionActivationEvent(
false, false,
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM, new Tuple<>(
ActionDefinition.MONITOR_EXAM_CLOSE_TOWNHALL_PROCTOR_ROOM), ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM,
ActionDefinition.MONITOR_EXAM_CLOSE_TOWNHALL_PROCTOR_ROOM)),
pageContext); pageContext);
} }
} else { } else {
this.pageService.firePageEvent( this.pageService.firePageEvent(
new ActionActivationEvent( new ActionActivationEvent(
proctoringGUIService.getNumberOfProctoringParticipants() > 0, proctoringGUIService.getNumberOfProctoringParticipants() > 0,
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM, new Tuple<>(
ActionDefinition.MONITOR_EXAM_CLOSE_TOWNHALL_PROCTOR_ROOM), ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM,
ActionDefinition.MONITOR_EXAM_OPEN_TOWNHALL_PROCTOR_ROOM)),
pageContext); pageContext);
} }
} }

View file

@ -246,7 +246,9 @@ public class ExamAdminServiceImpl implements ExamAdminService {
EntityType.EXAM, EntityType.EXAM,
examId, examId,
ProctoringServiceSettings.ATTR_APP_SECRET, ProctoringServiceSettings.ATTR_APP_SECRET,
this.cryptor.encrypt(proctoringServiceSettings.appSecret).toString()); this.cryptor.encrypt(proctoringServiceSettings.appSecret)
.getOrThrow()
.toString());
return proctoringServiceSettings; return proctoringServiceSettings;
}); });

View file

@ -11,6 +11,7 @@ package ch.ethz.seb.sebserver.webservice.servicelayer.session;
import java.util.Collection; import java.util.Collection;
import java.util.Map; import java.util.Map;
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
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.ProctoringRoomConnection; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringRoomConnection;
import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection; import ch.ethz.seb.sebserver.gbl.model.session.ClientConnection;
@ -75,6 +76,8 @@ public interface ExamProctoringRoomService {
* @return Result refer to the RemoteProctoringRoom data or to an error when happened */ * @return Result refer to the RemoteProctoringRoom data or to an error when happened */
Result<RemoteProctoringRoom> getTownhallRoomData(final Long examId); Result<RemoteProctoringRoom> getTownhallRoomData(final Long examId);
Result<EntityKey> closeTownhallRoom(Long examId);
/** Used to create a break out room for all active SEB clients given by the connectionTokens. /** Used to create a break out room for all active SEB clients given by the connectionTokens.
* This first notifies the underling proctoring specific service layer on room creation that will create a room * This first notifies the underling proctoring specific service layer on room creation that will create a room
* on the meeting service if necessary. Then creating the room internally for holding data and tracking the new * on the meeting service if necessary. Then creating the room internally for holding data and tracking the new

View file

@ -22,6 +22,7 @@ import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import ch.ethz.seb.sebserver.gbl.Constants; import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
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.ProctoringRoomConnection; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringRoomConnection;
@ -323,6 +324,16 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
return this.remoteProctoringRoomDAO.isTownhallRoomActive(examId); return this.remoteProctoringRoomDAO.isTownhallRoomActive(examId);
} }
@Override
public Result<EntityKey> closeTownhallRoom(final Long examId) {
if (isTownhallRoomActive(examId)) {
return this.remoteProctoringRoomDAO.getTownhallRoom(examId)
.flatMap(room -> this.remoteProctoringRoomDAO.deleteRoom(room.id));
}
return Result.ofRuntimeError("No active town-hall for exam: " + examId);
}
private void closeTownhall( private void closeTownhall(
final Long examId, final Long examId,
final ProctoringServiceSettings proctoringSettings, final ProctoringServiceSettings proctoringSettings,

View file

@ -286,6 +286,9 @@ public class ExamProctoringController {
checkAccess(institutionId, examId); checkAccess(institutionId, examId);
return this.examProcotringRoomService return this.examProcotringRoomService
.openTownhallRoom(examId, subject) .openTownhallRoom(examId, subject)
.onError(error -> this.examProcotringRoomService.closeTownhallRoom(examId)
.onError(err -> log.error("Failed to close town-hall after failed opening: {}",
err.getMessage())))
.getOrThrow(); .getOrThrow();
} }

View file

@ -665,6 +665,13 @@ sebserver.exam.proctoring.type.servertype.JITSI_MEET.tooltip=Use a Jitsi Meet se
sebserver.exam.proctoring.type.servertype.ZOOM=Zoom Server sebserver.exam.proctoring.type.servertype.ZOOM=Zoom Server
sebserver.exam.proctoring.type.servertype.ZOOM.tooltip=Use a Zoom meeting server for proctoring sebserver.exam.proctoring.type.servertype.ZOOM.tooltip=Use a Zoom meeting server for proctoring
sebserver.exam.proctoring.townhall.open.error=Failed to open the town-hall room.
sebserver.exam.proctoring.townhall.close.error=Failed to close the town-hall room properly.
sebserver.exam.proctoring.one.open.error=Failed to open the one-to-one room.
sebserver.exam.proctoring.one.close.error=Failed to close the one-to-one room properly.
sebserver.exam.proctoring.collecting.open.error=Failed to open the collecting room.
sebserver.exam.proctoring.collecting.close.error=Failed to close the collecting room properly.
################################ ################################
# Connection Configuration # Connection Configuration