diff --git a/src/main/java/org/olat/admin/sysinfo/LargeFilesController.java b/src/main/java/org/olat/admin/sysinfo/LargeFilesController.java
index 44a328014955a5a781ccf8a90d4924794e0a3113..c13b68c584bed87516fcf103fc91d14bd9b6cd2f 100644
--- a/src/main/java/org/olat/admin/sysinfo/LargeFilesController.java
+++ b/src/main/java/org/olat/admin/sysinfo/LargeFilesController.java
@@ -79,6 +79,7 @@ import org.olat.core.util.StringHelper;
 import org.olat.core.util.mail.ContactList;
 import org.olat.core.util.mail.ContactMessage;
 import org.olat.modules.co.ContactFormController;
+import org.olat.user.ui.UserDisplayNameCellRenderer;
 import org.springframework.beans.factory.annotation.Autowired;
 
 
@@ -215,9 +216,7 @@ public class LargeFilesController extends FormBasicController implements Extende
 			}
 		}
 
-		Collections.sort(rows, (row1,row2) -> {
-			return row2.getSize().intValue() - row1.getSize().intValue();
-		});
+		Collections.sort(rows, (row1, row2) -> (row2.getSize().intValue() - row1.getSize().intValue()));
 
 		if(maxResults != 0 && maxResults < rows.size()) {
 			rows = rows.subList(0, maxResults);
@@ -332,7 +331,9 @@ public class LargeFilesController extends FormBasicController implements Extende
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.fileType));
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.downloadCount));
 
-		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.author, "selectAuthor"));
+		DefaultFlexiColumnModel authorModel = new DefaultFlexiColumnModel(false, LargeFilesTableColumns.author, "selectAuthor");
+		authorModel.setCellRenderer(UserDisplayNameCellRenderer.get());
+		columnsModel.addFlexiColumnModel(authorModel);
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.license));
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.language));
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.source));
@@ -342,7 +343,9 @@ public class LargeFilesController extends FormBasicController implements Extende
 		columnsModel.addFlexiColumnModel(column);
 
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.lockedAt));
-		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.lockedBy, "selectLockedBy"));
+		DefaultFlexiColumnModel lockedByModel = new DefaultFlexiColumnModel(false, LargeFilesTableColumns.lockedBy, "selectLockedBy");
+		lockedByModel.setCellRenderer(UserDisplayNameCellRenderer.get());
+		columnsModel.addFlexiColumnModel(lockedByModel);
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.creator));
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.publisher));
 		columnsModel.addFlexiColumnModel(new DefaultFlexiColumnModel(false, LargeFilesTableColumns.pubDate));
