SEBSERV-204 added collecting room restrictions (only one open at a time)

This commit is contained in:
anhefti 2021-07-08 13:08:47 +02:00
parent e88f5146ab
commit 181fb6d95e
10 changed files with 131 additions and 30 deletions

View file

@ -22,8 +22,10 @@ import ch.ethz.seb.sebserver.gbl.util.Utils;
@JsonIgnoreProperties(ignoreUnknown = true) @JsonIgnoreProperties(ignoreUnknown = true)
public class RemoteProctoringRoom { public class RemoteProctoringRoom {
public static final String ATTR_IS_OPEN = "isOpen";
public static final RemoteProctoringRoom NULL_ROOM = new RemoteProctoringRoom( public static final RemoteProctoringRoom NULL_ROOM = new RemoteProctoringRoom(
null, null, null, null, null, false, null, null, null); null, null, null, null, null, false, null, null, null, false);
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ID) @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ID)
public final Long id; public final Long id;
@ -52,6 +54,9 @@ public class RemoteProctoringRoom {
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ROOM_DATA) @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ROOM_DATA)
public final String additionalRoomData; public final String additionalRoomData;
@JsonProperty(ATTR_IS_OPEN)
public final Boolean isOpen;
@JsonCreator @JsonCreator
public RemoteProctoringRoom( public RemoteProctoringRoom(
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ID) final Long id, @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ID) final Long id,
@ -62,7 +67,8 @@ public class RemoteProctoringRoom {
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_TOWNHALL_ROOM) final Boolean townhallRoom, @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_TOWNHALL_ROOM) final Boolean townhallRoom,
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_BREAK_OUT_CONNECTIONS) final Collection<String> breakOutConnections, @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_BREAK_OUT_CONNECTIONS) final Collection<String> breakOutConnections,
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_JOIN_KEY) final CharSequence joinKey, @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_JOIN_KEY) final CharSequence joinKey,
@JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ROOM_DATA) final String additionalRoomData) { @JsonProperty(Domain.REMOTE_PROCTORING_ROOM.ATTR_ROOM_DATA) final String additionalRoomData,
@JsonProperty(ATTR_IS_OPEN) final Boolean isOpen) {
this.id = id; this.id = id;
this.examId = examId; this.examId = examId;
@ -73,6 +79,7 @@ public class RemoteProctoringRoom {
this.breakOutConnections = Utils.immutableCollectionOf(breakOutConnections); this.breakOutConnections = Utils.immutableCollectionOf(breakOutConnections);
this.joinKey = joinKey; this.joinKey = joinKey;
this.additionalRoomData = additionalRoomData; this.additionalRoomData = additionalRoomData;
this.isOpen = isOpen;
} }
public Long getId() { public Long getId() {
@ -111,6 +118,10 @@ public class RemoteProctoringRoom {
return this.additionalRoomData; return this.additionalRoomData;
} }
public Boolean getIsOpen() {
return this.isOpen;
}
@Override @Override
public String toString() { public String toString() {
final StringBuilder builder = new StringBuilder(); final StringBuilder builder = new StringBuilder();
@ -128,6 +139,12 @@ public class RemoteProctoringRoom {
builder.append(this.townhallRoom); builder.append(this.townhallRoom);
builder.append(", breakOutConnections="); builder.append(", breakOutConnections=");
builder.append(this.breakOutConnections); builder.append(this.breakOutConnections);
builder.append(", joinKey=");
builder.append(this.joinKey);
builder.append(", additionalRoomData=");
builder.append(this.additionalRoomData);
builder.append(", isOpen=");
builder.append(this.isOpen);
builder.append("]"); builder.append("]");
return builder.toString(); return builder.toString();
} }

View file

@ -354,7 +354,7 @@ public final class ClientConnectionTable {
data.getConnectionId(), data.getConnectionId(),
UpdatableTableItem::new); UpdatableTableItem::new);
tableItem.push(data); tableItem.push(data);
if (this.statusFilterChanged || this.forceUpdateAll) { if (this.statusFilterChanged || this.forceUpdateAll || needsSync) {
this.toDelete.remove(data.getConnectionId()); this.toDelete.remove(data.getConnectionId());
} }
}); });

