From 3e35a7745bc9af255170aa8fa7e38c9951f77a73 Mon Sep 17 00:00:00 2001 From: anhefti Date: Thu, 18 Nov 2021 08:21:03 +0100 Subject: [PATCH] fixes and fine-tuning for distributed setup --- .../sebserver/webservice/WebserviceInit.java | 7 ++++++ .../dao/impl/WebserviceInfoDAOImpl.java | 2 +- .../session/impl/ExamSessionServiceImpl.java | 22 ++++++++++++------- .../impl/SEBClientConnectionServiceImpl.java | 1 + .../impl/indicator/DistributedPingCache.java | 14 ++++++++++-- .../weblayer/api/ExamAPI_V1_Controller.java | 10 ++++----- .../config/application-dev-gui.properties | 2 +- .../config/application-dev-ws.properties | 6 +++-- .../config/application-dev.properties | 16 +++++++++----- .../config/application-gui.properties | 6 ++--- .../config/application-ws.properties | 1 + src/main/resources/config/ehcache.xml | 3 +-- .../integration/api/admin/WebserviceTest.java | 4 ++-- .../resources/application-test.properties | 3 ++- 14 files changed, 64 insertions(+), 33 deletions(-) diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java b/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java index 860f8358..91322c60 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/WebserviceInit.java @@ -86,8 +86,15 @@ public class WebserviceInit implements ApplicationListener"); SEBServerInit.INIT_LOGGER.info("----> *** Info:"); + SEBServerInit.INIT_LOGGER.info("----> JDBC connection pool max size: {}", + this.environment.getProperty("spring.datasource.hikari.maximumPoolSize")); + if (this.webserviceInfo.isDistributed()) { SEBServerInit.INIT_LOGGER.info("----> Distributed Setup: {}", this.webserviceInfo.getWebserviceUUID()); + SEBServerInit.INIT_LOGGER.info("------> Ping update time: {}", + this.environment.getProperty("sebserver.webservice.distributed.pingUpdate")); + SEBServerInit.INIT_LOGGER.info("------> Connection update time: {}", + this.environment.getProperty("sebserver.webservice.distributed.connectionUpdate")); } try { diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/dao/impl/WebserviceInfoDAOImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/dao/impl/WebserviceInfoDAOImpl.java index 7c03a347..cc452b89 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/dao/impl/WebserviceInfoDAOImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/dao/impl/WebserviceInfoDAOImpl.java @@ -40,7 +40,7 @@ public class WebserviceInfoDAOImpl implements WebserviceInfoDAO { public WebserviceInfoDAOImpl( final WebserviceServerInfoRecordMapper webserviceServerInfoRecordMapper, @Value("${sebserver.webservice.forceMaster:false}") final boolean forceMaster, - @Value("${sebserver.webservice.master.delay.threshold:10000}") final long masterDelayTimeThreshold) { + @Value("${sebserver.webservice.master.delay.threshold:30000}") final long masterDelayTimeThreshold) { this.webserviceServerInfoRecordMapper = webserviceServerInfoRecordMapper; this.masterDelayTimeThreshold = masterDelayTimeThreshold; diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ExamSessionServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ExamSessionServiceImpl.java index ad376ef0..eb2d0ad1 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ExamSessionServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/ExamSessionServiceImpl.java @@ -61,6 +61,7 @@ public class ExamSessionServiceImpl implements ExamSessionService { private final CacheManager cacheManager; private final SEBRestrictionService sebRestrictionService; private final boolean distributedSetup; + private final long distributedConnectionUpdate; private long lastConnectionTokenCacheUpdate = 0; @@ -72,7 +73,8 @@ public class ExamSessionServiceImpl implements ExamSessionService { final IndicatorDAO indicatorDAO, final CacheManager cacheManager, final SEBRestrictionService sebRestrictionService, - @Value("${sebserver.webservice.distributed:false}") final boolean distributedSetup) { + @Value("${sebserver.webservice.distributed:false}") final boolean distributedSetup, + @Value("${sebserver.webservice.distributed.connectionUpdate:2000}") final long distributedConnectionUpdate) { this.examSessionCacheService = examSessionCacheService; this.examDAO = examDAO; @@ -82,6 +84,7 @@ public class ExamSessionServiceImpl implements ExamSessionService { this.indicatorDAO = indicatorDAO; this.sebRestrictionService = sebRestrictionService; this.distributedSetup = distributedSetup; + this.distributedConnectionUpdate = distributedConnectionUpdate; } @Override @@ -114,9 +117,11 @@ public class ExamSessionServiceImpl implements ExamSessionService { return Result.tryCatch(() -> { final Collection result = new ArrayList<>(); - final Exam exam = this.examDAO - .byPK(examId) - .getOrThrow(); + final Exam exam = (this.isExamRunning(examId)) + ? this.examSessionCacheService.getRunningExam(examId) + : this.examDAO + .byPK(examId) + .getOrThrow(); // check lms connection if (exam.status == ExamStatus.CORRUPT_NO_LMS_CONNECTION) { @@ -193,7 +198,7 @@ public class ExamSessionServiceImpl implements ExamSessionService { } @Override - public synchronized Result getRunningExam(final Long examId) { + public /* synchronized */ Result getRunningExam(final Long examId) { if (log.isTraceEnabled()) { log.trace("Running exam request for exam {}", examId); } @@ -361,6 +366,7 @@ public class ExamSessionServiceImpl implements ExamSessionService { @Override public Result updateExamCache(final Long examId) { + final Exam exam = this.examSessionCacheService.getRunningExam(examId); if (exam == null) { return Result.ofEmpty(); @@ -398,10 +404,10 @@ public class ExamSessionServiceImpl implements ExamSessionService { // at least every second. This allows caching over multiple monitoring requests but // ensure an update every second for new incoming connections private void updateClientConnections(final Long examId) { - try { + final long currentTimeMillis = System.currentTimeMillis(); if (this.distributedSetup && - System.currentTimeMillis() - this.lastConnectionTokenCacheUpdate > Constants.SECOND_IN_MILLIS) { + currentTimeMillis - this.lastConnectionTokenCacheUpdate > this.distributedConnectionUpdate) { // go trough all client connection and update the ones that not up to date this.clientConnectionDAO.evictConnectionTokenCache(examId); @@ -420,7 +426,7 @@ public class ExamSessionServiceImpl implements ExamSessionService { .stream() .forEach(this.examSessionCacheService::evictClientConnection); - this.lastConnectionTokenCacheUpdate = System.currentTimeMillis(); + this.lastConnectionTokenCacheUpdate = currentTimeMillis; } } catch (final Exception e) { log.error("Unexpected error while trying to update client connections: ", e); diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/SEBClientConnectionServiceImpl.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/SEBClientConnectionServiceImpl.java index 804f560b..905a9953 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/SEBClientConnectionServiceImpl.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/SEBClientConnectionServiceImpl.java @@ -555,6 +555,7 @@ public class SEBClientConnectionServiceImpl implements SEBClientConnectionServic .getOrThrow() .stream() .flatMap(examId -> distributed + // TODO fetch only the connection tokens form active connections here ? this.clientConnectionDAO .getConnectionTokensNoCache(examId) .getOrThrow() diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/indicator/DistributedPingCache.java b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/indicator/DistributedPingCache.java index 5c027577..3384fc89 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/indicator/DistributedPingCache.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/servicelayer/session/impl/indicator/DistributedPingCache.java @@ -30,6 +30,7 @@ import org.springframework.transaction.annotation.Transactional; 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.Utils; import ch.ethz.seb.sebserver.webservice.WebserviceInfo; import ch.ethz.seb.sebserver.webservice.datalayer.batis.ClientEventLastPingMapper; import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.ClientEventRecordDynamicSqlSupport; @@ -45,9 +46,11 @@ public class DistributedPingCache implements DisposableBean { private final ClientEventLastPingMapper clientEventLastPingMapper; private final ClientEventRecordMapper clientEventRecordMapper; - private ScheduledFuture taskRef; + private final long pingUpdateTolerance; + private ScheduledFuture taskRef; private final Map pingCache = new ConcurrentHashMap<>(); + private long lastUpdate = 0L; public DistributedPingCache( final ClientEventLastPingMapper clientEventLastPingMapper, @@ -58,6 +61,7 @@ public class DistributedPingCache implements DisposableBean { this.clientEventLastPingMapper = clientEventLastPingMapper; this.clientEventRecordMapper = clientEventRecordMapper; + this.pingUpdateTolerance = pingUpdate * 2 / 3; if (webserviceInfo.isDistributed()) { try { this.taskRef = taskScheduler.scheduleAtFixedRate(this::updateCache, pingUpdate); @@ -166,7 +170,6 @@ public class DistributedPingCache implements DisposableBean { try { Long ping = this.pingCache.get(pingRecordId); if (ping == null) { - if (log.isDebugEnabled()) { log.debug("*** Get and cache ping time: {}", pingRecordId); } @@ -187,6 +190,12 @@ public class DistributedPingCache implements DisposableBean { @Transactional(readOnly = true, isolation = Isolation.READ_UNCOMMITTED) public void updateCache() { + final long millisecondsNow = Utils.getMillisecondsNow(); + if (millisecondsNow - this.lastUpdate < this.pingUpdateTolerance) { + log.warn("Skip ping update schedule because the last one was less then 2 seconds ago"); + return; + } + if (this.pingCache.isEmpty()) { return; } @@ -210,6 +219,7 @@ public class DistributedPingCache implements DisposableBean { if (mapping != null) { this.pingCache.clear(); this.pingCache.putAll(mapping); + this.lastUpdate = millisecondsNow; } } catch (final Exception e) { diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ExamAPI_V1_Controller.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ExamAPI_V1_Controller.java index d3c7d614..5aed1f6c 100644 --- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ExamAPI_V1_Controller.java +++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/ExamAPI_V1_Controller.java @@ -270,20 +270,20 @@ public class ExamAPI_V1_Controller { final String pingNumString = request.getParameter(API.EXAM_API_PING_NUMBER); final String instructionConfirm = request.getParameter(API.EXAM_API_PING_INSTRUCTION_CONFIRM); - if (log.isTraceEnabled()) { - log.trace("****************** SEB client connection: {} ip: {}", - connectionToken, - getClientAddress(request)); + if (connectionToken == null) { + log.warn("Missing connection token on ping. Ignore the request"); + return; } if (instructionConfirm != null) { this.sebClientConnectionService.confirmInstructionDone(connectionToken, instructionConfirm); } + final Long clientTime = timeStampString != null ? Long.parseLong(timeStampString) : 0L; final String instruction = this.sebClientConnectionService .notifyPing( connectionToken, - Long.parseLong(timeStampString), + clientTime, pingNumString != null ? Integer.parseInt(pingNumString) : -1); if (instruction == null) { diff --git a/src/main/resources/config/application-dev-gui.properties b/src/main/resources/config/application-dev-gui.properties index 3e830c82..8069b264 100644 --- a/src/main/resources/config/application-dev-gui.properties +++ b/src/main/resources/config/application-dev-gui.properties @@ -8,7 +8,7 @@ sebserver.gui.webservice.address=localhost sebserver.gui.webservice.port=8080 sebserver.gui.webservice.apipath=/admin-api/v1 # defines the polling interval that is used to poll the webservice for client connection data on a monitored exam page -#sebserver.gui.webservice.poll-interval=1000 +sebserver.gui.webservice.poll-interval=1000 sebserver.gui.theme=css/sebserver.css sebserver.gui.list.page.size=15 diff --git a/src/main/resources/config/application-dev-ws.properties b/src/main/resources/config/application-dev-ws.properties index 5b2e5488..cef30a70 100644 --- a/src/main/resources/config/application-dev-ws.properties +++ b/src/main/resources/config/application-dev-ws.properties @@ -13,12 +13,14 @@ spring.datasource.hikari.initializationFailTimeout=30000 spring.datasource.hikari.connectionTimeout=30000 spring.datasource.hikari.idleTimeout=600000 spring.datasource.hikari.maxLifetime=1800000 -spring.datasource.hikari.maximumPoolSize=5 +spring.datasource.hikari.maximumPoolSize=10 +spring.datasource.hikari.leakDetectionThreshold=2000 sebserver.http.client.connect-timeout=15000 sebserver.http.client.connection-request-timeout=10000 sebserver.http.client.read-timeout=20000 - +sebserver.webservice.distributed.pingUpdate=1000 +sebserver.webservice.distributed.connectionUpdate=1000 sebserver.webservice.clean-db-on-startup=false # webservice configuration diff --git a/src/main/resources/config/application-dev.properties b/src/main/resources/config/application-dev.properties index ae66eec3..ed384d92 100644 --- a/src/main/resources/config/application-dev.properties +++ b/src/main/resources/config/application-dev.properties @@ -10,12 +10,16 @@ server.tomcat.uri-encoding=UTF-8 logging.level.ch=INFO logging.level.ch.ethz.seb.sebserver.webservice.datalayer=INFO logging.level.org.springframework.cache=INFO -logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl=TRACE -logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session=DEBUG -logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring=TRACE +logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl=INFO +logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session=INFO +logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring=INFO +logging.level.ch.ethz.seb.sebserver.webservice.servicelayer.dao.impl=DEBUG +#logging.level.ch.ethz.seb.sebserver.webservice.datalayer.batis=DEBUG +#logging.level.ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper=DEBUG #logging.level.ch.ethz.seb.sebserver.webservice.weblayer.api.ExamAPI_V1_Controller=TRACE +logging.level.com.zaxxer.hikari=DEBUG -sebserver.http.client.connect-timeout=150000 -sebserver.http.client.connection-request-timeout=100000 -sebserver.http.client.read-timeout=200000 +sebserver.http.client.connect-timeout=15000 +sebserver.http.client.connection-request-timeout=10000 +sebserver.http.client.read-timeout=60000 diff --git a/src/main/resources/config/application-gui.properties b/src/main/resources/config/application-gui.properties index 2fe17115..dd575a20 100644 --- a/src/main/resources/config/application-gui.properties +++ b/src/main/resources/config/application-gui.properties @@ -24,9 +24,9 @@ sebserver.gui.http.webservice.contextPath=${server.servlet.context-path} sebserver.gui.entrypoint=/gui -sebserver.http.client.connect-timeout=60000 -sebserver.http.client.connection-request-timeout=100000 -sebserver.http.client.read-timeout=200000 +sebserver.http.client.connect-timeout=15000 +sebserver.http.client.connection-request-timeout=10000 +sebserver.http.client.read-timeout=60000 sebserver.gui.webservice.apipath=${sebserver.webservice.api.admin.endpoint} # defines the polling interval that is used to poll the webservice for client connection data on a monitored exam page diff --git a/src/main/resources/config/application-ws.properties b/src/main/resources/config/application-ws.properties index 5c3fc468..4325f14a 100644 --- a/src/main/resources/config/application-ws.properties +++ b/src/main/resources/config/application-ws.properties @@ -32,6 +32,7 @@ spring.datasource.hikari.connectionTimeout=30000 spring.datasource.hikari.idleTimeout=600000 spring.datasource.hikari.maxLifetime=1800000 spring.datasource.hikari.maximumPoolSize=100 +spring.datasource.hikari.leakDetectionThreshold=10000 ### webservice security spring.datasource.password=${sebserver.mariadb.password} diff --git a/src/main/resources/config/ehcache.xml b/src/main/resources/config/ehcache.xml index 328874b7..ecc1b6f6 100644 --- a/src/main/resources/config/ehcache.xml +++ b/src/main/resources/config/ehcache.xml @@ -86,12 +86,11 @@ java.lang.String ch.ethz.seb.sebserver.gbl.model.exam.QuizData - 5 + 10 10000 - \ No newline at end of file diff --git a/src/test/java/ch/ethz/seb/sebserver/webservice/integration/api/admin/WebserviceTest.java b/src/test/java/ch/ethz/seb/sebserver/webservice/integration/api/admin/WebserviceTest.java index 3770935c..573be2e1 100644 --- a/src/test/java/ch/ethz/seb/sebserver/webservice/integration/api/admin/WebserviceTest.java +++ b/src/test/java/ch/ethz/seb/sebserver/webservice/integration/api/admin/WebserviceTest.java @@ -67,7 +67,7 @@ public class WebserviceTest extends AdministrationAPIIntegrationTester { assertFalse(this.webserviceInfoDAO.isMaster(WEBSERVICE_2)); try { - Thread.sleep(5000); + Thread.sleep(500); } catch (final InterruptedException e) { } @@ -75,7 +75,7 @@ public class WebserviceTest extends AdministrationAPIIntegrationTester { assertFalse(this.webserviceInfoDAO.isMaster(WEBSERVICE_2)); try { - Thread.sleep(6000); + Thread.sleep(600); } catch (final InterruptedException e) { } diff --git a/src/test/resources/application-test.properties b/src/test/resources/application-test.properties index 4f74dda7..46784e83 100644 --- a/src/test/resources/application-test.properties +++ b/src/test/resources/application-test.properties @@ -41,4 +41,5 @@ sebserver.webservice.api.redirect.unauthorized=none sebserver.webservice.lms.openedx.api.token.request.paths=/oauth2/access_token sebserver.webservice.lms.moodle.api.token.request.paths -management.endpoints.web.base-path=/actuator \ No newline at end of file +management.endpoints.web.base-path=/actuator +sebserver.webservice.master.delay.threshold=1000 \ No newline at end of file