diff --git a/src/main/java/org/olat/admin/sysinfo/model/LargeFilesTableModel.java b/src/main/java/org/olat/admin/sysinfo/model/LargeFilesTableModel.java
index ad95804a6db32ee2f9932cbc97e769c7df97d601..4fa781895703991066a6846b73d0304c35b650b7 100644
--- a/src/main/java/org/olat/admin/sysinfo/model/LargeFilesTableModel.java
+++ b/src/main/java/org/olat/admin/sysinfo/model/LargeFilesTableModel.java
@@ -42,7 +42,7 @@ public class LargeFilesTableModel extends DefaultFlexiTableDataModel<LargeFilesT
 implements SortableFlexiTableDataModel<LargeFilesTableContentRow> {
 
 	private final Locale locale;
-
+	
 	public LargeFilesTableModel(FlexiTableColumnModel columnModel, Locale locale) {
 		super(columnModel);
 		this.locale = locale;
@@ -72,7 +72,7 @@ implements SortableFlexiTableDataModel<LargeFilesTableContentRow> {
 		case contextInfo: return row.getContextInfo();
 		case path: return row.getPath();
 		case showPath: return row.getShowPath();
-		case author: return row.getAuthor() != null ? row.getAuthor().getUser().getFirstName() + " " + row.getAuthor().getUser().getLastName() : null;
+		case author: return row.getAuthor();
 		case revision: return row.isRevision();
 		case fileCategory: return row.getFileCategory();
 		case fileType: return row.getFileType();
@@ -91,7 +91,7 @@ implements SortableFlexiTableDataModel<LargeFilesTableContentRow> {
 		case pubDate: return returnNullSafeDate(row.getPubDate());
 		case language: return row.getLanguage();
 		case locked: return row.isLocked();
-		case lockedBy: return row.getLockedBy() != null ? row.getLockedBy().getUser().getFirstName() + " " + row.getLockedBy().getUser().getLastName() : null;
+		case lockedBy: return row.getLockedBy();
 		case lockedAt: return returnNullSafeDate(row.getLockedAt());
 		case revisionComment: return row.getRevisionComment();
 		case revisionNr: return row.getRevisionNr();
diff --git a/src/main/java/org/olat/core/commons/editor/htmleditor/_i18n/LocalStrings_de.properties b/src/main/java/org/olat/core/commons/editor/htmleditor/_i18n/LocalStrings_de.properties
index f2f3f3a5dc85b8102b3dfffbe29c6f3a3ca1a28a..45ed72dceee7e321e19f031ce3339e53ab89cb9f 100644
--- a/src/main/java/org/olat/core/commons/editor/htmleditor/_i18n/LocalStrings_de.properties
+++ b/src/main/java/org/olat/core/commons/editor/htmleditor/_i18n/LocalStrings_de.properties
@@ -7,4 +7,4 @@ saveandclosebuttontext=Speichern und schliessen
 savebuttontext=Speichern
 warn.foreigneditor=Achtung\! Diese Datei wurde urspr\u00FCnglich mit einem anderen HTML-Editor erstellt. Wenn sie mit dem OLAT-internen Editor bearbeitet und gespeichert wird, kann die Formatierung und der resultierende Quellcode stark ver\u00E4ndert werden\!
 plaintext.error.tolarge=$org.olat.core.commons.editor.plaintexteditor\:plaintext.error.tolarge
-file.too.large.server=Die Datei konnte nicht gespeichert werden da sie zu gross ist ({0}KB). Zugelassen sind Dateien bis {1}KB. 
+file.too.large.server=Die Datei konnte nicht gespeichert werden, da sie zu gross ist ({0}KB). Zugelassen sind Dateien bis {1}KB. 
diff --git a/src/main/java/org/olat/core/commons/services/doceditor/onlyoffice/OnlyOfficeModule.java b/src/main/java/org/olat/core/commons/services/doceditor/onlyoffice/OnlyOfficeModule.java
index cbde2c1c42b5ee5007cac8169e99420096a35607..3bad8795a747cd12a3daa7ff747dfd557bb4e432 100644
--- a/src/main/java/org/olat/core/commons/services/doceditor/onlyoffice/OnlyOfficeModule.java
+++ b/src/main/java/org/olat/core/commons/services/doceditor/onlyoffice/OnlyOfficeModule.java
@@ -241,8 +241,9 @@ public class OnlyOfficeModule extends AbstractSpringModule implements ConfigOnOf
 		if (mobileModes == null) {
 			if (StringHelper.containsNonWhitespace(mobileModesConfig)) {
 				mobileModes = Arrays.stream(mobileModesConfig.split(",")).map(Mode::valueOf).collect(Collectors.toSet());
+			} else {
+				mobileModes = Collections.emptySet();
 			}
-			mobileModes = Collections.emptySet();
 		}
 		return mobileModes;
 	}
diff --git a/src/main/java/org/olat/core/commons/services/webdav/manager/WebDAVManagerImpl.java b/src/main/java/org/olat/core/commons/services/webdav/manager/WebDAVManagerImpl.java
index 1e74df3c3c187e65707244a195fe8e651433b701..99b2037912278a8373a3ff778ecb26a939ff2c21 100644
--- a/src/main/java/org/olat/core/commons/services/webdav/manager/WebDAVManagerImpl.java
+++ b/src/main/java/org/olat/core/commons/services/webdav/manager/WebDAVManagerImpl.java
@@ -77,8 +77,10 @@ public class WebDAVManagerImpl implements WebDAVManager, InitializingBean {
 	private static boolean enabled = true;
 	
 	public static final String BASIC_AUTH_REALM = "OLAT WebDAV Access";
+	private static final String DIR_CONTEXT = "_DIRCTX";
 	private CoordinatorManager coordinatorManager;
 
+	private CacheWrapper<Long,VFSResourceRoot> resourceCache;
 	private CacheWrapper<CacheKey,UserSession> timedSessionCache;
 
 	@Autowired
@@ -97,7 +99,10 @@ public class WebDAVManagerImpl implements WebDAVManager, InitializingBean {
 
 	@Override
 	public void afterPropertiesSet() throws Exception {
-		timedSessionCache = coordinatorManager.getCoordinator().getCacher().getCache(WebDAVManager.class.getSimpleName(), "webdav");
+		resourceCache = coordinatorManager.getCoordinator().getCacher()
+				.getCache(WebDAVManager.class.getSimpleName(), "resources");
+		timedSessionCache = coordinatorManager.getCoordinator().getCacher()
+				.getCache(WebDAVManager.class.getSimpleName(), "webdav");
 	}
 	
 	@Override
@@ -108,21 +113,23 @@ public class WebDAVManagerImpl implements WebDAVManager, InitializingBean {
 		}
 
 		usess.getSessionInfo().setLastClickTime();
-		VFSResourceRoot fdc = (VFSResourceRoot)usess.getEntry("_DIRCTX");
+		VFSResourceRoot fdc = (VFSResourceRoot)usess.getEntry(DIR_CONTEXT);
 		if (fdc != null) {
 			return fdc;
 		}
-		
-		IdentityEnvironment identityEnv = usess.getIdentityEnvironment();
-		VFSContainer webdavContainer = getMountableRoot(identityEnv);
-		
-		//create the / folder
-		VirtualContainer rootContainer = new VirtualContainer("");
-		rootContainer.addItem(webdavContainer);
-		rootContainer.setLocalSecurityCallback(new ReadOnlyCallback());
 
-		fdc = new VFSResourceRoot(identityEnv.getIdentity(), rootContainer);
-		usess.putEntry("_DIRCTX", fdc);
+		final IdentityEnvironment identityEnv = usess.getIdentityEnvironment();
+		final Identity identity = identityEnv.getIdentity();
+		
+		fdc = resourceCache.computeIfAbsent(identity.getKey(), key -> {
+			VFSContainer webdavContainer = getMountableRoot(identityEnv);
+			//create the / folder
+			VirtualContainer rootContainer = new VirtualContainer("");
+			rootContainer.addItem(webdavContainer);
+			rootContainer.setLocalSecurityCallback(new ReadOnlyCallback());
+			return new VFSResourceRoot(identity, rootContainer);
+		});
+		usess.putEntry(DIR_CONTEXT, fdc);
 		return fdc;
 	}
 	
diff --git a/src/main/java/org/olat/course/nodes/iq/EndTimeCellRenderer.java b/src/main/java/org/olat/course/nodes/iq/EndTimeCellRenderer.java
index 2facab1d34134dd4df769b255327eea26ecf102e..4a96a4eeac5a56b5e03a2640eb1cb75b6007ebcf 100644
--- a/src/main/java/org/olat/course/nodes/iq/EndTimeCellRenderer.java
+++ b/src/main/java/org/olat/course/nodes/iq/EndTimeCellRenderer.java
@@ -58,28 +58,27 @@ public class EndTimeCellRenderer implements FlexiCellRenderer {
 			Integer compensationExtraTimeInSeconds = infos.getCompensationExtraTimeInSeconds();
 			
 			if(infos.getStart() != null) {
-				int totalTime = timeLimitInSeconds;
+				long totalTimeMs = timeLimitInSeconds * 1000l;
 				if(endDate != null) {
-					long leadingTimeInMilliSeconds = endDate.getTime() - infos.getStart().getTime();
-					int leadingTime = Math.round(leadingTimeInMilliSeconds / 1000f);
+					long leadingTimeInMs = endDate.getTime() - infos.getStart().getTime();
 					if(timeLimitInSeconds > 0) {
-						totalTime = Math.min(totalTime, leadingTime);
+						totalTimeMs = Math.min(totalTimeMs, leadingTimeInMs);
 					} else {
-						totalTime = leadingTime;
+						totalTimeMs = leadingTimeInMs;
 					}
 				}
 
 				if(extraTimeInSeconds != null) {
-					totalTime += extraTimeInSeconds;
+					totalTimeMs += (extraTimeInSeconds * 1000l);
 				}
 				if(compensationExtraTimeInSeconds != null) {
-					totalTime += compensationExtraTimeInSeconds;
+					totalTimeMs += (compensationExtraTimeInSeconds * 1000l);
 				}
 				
 				Calendar now = Calendar.getInstance();
 				Calendar cal = Calendar.getInstance();
 				cal.setTime(infos.getStart());
-				cal.add(Calendar.SECOND, totalTime);
+				cal.add(Calendar.MILLISECOND, (int)totalTimeMs);
 				Date dueDate = cal.getTime();
 
 				boolean sameDay = now.get(Calendar.YEAR) == cal.get(Calendar.YEAR)
diff --git a/src/main/java/org/olat/course/run/CourseRuntimeController.java b/src/main/java/org/olat/course/run/CourseRuntimeController.java
index 1dbc506ca73a419793cb8d8840e2dff59aecdaa8..d6b4ee348f3780de04cf8af9ef7562e771975d00 100644
--- a/src/main/java/org/olat/course/run/CourseRuntimeController.java
+++ b/src/main/java/org/olat/course/run/CourseRuntimeController.java
@@ -468,11 +468,13 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.setBreadcrumbEnabled(false);
 		}
 		
+		RunMainController rmc = getRunMainController();
+		boolean disclaimerAccepted = rmc != null && rmc.isDisclaimerAccepted();
+		
 		initToolsMenu(toolsDropdown);
 		initToolsMyCourse(course);
-		initGeneralTools(course);
+		initGeneralTools(course, disclaimerAccepted);
 		
-		RunMainController rmc = getRunMainController();
 		if(rmc != null) {
 			rmc.initToolbarAndProgressbar();
 		}
@@ -840,7 +842,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 		return (userCourseEnv.isParticipant() || !userCourseEnv.getParticipatingGroups().isEmpty());
 	}
 	
-	private void initGeneralTools(ICourse course) {
+	private void initGeneralTools(ICourse course, boolean disclaimerAccepted) {
 		boolean assessmentLock = isAssessmentLock();
 		UserCourseEnvironment userCourseEnv = getUserCourseEnvironment();
 
@@ -852,7 +854,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(detailsLink);
 		}
 		
-		if (!assessmentLock && !isGuestOnly
+		if (!assessmentLock && !isGuestOnly && disclaimerAccepted
 				&& LearningPathNodeAccessProvider.TYPE.equals(cc.getNodeAccessType().getType())) {
 			learningPathLink = LinkFactory.createToolLink("learningPath", translate("command.learning.path"), this, CourseTool.learningpath.getIconCss());
 			learningPathLink.setUrl(BusinessControlFactory.getInstance()
@@ -863,7 +865,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			learningPathLink.setVisible(userCourseEnv != null && userCourseEnv.isParticipant());
 		}
 		
-		if (!assessmentLock && !isGuestOnly
+		if (!assessmentLock && !isGuestOnly && disclaimerAccepted
 				&& LearningPathNodeAccessProvider.TYPE.equals(cc.getNodeAccessType().getType())) {
 			learningPathsLink = LinkFactory.createToolLink("learningPaths", translate("command.learning.paths"), this, CourseTool.learningpath.getIconCss());
 			learningPathsLink.setUrl(BusinessControlFactory.getInstance()
@@ -874,7 +876,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			learningPathsLink.setVisible(userCourseEnv != null && (userCourseEnv.isCoach() || userCourseEnv.isAdmin()));
 		}
 		
-		boolean calendarIsEnabled =  !assessmentLock && !isGuestOnly && calendarModule.isEnabled()
+		boolean calendarIsEnabled = !assessmentLock && !isGuestOnly && disclaimerAccepted && calendarModule.isEnabled()
 				&& calendarModule.isEnableCourseToolCalendar() && reSecurity.canLaunch();
 		if (calendarIsEnabled && userCourseEnv != null) {
 			calendarLink = LinkFactory.createToolLink("calendar",translate("command.calendar"), this, "o_icon_calendar");
@@ -885,14 +887,14 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(calendarLink);
 		}
 		
-		if(!assessmentLock && isLecturesLinkEnabled()) {
+		if(!assessmentLock && disclaimerAccepted && isLecturesLinkEnabled()) {
 			lecturesLink = LinkFactory.createToolLink("command.lectures", translate("command.lectures"), this, "o_icon_lecture");
 			lecturesLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[Lectures:0]"));
 			toolbarPanel.addTool(lecturesLink);
 		}
 		
-		if(!assessmentLock && !isGuestOnly && userCourseEnv != null) {
+		if(!assessmentLock && !isGuestOnly && disclaimerAccepted && userCourseEnv != null) {
 			participantListLink = LinkFactory.createToolLink("participantlist",
 					translate(CourseTool.participantlist.getI18nKey()), this,
 					CourseTool.participantlist.getIconCss());
@@ -902,7 +904,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(participantListLink);
 		}
 		
-		if(!assessmentLock && userCourseEnv != null) {
+		if(!assessmentLock && disclaimerAccepted && userCourseEnv != null) {
 			participantInfoLink = LinkFactory.createToolLink("participantinfo",
 					translate(CourseTool.participantinfos.getI18nKey()), this,
 					CourseTool.participantinfos.getIconCss());
@@ -912,7 +914,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(participantInfoLink);
 		}
 		
-		if(!assessmentLock && !isGuestOnly && userCourseEnv != null && !userCourseEnv.isCourseReadOnly()) {
+		if(!assessmentLock && !isGuestOnly && disclaimerAccepted && userCourseEnv != null && !userCourseEnv.isCourseReadOnly()) {
 			emailLink = LinkFactory.createToolLink("email", translate(CourseTool.email.getI18nKey()), this, CourseTool.email.getIconCss());
 			emailLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[email:0]"));
@@ -920,7 +922,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(emailLink);
 		}
 		
-		if(!assessmentLock && userCourseEnv != null) {
+		if(!assessmentLock && disclaimerAccepted && userCourseEnv != null) {
 			blogLink = LinkFactory.createToolLink("blog", translate(CourseTool.blog.getI18nKey()), this, CourseTool.blog.getIconCss());
 			blogLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[blog:0]"));
@@ -928,7 +930,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(blogLink);
 		}
 		
-		if(!assessmentLock && userCourseEnv != null) {
+		if(!assessmentLock && disclaimerAccepted && userCourseEnv != null) {
 			wikiLink = LinkFactory.createToolLink("wiki", translate(CourseTool.wiki.getI18nKey()), this, CourseTool.wiki.getIconCss());
 			wikiLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[wiki:0]"));
@@ -936,7 +938,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(wikiLink);
 		}
 		
-		if(!assessmentLock && userCourseEnv != null) {
+		if(!assessmentLock && disclaimerAccepted && userCourseEnv != null) {
 			forumLink = LinkFactory.createToolLink("forum", translate(CourseTool.forum.getI18nKey()), this, CourseTool.forum.getIconCss());
 			forumLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[forum:0]"));
@@ -944,7 +946,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(forumLink);
 		}
 		
-		if(!assessmentLock && userCourseEnv != null) {
+		if(!assessmentLock && disclaimerAccepted && userCourseEnv != null) {
 			documentsLink = LinkFactory.createToolLink("documents", translate(CourseTool.documents.getI18nKey()), this, CourseTool.documents.getIconCss());
 			documentsLink.setUrl(BusinessControlFactory.getInstance()
 					.getAuthenticatedURLFromBusinessPathStrings(businessPathEntry, "[documents:0]"));
@@ -952,7 +954,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			toolbarPanel.addTool(documentsLink);
 		}
 		
-		if(!assessmentLock) {
+		if(!assessmentLock && disclaimerAccepted) {
 			glossary = new Dropdown("glossary", "command.glossary", false, getTranslator());
 			glossary.setIconCSS("o_icon o_FileResource-GLOSSARY_icon");
 			glossary.setVisible(cc.hasGlossary() && cc.isGlossaryEnabled());
@@ -967,7 +969,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 		}
 		
 		//add group chat to toolbox
-		boolean chatIsEnabled = !assessmentLock && !isGuestOnly && imModule.isEnabled()
+		boolean chatIsEnabled = !assessmentLock && !isGuestOnly && disclaimerAccepted && imModule.isEnabled()
 				&& imModule.isCourseEnabled() && reSecurity.canLaunch();
 		if(chatIsEnabled && userCourseEnv != null && !userCourseEnv.isCourseReadOnly()) {
 			chatLink = LinkFactory.createToolLink("chat",translate("command.coursechat"), this, "o_icon_chat");
@@ -976,7 +978,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 		}
 		
 		// add course search to toolbox 
-		boolean isSearchEnabled = !assessmentLock && searchModule.isSearchAllowed(roles);
+		boolean isSearchEnabled = !assessmentLock && disclaimerAccepted && searchModule.isSearchAllowed(roles);
 		if (isSearchEnabled) {
 			searchLink = LinkFactory.createToolLink("coursesearch", translate("command.coursesearch"), this, "o_icon_search");
 			searchLink.setVisible(cc.isCourseSearchEnabled());
@@ -1152,6 +1154,8 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			} else if (event instanceof OpenCourseToolEvent) {
 				CourseTool tool = ((OpenCourseToolEvent)event).getTool();
 				doOpenTool(ureq, tool);
+			} else if (event == RunMainController.COURSE_DISCLAIMER_ACCEPTED) {
+				initToolbar();
 			}
 		} else if (lifeCycleChangeCtr == source) {
 			if (event == RepositoryEntryLifeCycleChangeController.deletedEvent) {
@@ -2026,7 +2030,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			currentToolCtr = participatListCtrl;
 		} else {
 			delayedClose = Delayed.participantList;
-		};
+		}
 	}
 	
 	private void doParticipantInfo(UserRequest ureq) {
@@ -2056,7 +2060,7 @@ public class CourseRuntimeController extends RepositoryEntryRuntimeController im
 			currentToolCtr = participatInfoCtrl;
 		} else {
 			delayedClose = Delayed.participantInfo;
-		};
+		}
 	}
 	
 	private void doEmail(UserRequest ureq) {
diff --git a/src/main/java/org/olat/course/run/RunMainController.java b/src/main/java/org/olat/course/run/RunMainController.java
index 3ea0696c395d78bc548e1eca83bde513d8b853ef..2a970478296df30e4fe637387cc9152a690b631d 100644
--- a/src/main/java/org/olat/course/run/RunMainController.java
+++ b/src/main/java/org/olat/course/run/RunMainController.java
@@ -125,6 +125,7 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 
 	private static final Logger log = Tracing.createLoggerFor(RunMainController.class);
 
+	public static final Event COURSE_DISCLAIMER_ACCEPTED = new Event("course-disclaimer-accepted");
 	public static final String REBUILD = "rebuild";
 	public static final String ORES_TYPE_COURSE_RUN = OresHelper.calculateTypeName(RunMainController.class, CourseModule.ORES_TYPE_COURSE);
 	private final OLATResourceable courseRunOres; //course run ores for course run channel 
@@ -154,6 +155,7 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 	private boolean needsRebuildAfter = false;
 	private boolean needsRebuildAfterPublish = false;
 	private boolean needsRebuildAfterRunDone = false;
+	private boolean disclaimerAccepted = true;
 	
 	private String courseTitle;
 	private Link nextLink, previousLink;
@@ -283,13 +285,15 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 		coursemain.contextPut("courserepokey", courseRepositoryEntry.getKey());
 		
 		// if a disclaimer is enabled, show it first
-		if (courseModule.isDisclaimerEnabled() && course.getCourseEnvironment().getCourseConfig().isDisclaimerEnabled() && 
-				!disclaimerManager.isAccessGranted(courseRepositoryEntry, getIdentity(), ureq.getUserSession().getRoles())) {
+		disclaimerAccepted = !courseModule.isDisclaimerEnabled()
+				|| !course.getCourseEnvironment().getCourseConfig().isDisclaimerEnabled() 
+				|| disclaimerManager.isAccessGranted(courseRepositoryEntry, getIdentity(), ureq.getUserSession().getRoles());
+		if (disclaimerAccepted) {
+			coursemain.put("coursemain", columnLayoutCtr.getInitialComponent());
+		} else {
 			disclaimerController = new CourseDisclaimerConsentController(ureq, getWindowControl(), courseRepositoryEntry);
 			listenTo(disclaimerController);
 			coursemain.put("coursemain", disclaimerController.getInitialComponent());
-		} else {
-			coursemain.put("coursemain", columnLayoutCtr.getInitialComponent());
 		}
 		
 		// on initial call we have to set the data-nodeid manually. later it
@@ -318,6 +322,10 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 		CoordinatorManager.getInstance().getCoordinator().getEventBus().registerFor(this, identity, courseRepositoryEntry);
 	}
 	
+	public boolean isDisclaimerAccepted() {
+		return disclaimerAccepted;
+	}
+
 	protected void setTextMarkingEnabled(boolean enabled) {
 		if (glossaryMarkerCtr != null) {
 			glossaryMarkerCtr.setTextMarkingEnabled(enabled);
@@ -431,7 +439,7 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 		} else {
 			List<TreeNode> flatTree = new ArrayList<>();
 			TreeHelper.makeTreeFlat(luTree.getTreeModel().getRootNode(), flatTree);
-			hasPrevious = getPreviousNonDelegatingNode(flatTree, luTree.getSelectedNode()) != null;;
+			hasPrevious = getPreviousNonDelegatingNode(flatTree, luTree.getSelectedNode()) != null;
 			int index = flatTree.indexOf(luTree.getSelectedNode());
 			hasNext = index  >= 0 && index+1 < flatTree.size();
 		}
@@ -754,8 +762,10 @@ public class RunMainController extends MainLayoutBasicController implements Gene
 			}
 		} else if (source == disclaimerController) {
 			if (event == Event.DONE_EVENT) {
+				disclaimerAccepted = true;
 				coursemain.put("coursemain", columnLayoutCtr.getInitialComponent());
 				coursemain.setDirty(true);
+				fireEvent(ureq, COURSE_DISCLAIMER_ACCEPTED);
 			}
 		}
 	}
diff --git a/src/main/java/org/olat/ims/qti21/manager/AssessmentTestSessionDAO.java b/src/main/java/org/olat/ims/qti21/manager/AssessmentTestSessionDAO.java
index 6dc6cf4f41dec7d84556f2092f9bf90b850e6d70..f3aab857695d9885d84827223e41df27065c5c62 100644
--- a/src/main/java/org/olat/ims/qti21/manager/AssessmentTestSessionDAO.java
+++ b/src/main/java/org/olat/ims/qti21/manager/AssessmentTestSessionDAO.java
@@ -785,7 +785,7 @@ public class AssessmentTestSessionDAO {
 		
 		decorateTestSessionPermission(sb, searchParams);
 		//need to be anonymized
-		sb.append(" order by usr.lastName, testSession.key");
+		sb.append(" order by usr.lastName, usr.firstName, testSession.key");
 		
 		TypedQuery<AssessmentTestSession> query = dbInstance.getCurrentEntityManager()
 				.createQuery(sb.toString(), AssessmentTestSession.class);
diff --git a/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java b/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java
index e0d36e83ee553e6ab547dba8e593129bec7c774a..7fd5d9e0fc52c2d9e9b692a4185c69056c44cd14 100644
--- a/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java
+++ b/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java
@@ -707,7 +707,7 @@ public class AssessmentTestDisplayController extends BasicController implements
 	private boolean timeLimitBarrier(UserRequest ureq) {
 		Long assessmentTestMaxTimeLimits = getAssessmentTestMaxTimeLimit();
 		if(assessmentTestMaxTimeLimits != null) {
-			long maximumAssessmentTestDuration = assessmentTestMaxTimeLimits.longValue() * 1000;//convert in milliseconds
+			long maximumAssessmentTestDuration = assessmentTestMaxTimeLimits.longValue() * 1000l;//convert in milliseconds
 			TestSessionState testSessionState = testSessionController.getTestSessionState();
 			if(!testSessionState.isEnded() && !testSessionState.isExited()) {
 				long durationMillis = testSessionState.getDurationAccumulated();
@@ -768,10 +768,7 @@ public class AssessmentTestDisplayController extends BasicController implements
 		if(overrideOptions != null && overrideOptions.getEndTestDate() != null) {
 			Date endTestDate = overrideOptions.getEndTestDate();
 			long diff = endTestDate.getTime() - testSessionController.getCurrentRequestTimestamp().getTime();
-			if(diff < 0l) {
-				diff = 0l;
-			}
-			return diff;
+			return Long.valueOf(diff);
 		}
 		return null;// default is a year
 	}
@@ -1588,6 +1585,9 @@ public class AssessmentTestDisplayController extends BasicController implements
         }
         
         boolean terminated = isTerminated();
+        
+        /* Enter first assessment item if possible */
+        enterFirstAssessmentItemOfAdvancedTestPart(nextTestPart, ureq);
 
         /* Record current result state */
         computeAndRecordTestAssessmentResult(currentTimestamp, testSessionState, terminated);
@@ -1604,6 +1604,22 @@ public class AssessmentTestDisplayController extends BasicController implements
         }
 	}
 	
+	private void enterFirstAssessmentItemOfAdvancedTestPart(TestPlanNode nextTestPart, UserRequest ureq) {
+		try {
+			if(!isTerminated() && nextTestPart != null) {
+			    TestPart currentTestPart = testSessionController.getCurrentTestPart();
+				if(currentTestPart != null && currentTestPart.getNavigationMode() == NavigationMode.NONLINEAR) {
+			    	//go to the first assessment item
+			    	if(testSessionController.hasFollowingNonLinearItem()) {
+			    		testSessionController.selectFollowingItemNonLinear(ureq.getRequestTimestamp());
+			    	}
+				}
+			}
+		} catch (Exception e) {
+			logError("", e);
+		}
+	}
+	
 	private void processReviewTestPart() {
 		NotificationRecorder notificationRecorder = new NotificationRecorder(NotificationLevel.INFO);
         TestSessionState testSessionState = testSessionController.getTestSessionState();
diff --git a/src/main/java/org/olat/login/oauth/spi/OpenIdConnectApi.java b/src/main/java/org/olat/login/oauth/spi/OpenIdConnectApi.java
index bdb0bd5457e82d2d18c0493009ba761b71219601..77017acbaba527296e4bdbd9f0526268bfa67c8a 100644
--- a/src/main/java/org/olat/login/oauth/spi/OpenIdConnectApi.java
+++ b/src/main/java/org/olat/login/oauth/spi/OpenIdConnectApi.java
@@ -103,21 +103,21 @@ public class OpenIdConnectApi extends DefaultApi20 {
 				if(!oauthModule.getOpenIdConnectIFIssuer().equals(idJson.get("iss"))
 						|| !oauthModule.getOpenIdConnectIFIssuer().equals(accessJson.get("iss"))) {
 					allOk &= false;
-					log.error("iss don't match issuer");
+					log.info("iss don't match issuer");
 				}
 				
 				if(!getApiKey().equals(idJson.get("aud"))) {
 					allOk &= false;
-					log.error("aud don't match application key");
+					log.info("aud don't match application key");
 				}
 				if(!oVerifier.getState().equals(oVerifier.getSessionState())) {
 					allOk &= false;
-					log.error("state doesn't match session state");
+					log.info("state doesn't match session state");
 				}
 				
 				if(!oVerifier.getSessionNonce().equals(idJson.get("nonce"))) {
 					allOk &= false;
-					log.error("session nonce don't match verifier nonce");
+					log.info("session nonce don't match verifier nonce");
 				}
 				
 				return allOk ? new OAuth2AccessToken(idToken, oVerifier.getState()) : null;
diff --git a/src/main/java/org/olat/login/oauth/spi/OpenIdConnectFullConfigurableApi.java b/src/main/java/org/olat/login/oauth/spi/OpenIdConnectFullConfigurableApi.java
index c20e6cb592d1432b0c315d345c6d71e3b0e86e02..37bd356eeb49e84958d94506fd2d9f37c50bba94 100644
--- a/src/main/java/org/olat/login/oauth/spi/OpenIdConnectFullConfigurableApi.java
+++ b/src/main/java/org/olat/login/oauth/spi/OpenIdConnectFullConfigurableApi.java
@@ -100,22 +100,22 @@ public class OpenIdConnectFullConfigurableApi extends DefaultApi20 {
 				if(!provider.getIssuer().equals(idJson.get("iss"))
 						|| !provider.getIssuer().equals(accessJson.get("iss"))) {
 					allOk &= false;
-					log.error("iss don't match issuer");
+					log.info("iss don't match issuer");
 				}
 				
 				if(!provider.getAppKey().equals(idJson.get("aud"))) {
 					allOk &= false;
-					log.error("aud don't match application key");
+					log.info("aud don't match application key");
 				}
 
 				if(!oVerifier.getState().equals(oVerifier.getSessionState())) {
 					allOk &= false;
-					log.error("state doesn't match session state");
+					log.info("state doesn't match session state");
 				}
 				
 				if(!oVerifier.getSessionNonce().equals(idJson.get("nonce"))) {
 					allOk &= false;
-					log.error("session nonce don't match verifier nonce");
+					log.info("session nonce don't match verifier nonce");
 				}
 				return allOk ? new OAuth2AccessToken(idToken, oVerifier.getState()) : null;
 			} catch (JSONException e) {
diff --git a/src/main/java/org/olat/user/ui/UserDisplayNameCellRenderer.java b/src/main/java/org/olat/user/ui/UserDisplayNameCellRenderer.java
new file mode 100644
index 0000000000000000000000000000000000000000..add40df57307a5a66bf7676364da977d0c631d1c
--- /dev/null
+++ b/src/main/java/org/olat/user/ui/UserDisplayNameCellRenderer.java
@@ -0,0 +1,65 @@
+/**
+ * <a href="http://www.openolat.org">
+ * OpenOLAT - Online Learning and Training</a><br>
+ * <p>
+ * Licensed under the Apache License, Version 2.0 (the "License"); <br>
+ * you may not use this file except in compliance with the License.<br>
+ * You may obtain a copy of the License at the
+ * <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a>
+ * <p>
+ * Unless required by applicable law or agreed to in writing,<br>
+ * software distributed under the License is distributed on an "AS IS" BASIS, <br>
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br>
+ * See the License for the specific language governing permissions and <br>
+ * limitations under the License.
+ * <p>
+ * Initial code contributed and copyrighted by<br>
+ * frentix GmbH, http://www.frentix.com
+ * <p>
+ */
+package org.olat.user.ui;
+
+import org.olat.core.CoreSpringFactory;
+import org.olat.core.gui.components.form.flexible.impl.elements.table.FlexiCellRenderer;
+import org.olat.core.gui.components.form.flexible.impl.elements.table.FlexiTableComponent;
+import org.olat.core.gui.render.Renderer;
+import org.olat.core.gui.render.StringOutput;
+import org.olat.core.gui.render.URLBuilder;
+import org.olat.core.gui.translator.Translator;
+import org.olat.core.id.Identity;
+import org.olat.core.id.User;
+import org.olat.user.UserManager;
+
+/**
+ * 
+ * Initial date: 19 Feb 2021<br>
+ * @author uhensler, urs.hensler@frentix.com, http://www.frentix.com
+ *
+ */
+public class UserDisplayNameCellRenderer implements FlexiCellRenderer {
+	
+	private static final UserDisplayNameCellRenderer INSTANCE = new UserDisplayNameCellRenderer();
+	
+	private final UserManager userManager;
+	
+	public static UserDisplayNameCellRenderer get() {
+		return INSTANCE;
+	}
+	
+	private UserDisplayNameCellRenderer() {
+		this.userManager = CoreSpringFactory.getImpl(UserManager.class);
+	}
+
+	@Override
+	public void render(Renderer renderer, StringOutput target, Object cellValue, int row, FlexiTableComponent source,
+			URLBuilder ubu, Translator translator) {
+		if (cellValue instanceof Identity) {
+			String userDisplayName = userManager.getUserDisplayName(((Identity)cellValue).getKey());
+			target.append(userDisplayName);
+		} else if (cellValue instanceof User) {
+			String userDisplayName = userManager.getUserDisplayName(((User)cellValue));
+			target.append(userDisplayName);
+		}
+	}
+
+}
diff --git a/src/main/resources/infinispan-config.xml b/src/main/resources/infinispan-config.xml
index 004bc3b9d9c1531918b63b01b4b0f748c14e073a..f35866c3d20a3a8b125036da6e9b782f3f23b0cf 100644
--- a/src/main/resources/infinispan-config.xml
+++ b/src/main/resources/infinispan-config.xml
@@ -92,6 +92,14 @@
 			<!-- max-idle same as session time out for WebDAV session in UserSessionManager -->
 		</local-cache>
 		
+		<local-cache name="WebDAVManager@resources" simple-cache="true" statistics="true" statistics-available="true">
+			<locking isolation="READ_COMMITTED" concurrency-level="1000" acquire-timeout="15000" striping="false" />
+			<transaction mode="NONE" auto-commit="true" />
+			<memory max-count="10000" when-full="REMOVE" />
+			<expiration max-idle="900000" interval="5000" />
+			<!-- max-idle same as session time out for WebDAV session in UserSessionManager -->
+		</local-cache>
+		
 		<local-cache name="UserManager@username" simple-cache="true" statistics="true" statistics-available="true">
 			<locking isolation="READ_COMMITTED" concurrency-level="1000" acquire-timeout="15000" striping="false" />
 			<transaction mode="NONE" auto-commit="true" />