View file

@ -201,14 +201,11 @@ public class MonitoringProctoringService {
final PageAction action = final PageAction action =
actionBuilder.newAction(ActionDefinition.MONITOR_EXAM_VIEW_PROCTOR_ROOM) actionBuilder.newAction(ActionDefinition.MONITOR_EXAM_VIEW_PROCTOR_ROOM)
.withEntityKey(entityKey) .withEntityKey(entityKey)
.withExec(_action -> { .withExec(_action -> openExamProctoringRoom(
final int actualRoomSize = proctoringGUIService proctoringGUIService,
.getActualCollectingRoomSize(room.name); proctoringSettings,
if (actualRoomSize <= 0) { room,
return _action; _action))
}
return openExamProctoringRoom(proctoringSettings, room, _action);
})
.withNameAttributes( .withNameAttributes(
room.subject, room.subject,
room.roomSize, room.roomSize,
@ -229,6 +226,7 @@ public class MonitoringProctoringService {
.withParentEntityKey(entityKey); .withParentEntityKey(entityKey);
this.proctorRoomConnectionsPopup.show(pc, collectingRoom.subject); this.proctorRoomConnectionsPopup.show(pc, collectingRoom.subject);
})); }));
processProctorRoomActionActivation( processProctorRoomActionActivation(
proctoringGUIService.getCollectingRoomActionItem(room.name), proctoringGUIService.getCollectingRoomActionItem(room.name),
room, pageContext); room, pageContext);
@ -239,10 +237,15 @@ public class MonitoringProctoringService {
} }
private PageAction openExamProctoringRoom( private PageAction openExamProctoringRoom(
final ProctoringGUIService proctoringGUIService,
final ProctoringServiceSettings proctoringSettings, final ProctoringServiceSettings proctoringSettings,
final RemoteProctoringRoom room, final RemoteProctoringRoom room,
final PageAction action) { final PageAction action) {
if (!proctoringGUIService.isCollectingRoomEnabled(room.name)) {
return action;
}
final ProctoringRoomConnection proctoringConnectionData = this.pageService final ProctoringRoomConnection proctoringConnectionData = this.pageService
.getRestService() .getRestService()
.getBuilder(GetProctorRoomConnection.class) .getBuilder(GetProctorRoomConnection.class)
@ -437,13 +440,16 @@ public class MonitoringProctoringService {
final PageContext pageContext) { final PageContext pageContext) {
try { try {
final boolean active = room.roomSize > 0 && !room.isOpen;
final Display display = pageContext.getRoot().getDisplay(); final Display display = pageContext.getRoot().getDisplay();
final PageAction action = (PageAction) treeItem.getData(ActionPane.ACTION_EVENT_CALL_KEY); final PageAction action = (PageAction) treeItem.getData(ActionPane.ACTION_EVENT_CALL_KEY);
final Image image = room.roomSize > 0 final Image image = active
? action.definition.icon.getImage(display) ? action.definition.icon.getImage(display)
: action.definition.icon.getGreyedImage(display); : action.definition.icon.getGreyedImage(display);
treeItem.setImage(image); treeItem.setImage(image);
treeItem.setForeground(room.roomSize > 0 ? null : new Color(display, Constants.GREY_DISABLED)); treeItem.setForeground(active ? null : new Color(display, Constants.GREY_DISABLED));
} catch (final Exception e) { } catch (final Exception e) {
log.warn("Failed to set Proctor-Room-Activation: ", e.getMessage()); log.warn("Failed to set Proctor-Room-Activation: ", e.getMessage());
} }

View file

@ -78,7 +78,6 @@ public class ProctoringGUIService {
final RemoteProctoringRoom remoteProctoringRoom = getRemoteProctoringRoom(item); final RemoteProctoringRoom remoteProctoringRoom = getRemoteProctoringRoom(item);
if (remoteProctoringRoom != null && remoteProctoringRoom.roomSize > 0) { if (remoteProctoringRoom != null && remoteProctoringRoom.roomSize > 0) {
showConnectionsPopup.accept(remoteProctoringRoom); showConnectionsPopup.accept(remoteProctoringRoom);
//this.proctorRoomConnectionsPopup.show(pc, remoteProctoringRoom.subject);
} }
} }
}); });
@ -99,12 +98,13 @@ public class ProctoringGUIService {
.orElse(null); .orElse(null);
} }
public int getActualCollectingRoomSize(final String roomName) { public boolean isCollectingRoomEnabled(final String roomName) {
try { try {
return this.collectingRoomsActionState.get(roomName).a.roomSize; final Pair<RemoteProctoringRoom, TreeItem> pair = this.collectingRoomsActionState.get(roomName);
return pair.a.roomSize > 0 && !pair.a.isOpen;
} catch (final Exception e) { } catch (final Exception e) {
log.error("Failed to get actual collecting room size for room: {} cause: ", roomName, e.getMessage()); log.error("Failed to get actual collecting room size for room: {} cause: ", roomName, e.getMessage());
return -1; return false;
} }
} }

