From 2f2a3670b792bd366f8fd9ad69985d0dd5a75ac6 Mon Sep 17 00:00:00 2001 From: anhefti Date: Wed, 3 Nov 2021 10:31:15 +0100 Subject: [PATCH 1/3] SEBSERV-191 backend implementation --- .../gbl/model/session/ClientEvent.java | 4 + .../servicelayer/PaginationService.java | 24 ++ .../servicelayer/PaginationServiceImpl.java | 14 + .../exam/SEBClientEventAdminService.java | 38 +++ .../exam/SEBClientEventExporter.java | 32 +++ .../impl/SEBClientEventAdminServiceImpl.java | 250 ++++++++++++++++++ .../exam/impl/SEBClientEventCSVExporter.java | 49 ++++ ...EBClientEventExportTransactionHandler.java | 113 ++++++++ .../weblayer/api/ClientEventController.java | 37 +-- 9 files changed, 532 insertions(+), 29 deletions(-) create mode 100644 src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java create mode 100644 src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventExporter.java create mode 100644 src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java create mode 100644 src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java create mode 100644 src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventExportTransactionHandler.java diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/model/session/ClientEvent.java b/src/main/java/ch/ethz/seb/sebserver/gbl/model/session/ClientEvent.java index 535c07c2..886bd5ed 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/model/session/ClientEvent.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/model/session/ClientEvent.java @@ -68,6 +68,10 @@ public class ClientEvent implements Entity, IndicatorValueHolder { } } + public enum ExportType { + CSV + } + @JsonProperty(Domain.CLIENT_EVENT.ATTR_ID) public final Long id; diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java index e01725aa..0d05a0fe 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java @@ -77,6 +77,30 @@ public interface PaginationService { final String tableName, final Supplier>> 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 */ + Result> fetch( + final int pageNumber, + final int pageSize, + final String sort, + final String tableName, + final Supplier>> delegate); + /** Use this to build a current Page from a given list of objects. * * @param the Type if list entities diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java index 8da775b7..9bb99708 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java @@ -147,6 +147,20 @@ public class PaginationServiceImpl implements PaginationService { }); } + @Override + public Result> fetch( + final int pageNumber, + final int pageSize, + final String sort, + final String tableName, + final Supplier>> delegate) { + + return Result.tryCatch(() -> { + setPagination(pageNumber, pageSize, sort, tableName); + return delegate.get().getOrThrow(); + }); + } + private String verifySortColumnName(final String sort, final String columnName) { if (StringUtils.isBlank(sort)) { diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java new file mode 100644 index 00000000..8acb9081 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java @@ -0,0 +1,38 @@ +/* + * 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 java.util.function.Predicate; + +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; +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.dao.FilterMap; + +public interface SEBClientEventAdminService { + + Result deleteAllClientEvents(Collection ids); + + @Async(AsyncServiceSpringConfig.EXECUTOR_BEAN_NAME) + void exportSEBClientLogs( + OutputStream output, + FilterMap filterMap, + String sort, + final Predicate predicate, + ExportType exportType, + boolean includeConnectionDetails, + boolean includeExamDetails); + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventExporter.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventExporter.java new file mode 100644 index 00000000..15f7fe23 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventExporter.java @@ -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); +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java new file mode 100644 index 00000000..be6b95bd --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java @@ -0,0 +1,250 @@ +/* + * 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.OutputStream; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.EnumMap; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import java.util.Set; +import java.util.function.Predicate; +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.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.exam.Exam; +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.profile.WebServiceProfile; +import ch.ethz.seb.sebserver.gbl.util.Result; +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.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 exporter; + + public SEBClientEventAdminServiceImpl( + final PaginationService paginationService, + final ClientEventDAO clientEventDAO, + final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler, + final Collection 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 deleteAllClientEvents(final Collection ids) { + return Result.tryCatch(() -> { + + if (ids == null || ids.isEmpty()) { + return EntityProcessingReport.ofEmptyError(); + } + + final Set sources = ids.stream() + .map(id -> new EntityKey(id, EntityType.CLIENT_EVENT)) + .collect(Collectors.toSet()); + + final Result> 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())))); + } else { + return new EntityProcessingReport( + sources, + delete.get(), + Collections.emptyList()); + } + }); + } + + @Override + public void exportSEBClientLogs( + final OutputStream output, + final FilterMap filterMap, + final String sort, + final Predicate predicate, + final ExportType exportType, + final boolean includeConnectionDetails, + final boolean includeExamDetails) { + + new exportRunner( + this.exporter.get(exportType), + includeConnectionDetails, + includeExamDetails, + new Pager(filterMap, sort, predicate), + output) + .run(); + + } + + private class exportRunner { + + private final SEBClientEventExporter exporter; + private final boolean includeConnectionDetails; + private final boolean includeExamDetails; + private final Iterator> pager; + private final OutputStream output; + + private final Map examCache; + private final Map connectionCache; + + public exportRunner( + final SEBClientEventExporter exporter, + final boolean includeConnectionDetails, + final boolean includeExamDetails, + final Iterator> pager, + final OutputStream output) { + + this.exporter = exporter; + this.includeConnectionDetails = includeConnectionDetails; + this.includeExamDetails = includeExamDetails; + this.pager = pager; + this.output = output; + + this.connectionCache = includeConnectionDetails ? new HashMap<>() : null; + this.examCache = includeExamDetails ? new HashMap<>() : null; + } + + public void run() { + + // 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 -> { + 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) + .map(e -> this.connectionCache.put(connectionId, e)) + .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) + .map(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> { + + private final FilterMap filterMap; + private final String sort; + private final Predicate predicate; + + private int pageNumber = 0; + private final int pageSize = 100; + + private Collection nextRecords; + + public Pager( + final FilterMap filterMap, + final String sort, + final Predicate predicate) { + + this.filterMap = filterMap; + this.sort = sort; + this.predicate = predicate; + + fetchNext(); + } + + @Override + public boolean hasNext() { + return this.nextRecords != null && !this.nextRecords.isEmpty(); + } + + @Override + public Collection next() { + final Collection result = this.nextRecords; + fetchNext(); + return result; + } + + private void fetchNext() { + try { + + this.nextRecords = SEBClientEventAdminServiceImpl.this.paginationService.fetch( + this.pageNumber, + this.pageSize, + this.sort, + ClientEventRecordDynamicSqlSupport.clientEventRecord.name(), + () -> SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler + .allMatching(this.filterMap, this.predicate)) + .getOrThrow(); + + this.pageNumber++; + + } catch (final Exception e) { + log.error("Failed to fetch next batch: ", e); + this.nextRecords = null; + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java new file mode 100644 index 00000000..a3290593 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java @@ -0,0 +1,49 @@ +/* + * 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.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; +import ch.ethz.seb.sebserver.webservice.servicelayer.exam.SEBClientEventExporter; + +public class SEBClientEventCSVExporter implements SEBClientEventExporter { + + @Override + public ExportType exportType() { + return ExportType.CSV; + } + + @Override + public void streamHeader( + final OutputStream output, + final boolean includeConnectionDetails, + final boolean includeExamDetails) { + + // TODO + } + + @Override + public void streamData( + final OutputStream output, + final ClientEventRecord eventData, + final ClientConnectionRecord connectionData, + final Exam examData) { + + // TODO + } + + private String toCSVString(final String text) { + + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventExportTransactionHandler.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventExportTransactionHandler.java new file mode 100644 index 00000000..6e24f501 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventExportTransactionHandler.java @@ -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> allMatching( + final FilterMap filterMap, + final Predicate 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 clientConnectionById(final Long id) { + return Result.tryCatch(() -> this.clientConnectionRecordMapper.selectByPrimaryKey(id)); + } + + public Result examById(final Long id) { + return this.examDAO.byPK(id); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java index 349c42eb..947ca057 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java @@ -8,13 +8,9 @@ package ch.ethz.seb.sebserver.webservice.weblayer.api; -import java.util.Arrays; import java.util.Collection; -import java.util.Collections; import java.util.EnumSet; import java.util.List; -import java.util.Set; -import java.util.stream.Collectors; import javax.servlet.http.HttpServletRequest; @@ -28,13 +24,10 @@ import org.springframework.web.bind.annotation.RestController; import ch.ethz.seb.sebserver.gbl.api.API; import ch.ethz.seb.sebserver.gbl.api.API.BulkActionType; -import ch.ethz.seb.sebserver.gbl.api.APIMessage; import ch.ethz.seb.sebserver.gbl.api.EntityType; import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; 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.ErrorEntry; import ch.ethz.seb.sebserver.gbl.model.GrantEntity; import ch.ethz.seb.sebserver.gbl.model.Page; import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent; @@ -53,6 +46,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.FilterMap; 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; @WebServiceProfile @@ -62,6 +56,7 @@ public class ClientEventController extends ReadonlyEntityController sources = ids.stream() - .map(id -> new EntityKey(id, EntityType.CLIENT_EVENT)) - .collect(Collectors.toSet()); - - final Result> 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())))); - } else { - return new EntityProcessingReport( - sources, - delete.get(), - Collections.emptyList()); - } + return this.sebClientEventAdminService + .deleteAllClientEvents(ids) + .getOrThrow(); } @Override From c414586fece6c5c390a5d7b0b71ac0bddd68d208 Mon Sep 17 00:00:00 2001 From: anhefti Date: Wed, 3 Nov 2021 13:18:40 +0100 Subject: [PATCH 2/3] SEBSERV-191 finished back-end implementation --- .../ch/ethz/seb/sebserver/gbl/api/API.java | 2 + .../ch/ethz/seb/sebserver/gbl/util/Utils.java | 7 ++ .../exam/SEBClientEventAdminService.java | 3 - .../impl/SEBClientEventAdminServiceImpl.java | 43 ++++++++----- .../exam/impl/SEBClientEventCSVExporter.java | 60 +++++++++++++++-- .../weblayer/api/ClientEventController.java | 64 +++++++++++++++++-- 6 files changed, 152 insertions(+), 27 deletions(-) diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java b/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java index e523be9a..b89fb671 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java @@ -197,6 +197,8 @@ public final class API { 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_EXPORT_PATH_SEGMENT = "/export"; + public static final String SEB_CLIENT_EVENT_EXPORT_TYPE = "exportType"; public static final String SEB_CLIENT_EVENT_EXTENDED_PAGE_ENDPOINT = SEB_CLIENT_EVENT_ENDPOINT + SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT; diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/util/Utils.java b/src/main/java/ch/ethz/seb/sebserver/gbl/util/Utils.java index 8d79797b..84c47e63 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/util/Utils.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/util/Utils.java @@ -666,4 +666,11 @@ public final class Utils { } } + public static String toCSVString(final String text) { + if (StringUtils.isBlank(text)) { + return StringUtils.EMPTY; + } + return Constants.DOUBLE_QUOTE + text.replace("\"", "\"\"") + Constants.DOUBLE_QUOTE; + } + } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java index 8acb9081..9acacb4a 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java @@ -10,13 +10,11 @@ package ch.ethz.seb.sebserver.webservice.servicelayer.exam; import java.io.OutputStream; import java.util.Collection; -import java.util.function.Predicate; 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; 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.dao.FilterMap; @@ -30,7 +28,6 @@ public interface SEBClientEventAdminService { OutputStream output, FilterMap filterMap, String sort, - final Predicate predicate, ExportType exportType, boolean includeConnectionDetails, boolean includeExamDetails); diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java index be6b95bd..759b5445 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java @@ -13,11 +13,11 @@ 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.function.Predicate; import java.util.stream.Collectors; import org.slf4j.Logger; @@ -31,14 +31,17 @@ 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.exam.Exam; -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.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.AuthorizationService; +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; @@ -55,16 +58,19 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic private final ClientEventDAO clientEventDAO; private final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler; private final EnumMap exporter; + private final AuthorizationService authorizationService; public SEBClientEventAdminServiceImpl( final PaginationService paginationService, final ClientEventDAO clientEventDAO, final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler, - final Collection exporter) { + final Collection exporter, + final AuthorizationService authorizationService) { this.paginationService = paginationService; this.clientEventDAO = clientEventDAO; this.sebClientEventExportTransactionHandler = sebClientEventExportTransactionHandler; + this.authorizationService = authorizationService; this.exporter = new EnumMap<>(ExportType.class); exporter.forEach(exp -> this.exporter.putIfAbsent(exp.exportType(), exp)); @@ -102,7 +108,6 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic final OutputStream output, final FilterMap filterMap, final String sort, - final Predicate predicate, final ExportType exportType, final boolean includeConnectionDetails, final boolean includeExamDetails) { @@ -111,7 +116,7 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic this.exporter.get(exportType), includeConnectionDetails, includeExamDetails, - new Pager(filterMap, sort, predicate), + new Pager(filterMap, sort), output) .run(); @@ -147,17 +152,28 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic public void run() { + final SEBServerUser currentUser = SEBClientEventAdminServiceImpl.this.authorizationService + .getUserService() + .getCurrentUser(); + final EnumSet 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 -> { - this.exporter.streamData( - this.output, - rec, - this.includeConnectionDetails ? getConnection(rec.getClientConnectionId()) : null, - this.includeExamDetails ? getExam(rec.getClientConnectionId()) : null); + + 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); + } }); } } @@ -195,7 +211,6 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic private final FilterMap filterMap; private final String sort; - private final Predicate predicate; private int pageNumber = 0; private final int pageSize = 100; @@ -204,12 +219,10 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic public Pager( final FilterMap filterMap, - final String sort, - final Predicate predicate) { + final String sort) { this.filterMap = filterMap; this.sort = sort; - this.predicate = predicate; fetchNext(); } @@ -235,7 +248,7 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic this.sort, ClientEventRecordDynamicSqlSupport.clientEventRecord.name(), () -> SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler - .allMatching(this.filterMap, this.predicate)) + .allMatching(this.filterMap, Utils.truePredicate())) .getOrThrow(); this.pageNumber++; diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java index a3290593..151b82c0 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java @@ -8,16 +8,25 @@ 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 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.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; public class SEBClientEventCSVExporter implements SEBClientEventExporter { + private static final Logger log = LoggerFactory.getLogger(SEBClientEventCSVExporter.class); + @Override public ExportType exportType() { return ExportType.CSV; @@ -29,7 +38,24 @@ public class SEBClientEventCSVExporter implements SEBClientEventExporter { final boolean includeConnectionDetails, final boolean includeExamDetails) { - // TODO + 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); + } } @Override @@ -39,11 +65,37 @@ public class SEBClientEventCSVExporter implements SEBClientEventExporter { final ClientConnectionRecord connectionData, final Exam examData) { - // TODO - } + final StringBuilder builder = new StringBuilder(); + final EventType type = EventType.byId(eventData.getType()); - private String toCSVString(final String text) { + builder.append(type.name()); + builder.append(Utils.toCSVString(eventData.getText())); + builder.append(eventData.getNumericValue()); + builder.append(Utils.toDateTimeUTC(eventData.getClientTime())); + builder.append(Utils.toDateTimeUTC(eventData.getServerTime())); + if (connectionData != null) { + builder.append(Utils.toCSVString(connectionData.getExamUserSessionId())); + builder.append(Utils.toCSVString(connectionData.getClientAddress())); + builder.append(connectionData.getStatus()); + builder.append(connectionData.getConnectionToken()); + } + + if (examData != null) { + builder.append(Utils.toCSVString(examData.getName())); + builder.append(Utils.toCSVString(examData.getDescription())); + builder.append(examData.getType().name()); + builder.append(examData.getStartTime()); + 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); + } } } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java index 947ca057..c7865d04 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java @@ -8,13 +8,19 @@ package ch.ethz.seb.sebserver.webservice.weblayer.api; +import java.io.IOException; import java.util.Collection; import java.util.EnumSet; import java.util.List; +import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; import org.mybatis.dynamic.sql.SqlTable; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.http.HttpStatus; import org.springframework.http.MediaType; import org.springframework.util.MultiValueMap; import org.springframework.web.bind.annotation.RequestMapping; @@ -31,6 +37,7 @@ import ch.ethz.seb.sebserver.gbl.model.EntityProcessingReport; import ch.ethz.seb.sebserver.gbl.model.GrantEntity; import ch.ethz.seb.sebserver.gbl.model.Page; import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent; +import ch.ethz.seb.sebserver.gbl.model.session.ClientEvent.ExportType; import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent; import ch.ethz.seb.sebserver.gbl.model.user.UserRole; import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile; @@ -54,6 +61,8 @@ import ch.ethz.seb.sebserver.webservice.servicelayer.validation.BeanValidationSe @RequestMapping("${sebserver.webservice.api.admin.endpoint}" + API.SEB_CLIENT_EVENT_ENDPOINT) public class ClientEventController extends ReadonlyEntityController { + private static final Logger log = LoggerFactory.getLogger(ClientEventController.class); + private final ExamDAO examDao; private final ClientEventDAO clientEventDAO; private final SEBClientEventAdminService sebClientEventAdminService; @@ -138,6 +147,49 @@ public class ClientEventController extends ReadonlyEntityController allRequestParams, + final HttpServletRequest request, + final HttpServletResponse response) throws IOException { + + // at least current user must have base read access for specified entity type within its own institution + checkReadPrivilege(institutionId); + + final FilterMap filterMap = new FilterMap(allRequestParams, request.getQueryString()); + populateFilterMap(filterMap, institutionId, sort); + + final ServletOutputStream outputStream = response.getOutputStream(); + + try { + + this.sebClientEventAdminService.exportSEBClientLogs( + outputStream, + filterMap, + sort, + type, + false, + false); + + response.setStatus(HttpStatus.OK.value()); + } catch (final Exception e) { + log.error("Unexpected error while trying to export SEB client logs: ", e); + response.setStatus(HttpStatus.INTERNAL_SERVER_ERROR.value()); + } finally { + outputStream.flush(); + outputStream.close(); + } + } + @Override public Collection getDependencies( final String modelId, @@ -154,12 +206,14 @@ public class ClientEventController extends ReadonlyEntityController checkReadAccess(final ClientEvent entity) { + final EnumSet userRoles = this.authorization + .getUserService() + .getCurrentUser() + .getUserRoles(); + final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER); + return Result.tryCatch(() -> { - final EnumSet userRoles = this.authorization - .getUserService() - .getCurrentUser() - .getUserRoles(); - final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER); + if (isSupporterOnly) { // check owner grant be getting exam return super.checkReadAccess(entity) From 06ce72a76f35dd30df790aebaaf130c0e5af38d8 Mon Sep 17 00:00:00 2001 From: anhefti Date: Thu, 4 Nov 2021 16:52:48 +0100 Subject: [PATCH 3/3] SEBSERV-191 gui implementation and download streaming --- .../ch/ethz/seb/sebserver/gbl/Constants.java | 2 + .../ch/ethz/seb/sebserver/gbl/api/API.java | 2 + .../ethz/seb/sebserver/gbl/util/Result.java | 7 ++ .../gui/content/SEBClientEvents.java | 54 +++++++++++++ .../gui/content/action/ActionDefinition.java | 4 + .../AbstractDownloadServiceHandler.java | 19 ++--- .../remote/download/DownloadService.java | 48 ++++++++--- .../download/SEBClientConfigDownload.java | 12 ++- .../remote/download/SEBClientLogExport.java | 80 +++++++++++++++++++ .../download/SEBExamConfigDownload.java | 8 +- .../SEBExamConfigPlaintextDownload.java | 8 +- .../api/exam/ExportSEBClientLogs.java | 42 ++++++++++ .../seb/sebserver/gui/table/EntityTable.java | 8 ++ .../servicelayer/PaginationService.java | 6 +- .../servicelayer/PaginationServiceImpl.java | 19 +++-- .../exam/SEBClientEventAdminService.java | 4 +- .../impl/SEBClientEventAdminServiceImpl.java | 78 +++++++++++------- .../exam/impl/SEBClientEventCSVExporter.java | 33 +++++++- .../weblayer/api/ClientEventController.java | 35 ++++++-- src/main/resources/messages.properties | 1 + 20 files changed, 396 insertions(+), 74 deletions(-) create mode 100644 src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientLogExport.java create mode 100644 src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/exam/ExportSEBClientLogs.java diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/Constants.java b/src/main/java/ch/ethz/seb/sebserver/gbl/Constants.java index 2ab1481f..8f026546 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/Constants.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/Constants.java @@ -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 */ 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_TIME_ZONE_CODE = "UTC"; public static final String TOOLTIP_TEXT_KEY_SUFFIX = ".tooltip"; diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java b/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java index b89fb671..6a37a5f5 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/api/API.java @@ -199,6 +199,8 @@ public final class API { 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 + SEB_CLIENT_EVENT_SEARCH_PATH_SEGMENT; diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/util/Result.java b/src/main/java/ch/ethz/seb/sebserver/gbl/util/Result.java index 5eba6fe6..fb67d943 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gbl/util/Result.java +++ b/src/main/java/ch/ethz/seb/sebserver/gbl/util/Result.java @@ -262,6 +262,13 @@ public final class Result { } } + public Result onSuccess(final Consumer 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 * for further processing. * diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/content/SEBClientEvents.java b/src/main/java/ch/ethz/seb/sebserver/gui/content/SEBClientEvents.java index bc303c35..58d65b91 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/content/SEBClientEvents.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/content/SEBClientEvents.java @@ -8,6 +8,7 @@ package ch.ethz.seb.sebserver.gui.content; +import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.function.Consumer; @@ -15,6 +16,8 @@ import java.util.function.Function; import java.util.stream.Collectors; import org.apache.tomcat.util.buf.StringUtils; +import org.eclipse.rap.rwt.RWT; +import org.eclipse.rap.rwt.client.service.UrlLauncher; import org.eclipse.swt.widgets.Composite; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -24,12 +27,15 @@ import org.springframework.stereotype.Component; import org.springframework.util.MultiValueMap; import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; import ch.ethz.seb.sebserver.gbl.api.EntityType; import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType; import ch.ethz.seb.sebserver.gbl.model.Domain; import ch.ethz.seb.sebserver.gbl.model.EntityName; +import ch.ethz.seb.sebserver.gbl.model.Page; 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.ExportType; import ch.ethz.seb.sebserver.gbl.model.session.ExtendedClientEvent; import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; import ch.ethz.seb.sebserver.gbl.util.Utils; @@ -42,6 +48,8 @@ import ch.ethz.seb.sebserver.gui.service.page.PageService; import ch.ethz.seb.sebserver.gui.service.page.PageService.PageActionBuilder; import ch.ethz.seb.sebserver.gui.service.page.TemplateComposer; import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction; +import ch.ethz.seb.sebserver.gui.service.remote.download.DownloadService; +import ch.ethz.seb.sebserver.gui.service.remote.download.SEBClientLogExport; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestService; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.logs.GetClientEventNames; import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.logs.GetExtendedClientEventPage; @@ -84,23 +92,29 @@ public class SEBClientEvents implements TemplateComposer { private final ResourceService resourceService; private final RestService restService; private final I18nSupport i18nSupport; + private final DownloadService downloadService; private final SEBClientEventDetailsPopup sebClientEventDetailsPopup; private final SEBClientEventDeletePopup sebClientEventDeletePopup; private final int pageSize; + private final String exportFileName; public SEBClientEvents( final PageService pageService, + final DownloadService downloadService, final SEBClientEventDetailsPopup sebClientEventDetailsPopup, final SEBClientEventDeletePopup sebClientEventDeletePopup, + @Value("${sebserver.gui.seb.client.logs.export.filename:SEBClientLogs}") final String exportFileName, @Value("${sebserver.gui.list.page.size:20}") final Integer pageSize) { this.pageService = pageService; + this.downloadService = downloadService; this.resourceService = pageService.getResourceService(); this.restService = this.resourceService.getRestService(); this.i18nSupport = this.resourceService.getI18nSupport(); this.sebClientEventDetailsPopup = sebClientEventDetailsPopup; this.sebClientEventDeletePopup = sebClientEventDeletePopup; this.pageSize = pageSize; + this.exportFileName = exportFileName; this.examFilter = new TableFilterAttribute( CriteriaType.SINGLE_SELECTION, @@ -219,12 +233,52 @@ public class SEBClientEvents implements TemplateComposer { .noEventPropagation() .publish(false) + .newAction(ActionDefinition.LOGS_SEB_CLIENT_EXPORT_CSV) + .withExec(action -> this.exportLogs(action, ExportType.CSV, table)) + .noEventPropagation() + .publishIf(() -> writeGrant, table.hasAnyContent()) + .newAction(ActionDefinition.LOGS_SEB_CLIENT_DELETE_ALL) .withExec(action -> this.getOpenDelete(action, table.getFilterCriteria())) .noEventPropagation() .publishIf(() -> writeGrant, table.hasAnyContent()); } + private PageAction exportLogs( + final PageAction action, + final ExportType type, + final EntityTable table) { + + try { + + final UrlLauncher urlLauncher = RWT.getClient().getService(UrlLauncher.class); + final String fileName = this.exportFileName + + Constants.UNDERLINE + + this.i18nSupport.formatDisplayDate(Utils.getMillisecondsNow()) + .replace(" ", "_") + .replace(".", "_") + + Constants.FILE_EXT_CSV; + final Map queryAttrs = new HashMap<>(); + + queryAttrs.put(API.SEB_CLIENT_EVENT_EXPORT_TYPE, type.name()); + final String sortAttr = table.getSortOrder().encode(table.getSortColumn()); + queryAttrs.put(Page.ATTR_SORT, sortAttr); + table.getFilterCriteria().forEach((name, value) -> queryAttrs.put(name, value.get(0))); + + final String downloadURL = this.downloadService + .createDownloadURL( + SEBClientLogExport.class, + fileName, + queryAttrs); + + urlLauncher.openURL(downloadURL); + } catch (final Exception e) { + log.error("Failed open export log download: ", e); + } + + return action; + } + private PageAction getOpenDelete( final PageAction pageAction, final MultiValueMap filterCriteria) { diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/content/action/ActionDefinition.java b/src/main/java/ch/ethz/seb/sebserver/gui/content/action/ActionDefinition.java index 759b47e1..b21f098c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/content/action/ActionDefinition.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/content/action/ActionDefinition.java @@ -770,6 +770,10 @@ public enum ActionDefinition { ImageIcon.DELETE, PageStateDefinitionImpl.SEB_CLIENT_LOGS, ActionCategory.LOGS_SEB_CLIENT_LIST), + LOGS_SEB_CLIENT_EXPORT_CSV( + new LocTextKey("sebserver.seblogs.action.export.csv"), + ImageIcon.EXPORT, + ActionCategory.LOGS_SEB_CLIENT_LIST), ; diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/AbstractDownloadServiceHandler.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/AbstractDownloadServiceHandler.java index 593a4029..d78880a4 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/AbstractDownloadServiceHandler.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/AbstractDownloadServiceHandler.java @@ -40,17 +40,8 @@ public abstract class AbstractDownloadServiceHandler implements DownloadServiceH log.debug("download requested... trying to get needed parameter from request"); 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()) { - log.debug( - "Found modelId: {} for {} download. Trying to request webservice...", - modelId, - downloadFileName); + log.debug("Found modelId: {} for {} download.", modelId); } 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.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE); - webserviceCall(modelId, parentModelId, response.getOutputStream()); + webserviceCall(modelId, parentModelId, response.getOutputStream(), request); } catch (final Exception e) { 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); } diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java index 5ba0c74f..e9b2b64f 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/DownloadService.java @@ -8,9 +8,14 @@ package ch.ethz.seb.sebserver.gui.service.remote.download; -import ch.ethz.seb.sebserver.gbl.Constants; -import ch.ethz.seb.sebserver.gbl.api.API; -import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import java.util.Collection; +import java.util.Map; +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.eclipse.rap.rwt.RWT; import org.eclipse.rap.rwt.service.ServiceHandler; @@ -19,12 +24,10 @@ import org.slf4j.LoggerFactory; import org.springframework.context.annotation.Lazy; import org.springframework.stereotype.Service; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import java.util.Collection; -import java.util.Map; -import java.util.function.Function; -import java.util.stream.Collectors; +import ch.ethz.seb.sebserver.gbl.Constants; +import ch.ethz.seb.sebserver.gbl.api.API; +import ch.ethz.seb.sebserver.gbl.profile.GuiProfile; +import ch.ethz.seb.sebserver.gbl.util.Utils; /** Implements a eclipse RAP ServiceHandler to handle downloads */ @Lazy @@ -73,6 +76,33 @@ public class DownloadService implements ServiceHandler { .processDownload(request, response); } + public String createDownloadURL( + final Class handlerClass, + final String downloadFileName, + final Map 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( final String modelId, final Class handlerClass, diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientConfigDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientConfigDownload.java index 55b49146..5ff68524 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientConfigDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientConfigDownload.java @@ -12,6 +12,8 @@ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; +import javax.servlet.http.HttpServletRequest; + import org.apache.commons.io.IOUtils; import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; @@ -35,14 +37,16 @@ public class SEBClientConfigDownload extends AbstractDownloadServiceHandler { private final RestService restService; - protected SEBClientConfigDownload( - final RestService restService) { - + protected SEBClientConfigDownload(final RestService restService) { this.restService = restService; } @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.RestCallBuilder restCallBuilder = this.restService .getBuilder(ExportClientConfig.class) diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientLogExport.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientLogExport.java new file mode 100644 index 00000000..df46d6d4 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBClientLogExport.java @@ -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 queryParams = new LinkedMultiValueMap<>(); + + final Enumeration 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"); + } + } + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigDownload.java index 638322b7..bca25b02 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigDownload.java @@ -12,6 +12,8 @@ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; +import javax.servlet.http.HttpServletRequest; + import org.apache.commons.io.IOUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -37,7 +39,11 @@ public class SEBExamConfigDownload extends AbstractDownloadServiceHandler { } @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) .withURIVariable(API.PARAM_MODEL_ID, modelId) diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigPlaintextDownload.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigPlaintextDownload.java index 4c72cc58..6d844c37 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigPlaintextDownload.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/download/SEBExamConfigPlaintextDownload.java @@ -12,6 +12,8 @@ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; +import javax.servlet.http.HttpServletRequest; + import org.apache.commons.io.IOUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -37,7 +39,11 @@ public class SEBExamConfigPlaintextDownload extends AbstractDownloadServiceHandl } @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) .withURIVariable(API.PARAM_MODEL_ID, modelId) diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/exam/ExportSEBClientLogs.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/exam/ExportSEBClientLogs.java new file mode 100644 index 00000000..34aa0a68 --- /dev/null +++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/exam/ExportSEBClientLogs.java @@ -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() { + }), + HttpMethod.GET, + MediaType.APPLICATION_FORM_URLENCODED, + API.SEB_CLIENT_EVENT_ENDPOINT + + API.SEB_CLIENT_EVENT_EXPORT_PATH_SEGMENT); + } + +} diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java b/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java index 613e4b39..0d0fcb3b 100644 --- a/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java +++ b/src/main/java/ch/ethz/seb/sebserver/gui/table/EntityTable.java @@ -243,6 +243,14 @@ public class EntityTable { return this.name; } + public String getSortColumn() { + return this.sortColumn; + } + + public PageSortOrder getSortOrder() { + return this.sortOrder; + } + public EntityType getEntityType() { if (this.pageSupplier != null) { return this.pageSupplier.getEntityType(); diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java index 0d05a0fe..f35d938c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationService.java @@ -94,9 +94,9 @@ public interface PaginationService { * @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 */ - Result> fetch( - final int pageNumber, - final int pageSize, + Result> getPageOf( + final Integer pageNumber, + final Integer pageSize, final String sort, final String tableName, final Supplier>> delegate); diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java index 9bb99708..6083b130 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/PaginationServiceImpl.java @@ -148,16 +148,25 @@ public class PaginationServiceImpl implements PaginationService { } @Override - public Result> fetch( - final int pageNumber, - final int pageSize, + public Result> getPageOf( + final Integer pageNumber, + final Integer pageSize, final String sort, final String tableName, final Supplier>> delegate) { return Result.tryCatch(() -> { - setPagination(pageNumber, pageSize, sort, tableName); - return delegate.get().getOrThrow(); + //final SqlTable table = SqlTable.of(tableName); + final com.github.pagehelper.Page page = + setPagination(pageNumber, pageSize, sort, tableName); + + final Collection list = delegate.get().getOrThrow(); + + return new Page<>( + page.getPages(), + page.getPageNum(), + sort, + list); }); } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java index 9acacb4a..db63ff5a 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/SEBClientEventAdminService.java @@ -17,6 +17,7 @@ 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 { @@ -30,6 +31,7 @@ public interface SEBClientEventAdminService { String sort, ExportType exportType, boolean includeConnectionDetails, - boolean includeExamDetails); + boolean includeExamDetails, + final SEBServerUser currentUser); } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java index 759b5445..3c395bdd 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventAdminServiceImpl.java @@ -8,6 +8,7 @@ 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; @@ -30,6 +31,7 @@ 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; @@ -40,7 +42,6 @@ import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecord 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.AuthorizationService; 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; @@ -58,19 +59,16 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic private final ClientEventDAO clientEventDAO; private final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler; private final EnumMap exporter; - private final AuthorizationService authorizationService; public SEBClientEventAdminServiceImpl( final PaginationService paginationService, final ClientEventDAO clientEventDAO, final SEBClientEventExportTransactionHandler sebClientEventExportTransactionHandler, - final Collection exporter, - final AuthorizationService authorizationService) { + final Collection exporter) { this.paginationService = paginationService; this.clientEventDAO = clientEventDAO; this.sebClientEventExportTransactionHandler = sebClientEventExportTransactionHandler; - this.authorizationService = authorizationService; this.exporter = new EnumMap<>(ExportType.class); exporter.forEach(exp -> this.exporter.putIfAbsent(exp.exportType(), exp)); @@ -110,15 +108,32 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic final String sort, final ExportType exportType, final boolean includeConnectionDetails, - final boolean includeExamDetails) { + final boolean includeExamDetails, + final SEBServerUser currentUser) { - new exportRunner( - this.exporter.get(exportType), - includeConnectionDetails, - includeExamDetails, - new Pager(filterMap, sort), - output) - .run(); + 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(); + } + } } @@ -146,15 +161,12 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic this.pager = pager; this.output = output; - this.connectionCache = includeConnectionDetails ? new HashMap<>() : null; - this.examCache = includeExamDetails ? new HashMap<>() : null; + this.connectionCache = new HashMap<>(); + this.examCache = new HashMap<>(); } - public void run() { + public void run(final SEBServerUser currentUser) { - final SEBServerUser currentUser = SEBClientEventAdminServiceImpl.this.authorizationService - .getUserService() - .getCurrentUser(); final EnumSet userRoles = currentUser.getUserRoles(); final boolean isSupporterOnly = userRoles.size() == 1 && userRoles.contains(UserRole.EXAM_SUPPORTER); @@ -182,7 +194,7 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic if (!this.connectionCache.containsKey(connectionId)) { SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler .clientConnectionById(connectionId) - .map(e -> this.connectionCache.put(connectionId, e)) + .onSuccess(rec -> this.connectionCache.put(rec.getId(), rec)) .onError(error -> log.error("Failed to get ClientConnectionRecord for id: {}", connectionId, error)); @@ -197,7 +209,7 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic if (!this.examCache.containsKey(examId)) { SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler .examById(examId) - .map(e -> this.examCache.put(examId, e)) + .onSuccess(e -> this.examCache.put(examId, e)) .onError(error -> log.error("Failed to get Exam for id: {}", examId, error)); @@ -212,7 +224,7 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic private final FilterMap filterMap; private final String sort; - private int pageNumber = 0; + private int pageNumber = 1; private final int pageSize = 100; private Collection nextRecords; @@ -242,16 +254,22 @@ public class SEBClientEventAdminServiceImpl implements SEBClientEventAdminServic private void fetchNext() { try { - this.nextRecords = SEBClientEventAdminServiceImpl.this.paginationService.fetch( - this.pageNumber, - this.pageSize, - this.sort, - ClientEventRecordDynamicSqlSupport.clientEventRecord.name(), - () -> SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler - .allMatching(this.filterMap, Utils.truePredicate())) + final Page nextPage = SEBClientEventAdminServiceImpl.this.paginationService + .getPageOf( + this.pageNumber, + this.pageSize, + this.sort, + ClientEventRecordDynamicSqlSupport.clientEventRecord.name(), + () -> SEBClientEventAdminServiceImpl.this.sebClientEventExportTransactionHandler + .allMatching(this.filterMap, Utils.truePredicate())) .getOrThrow(); - this.pageNumber++; + 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); diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java index 151b82c0..38f031a6 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/exam/impl/SEBClientEventCSVExporter.java @@ -13,16 +13,22 @@ 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); @@ -55,6 +61,12 @@ public class SEBClientEventCSVExporter implements SEBClientEventExporter { 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(); + } } } @@ -69,23 +81,36 @@ public class SEBClientEventCSVExporter implements SEBClientEventExporter { final EventType type = EventType.byId(eventData.getType()); builder.append(type.name()); + builder.append(Constants.COMMA); builder.append(Utils.toCSVString(eventData.getText())); - builder.append(eventData.getNumericValue()); + 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()); } @@ -95,6 +120,12 @@ public class SEBClientEventCSVExporter implements SEBClientEventExporter { 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(); + } } } diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java index c7865d04..aab3faa7 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ClientEventController.java @@ -9,6 +9,8 @@ package ch.ethz.seb.sebserver.webservice.weblayer.api; import java.io.IOException; +import java.io.PipedInputStream; +import java.io.PipedOutputStream; import java.util.Collection; import java.util.EnumSet; import java.util.List; @@ -17,6 +19,7 @@ import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; +import org.apache.commons.io.IOUtils; import org.mybatis.dynamic.sql.SqlTable; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -157,6 +160,14 @@ public class ClientEventController extends ReadonlyEntityController allRequestParams, final HttpServletRequest request, @@ -169,21 +180,31 @@ public class ClientEventController extends ReadonlyEntityControllerPlease check carefully if all SEB client logs from the list shall be deleted.

There are currently {0} logs within the list. sebserver.seblogs.delete.action.delete=Delete All Logs