View file

@ -69,7 +69,8 @@ public interface AdditionalAttributesDAO {
/** Use this to delete all additional attributes for a given entity. /** Use this to delete all additional attributes for a given entity.
* *
* @param type the entity type
* @param entityId the entity identifier (primary-key) */ * @param entityId the entity identifier (primary-key) */
void deleteAll(Long entityId); void deleteAll(EntityType type, Long entityId);
} }

View file

@ -132,4 +132,6 @@ public interface RemoteProctoringRoomDAO {
* @return Result refer to active break-out rooms or to an error when happened */ * @return Result refer to active break-out rooms or to an error when happened */
Result<Collection<String>> getConnectionsInBreakoutRooms(Long examId); Result<Collection<String>> getConnectionsInBreakoutRooms(Long examId);
void setCollectingRoomOpenFlag(Long roomId, boolean isOpen);
} }

View file

@ -12,6 +12,8 @@ import java.util.Collection;
import java.util.Optional; import java.util.Optional;
import org.mybatis.dynamic.sql.SqlBuilder; import org.mybatis.dynamic.sql.SqlBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Lazy; import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component; import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
@ -29,6 +31,8 @@ import ch.ethz.seb.sebserver.webservice.servicelayer.dao.AdditionalAttributesDAO
@WebServiceProfile @WebServiceProfile
public class AdditionalAttributesDAOImpl implements AdditionalAttributesDAO { public class AdditionalAttributesDAOImpl implements AdditionalAttributesDAO {
private static final Logger log = LoggerFactory.getLogger(AdditionalAttributesDAOImpl.class);
private final AdditionalAttributeRecordMapper additionalAttributeRecordMapper; private final AdditionalAttributeRecordMapper additionalAttributeRecordMapper;
protected AdditionalAttributesDAOImpl(final AdditionalAttributeRecordMapper additionalAttributeRecordMapper) { protected AdditionalAttributesDAOImpl(final AdditionalAttributeRecordMapper additionalAttributeRecordMapper) {
@ -164,14 +168,21 @@ public class AdditionalAttributesDAOImpl implements AdditionalAttributesDAO {
@Override @Override
@Transactional @Transactional
public void deleteAll(final Long entityId) { public void deleteAll(final EntityType type, final Long entityId) {
try {
this.additionalAttributeRecordMapper this.additionalAttributeRecordMapper
.deleteByExample() .deleteByExample()
.where( .where(
AdditionalAttributeRecordDynamicSqlSupport.entityType,
SqlBuilder.isEqualTo(type.name()))
.and(
AdditionalAttributeRecordDynamicSqlSupport.entityId, AdditionalAttributeRecordDynamicSqlSupport.entityId,
SqlBuilder.isEqualTo(entityId)) SqlBuilder.isEqualTo(entityId))
.build() .build()
.execute(); .execute();
} catch (final Exception e) {
log.warn("Failed to delete all additional attributes for: {} cause: {}", entityId, e.getMessage());
}
} }
} }

View file

@ -635,7 +635,8 @@ public class ExamDAOImpl implements ExamDAO {
.execute(); .execute();
// delete all additional attributes // delete all additional attributes
this.additionalAttributeRecordMapper.deleteByExample() this.additionalAttributeRecordMapper
.deleteByExample()
.where(AdditionalAttributeRecordDynamicSqlSupport.entityType, isEqualTo(EntityType.EXAM.name())) .where(AdditionalAttributeRecordDynamicSqlSupport.entityType, isEqualTo(EntityType.EXAM.name()))
.and(AdditionalAttributeRecordDynamicSqlSupport.entityId, isIn(ids)) .and(AdditionalAttributeRecordDynamicSqlSupport.entityId, isIn(ids))
.build() .build()

View file

@ -35,6 +35,7 @@ import ch.ethz.seb.sebserver.gbl.util.Result;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.RemoteProctoringRoomRecordDynamicSqlSupport; import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.RemoteProctoringRoomRecordDynamicSqlSupport;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.RemoteProctoringRoomRecordMapper; import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.RemoteProctoringRoomRecordMapper;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.RemoteProctoringRoomRecord; import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.RemoteProctoringRoomRecord;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.AdditionalAttributesDAO;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.RemoteProctoringRoomDAO; import ch.ethz.seb.sebserver.webservice.servicelayer.dao.RemoteProctoringRoomDAO;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.TransactionHandler; import ch.ethz.seb.sebserver.webservice.servicelayer.dao.TransactionHandler;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.NewRoom; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.NewRoom;
@ -49,11 +50,14 @@ public class RemoteProctoringRoomDAOImpl implements RemoteProctoringRoomDAO {
private static final Object RESERVE_ROOM_LOCK = new Object(); private static final Object RESERVE_ROOM_LOCK = new Object();
private final RemoteProctoringRoomRecordMapper remoteProctoringRoomRecordMapper; private final RemoteProctoringRoomRecordMapper remoteProctoringRoomRecordMapper;
private final AdditionalAttributesDAO additionalAttributesDAO;
protected RemoteProctoringRoomDAOImpl( protected RemoteProctoringRoomDAOImpl(
final RemoteProctoringRoomRecordMapper remoteProctoringRoomRecordMapper) { final RemoteProctoringRoomRecordMapper remoteProctoringRoomRecordMapper,
final AdditionalAttributesDAO additionalAttributesDAO) {
this.remoteProctoringRoomRecordMapper = remoteProctoringRoomRecordMapper; this.remoteProctoringRoomRecordMapper = remoteProctoringRoomRecordMapper;
this.additionalAttributesDAO = additionalAttributesDAO;
} }
@Override @Override
@ -227,6 +231,10 @@ public class RemoteProctoringRoomDAOImpl implements RemoteProctoringRoomDAO {
@Transactional @Transactional
public Result<EntityKey> deleteRoom(final Long roomId) { public Result<EntityKey> deleteRoom(final Long roomId) {
return Result.tryCatch(() -> { return Result.tryCatch(() -> {
this.additionalAttributesDAO
.deleteAll(EntityType.REMOTE_PROCTORING_ROOM, roomId);
this.remoteProctoringRoomRecordMapper this.remoteProctoringRoomRecordMapper
.deleteByPrimaryKey(roomId); .deleteByPrimaryKey(roomId);
@ -249,7 +257,15 @@ public class RemoteProctoringRoomDAOImpl implements RemoteProctoringRoomDAO {
.execute(); .execute();
return ids.stream() return ids.stream()
.map(id -> new EntityKey(String.valueOf(id), EntityType.REMOTE_PROCTORING_ROOM)) .map(roomId -> {
this.additionalAttributesDAO.deleteAll(
EntityType.REMOTE_PROCTORING_ROOM,
roomId);
return roomId;
})
.map(roomId -> new EntityKey(
String.valueOf(roomId),
EntityType.REMOTE_PROCTORING_ROOM))
.collect(Collectors.toList()); .collect(Collectors.toList());
}); });
@ -340,6 +356,19 @@ public class RemoteProctoringRoomDAOImpl implements RemoteProctoringRoomDAO {
.collect(Collectors.toList())); .collect(Collectors.toList()));
} }
@Override
@Transactional
public void setCollectingRoomOpenFlag(final Long roomId, final boolean isOpen) {
this.additionalAttributesDAO.saveAdditionalAttribute(
EntityType.REMOTE_PROCTORING_ROOM,
roomId,
RemoteProctoringRoom.ATTR_IS_OPEN,
BooleanUtils.toStringTrueFalse(isOpen))
.onError(error -> log.error("Failed to set open flag for proctoring room: {} : {}",
roomId,
error.getMessage()));
}
private RemoteProctoringRoom toDomainModel(final RemoteProctoringRoomRecord record) { private RemoteProctoringRoom toDomainModel(final RemoteProctoringRoomRecord record) {
final String breakOutConnections = record.getBreakOutConnections(); final String breakOutConnections = record.getBreakOutConnections();
final Collection<String> connections = StringUtils.isNotBlank(breakOutConnections) final Collection<String> connections = StringUtils.isNotBlank(breakOutConnections)
@ -355,7 +384,25 @@ public class RemoteProctoringRoomDAOImpl implements RemoteProctoringRoomDAO {
BooleanUtils.toBooleanObject(record.getTownhallRoom()), BooleanUtils.toBooleanObject(record.getTownhallRoom()),
connections, connections,
record.getJoinKey(), record.getJoinKey(),
record.getRoomData()); record.getRoomData(),
isOpen(record));
}
private boolean isOpen(final RemoteProctoringRoomRecord record) {
if (record.getTownhallRoom() != 0 || !StringUtils.isBlank(record.getBreakOutConnections())) {
return false;
} else {
return BooleanUtils.toBoolean(this.additionalAttributesDAO
.getAdditionalAttribute(
EntityType.REMOTE_PROCTORING_ROOM,
record.getId(),
RemoteProctoringRoom.ATTR_IS_OPEN)
.map(rec -> rec.getValue())
.onError(error -> log.error("Failed to get open flag for proctoring room: {} : {}",
record.getName(),
error.getMessage()))
.getOrElse(() -> Constants.FALSE_STRING));
}
} }
private RemoteProctoringRoomRecord createNewCollectingRoom( private RemoteProctoringRoomRecord createNewCollectingRoom(

View file

@ -394,6 +394,9 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
examProctoringService examProctoringService
.notifyCollectingRoomOpened(proctoringSettings, room, clientConnections) .notifyCollectingRoomOpened(proctoringSettings, room, clientConnections)
.getOrThrow(); .getOrThrow();
this.remoteProctoringRoomDAO
.setCollectingRoomOpenFlag(room.id, true);
} }
}); });
} }
@ -446,6 +449,19 @@ public class ExamProctoringRoomServiceImpl implements ExamProctoringRoomService
.map(cc -> cc.connectionToken) .map(cc -> cc.connectionToken)
.collect(Collectors.toList()); .collect(Collectors.toList());
final RemoteProctoringRoom room = this.remoteProctoringRoomDAO
.getRoom(examId, roomName)
.onError(error -> log.error("Failed to get room for setting closed: {} {} {}",
examId,
roomName,
error.getMessage()))
.getOr(null);
if (room != null) {
this.remoteProctoringRoomDAO
.setCollectingRoomOpenFlag(room.id, false);
}
// Send default settings to clients if feature is enabled // Send default settings to clients if feature is enabled
if (this.sendBroadcastReset) { if (this.sendBroadcastReset) {
this.sendReconfigurationInstructions( this.sendReconfigurationInstructions(