WIp
parent
fdd6fb928b
commit
cbdd6ebb47
|
@ -47,14 +47,16 @@ public class UnlockOnExpireListener implements HttpSessionListener {
|
||||||
final ServletContext servletContext = event.getSession().getServletContext();
|
final ServletContext servletContext = event.getSession().getServletContext();
|
||||||
final WebApplicationContext wc = WebApplicationContextUtils.getRequiredWebApplicationContext(servletContext);
|
final WebApplicationContext wc = WebApplicationContextUtils.getRequiredWebApplicationContext(servletContext);
|
||||||
final MindmapService mindmapService = (MindmapService) wc.getBean("mindmapService");
|
final MindmapService mindmapService = (MindmapService) wc.getBean("mindmapService");
|
||||||
final LockManager lockManager = mindmapService.getLockManager();
|
|
||||||
|
|
||||||
|
final LockManager lockManager = mindmapService.getLockManager();
|
||||||
final User user = Utils.getUser(false);
|
final User user = Utils.getUser(false);
|
||||||
if (user != null) {
|
if (user != null) {
|
||||||
try {
|
synchronized (mindmapService.getLockManager()) {
|
||||||
lockManager.unlockAll(user);
|
try {
|
||||||
} catch (LockException | AccessDeniedSecurityException e) {
|
lockManager.unlockAll(user);
|
||||||
logger.error(e);
|
} catch (LockException | AccessDeniedSecurityException e) {
|
||||||
|
logger.error(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -122,7 +122,8 @@ public class MindmapController extends BaseController {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Have permissions ?
|
// Have permissions ?
|
||||||
long result = verifyAndUpdateLock(mindmap, user, session, timestamp);
|
final LockManager lockManager = mindmapService.getLockManager();
|
||||||
|
long result = lockManager.verifyAndUpdateLock(mindmap, user, session, timestamp);
|
||||||
|
|
||||||
// Update collaboration properties ...
|
// Update collaboration properties ...
|
||||||
final CollaborationProperties collaborationProperties = mindmap.findCollaborationProperties(user);
|
final CollaborationProperties collaborationProperties = mindmap.findCollaborationProperties(user);
|
||||||
|
@ -580,45 +581,5 @@ public class MindmapController extends BaseController {
|
||||||
mindmapService.updateMindmap(mindmap, false);
|
mindmapService.updateMindmap(mindmap, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
private long verifyAndUpdateLock(@NotNull Mindmap mindmap, @NotNull User user, @Nullable long session, @NotNull long timestamp) throws WiseMappingException {
|
|
||||||
// Could the map be updated ?
|
|
||||||
verifyLock(mindmap, user, session, timestamp);
|
|
||||||
|
|
||||||
// Update timestamp for lock ...
|
|
||||||
final LockManager lockManager = mindmapService.getLockManager();
|
|
||||||
final LockInfo lockInfo = lockManager.updateExpirationTimeout(mindmap, user);
|
|
||||||
return lockInfo.getTimestamp();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void verifyLock(@NotNull Mindmap mindmap, @NotNull User user, long session, long timestamp) throws WiseMappingException {
|
|
||||||
|
|
||||||
// The lock was lost, reclaim as the ownership of it.
|
|
||||||
final LockManager lockManager = mindmapService.getLockManager();
|
|
||||||
final boolean lockLost = lockManager.isLocked(mindmap);
|
|
||||||
if (!lockLost) {
|
|
||||||
lockManager.lock(mindmap, user, session);
|
|
||||||
}
|
|
||||||
|
|
||||||
final LockInfo lockInfo = lockManager.getLockInfo(mindmap);
|
|
||||||
if (lockInfo.getUser().identityEquality(user)) {
|
|
||||||
long savedTimestamp = mindmap.getLastModificationTime().getTimeInMillis();
|
|
||||||
final boolean outdated = savedTimestamp > timestamp;
|
|
||||||
|
|
||||||
if (lockInfo.getSession() == session) {
|
|
||||||
// Timestamp might not be returned to the client. This try to cover this case, ignoring the client timestamp check.
|
|
||||||
final User lastEditor = mindmap.getLastEditor();
|
|
||||||
boolean editedBySameUser = lastEditor == null || user.identityEquality(lastEditor);
|
|
||||||
if (outdated && !editedBySameUser) {
|
|
||||||
throw new SessionExpiredException("Map has been updated by " + (lastEditor.getEmail()) + ",Timestamp:" + timestamp + "," + savedTimestamp + ", User:" + lastEditor.getId() + ":" + user.getId() + ",Mail:'" + lastEditor.getEmail() + "':'" + user.getEmail(), lastEditor);
|
|
||||||
}
|
|
||||||
} else if (outdated) {
|
|
||||||
logger.warn("Sessions:" + session + ":" + lockInfo.getSession() + ",Timestamp: " + timestamp + ": " + savedTimestamp);
|
|
||||||
// @Todo: Temporally disabled to unblock save action. More research needed.
|
|
||||||
// throw new MultipleSessionsOpenException("Sessions:" + session + ":" + lockInfo.getSession() + ",Timestamp: " + timestamp + ": " + savedTimestamp);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new SessionExpiredException("Different Users.", lockInfo.getUser());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@ package com.wisemapping.service;
|
||||||
|
|
||||||
import com.wisemapping.exceptions.AccessDeniedSecurityException;
|
import com.wisemapping.exceptions.AccessDeniedSecurityException;
|
||||||
import com.wisemapping.exceptions.LockException;
|
import com.wisemapping.exceptions.LockException;
|
||||||
|
import com.wisemapping.exceptions.SessionExpiredException;
|
||||||
import com.wisemapping.exceptions.WiseMappingException;
|
import com.wisemapping.exceptions.WiseMappingException;
|
||||||
import com.wisemapping.model.Mindmap;
|
import com.wisemapping.model.Mindmap;
|
||||||
import com.wisemapping.model.User;
|
import com.wisemapping.model.User;
|
||||||
|
@ -38,8 +39,7 @@ public interface LockManager {
|
||||||
|
|
||||||
boolean isLockedBy(@NotNull Mindmap mindmap, @NotNull User collaborator);
|
boolean isLockedBy(@NotNull Mindmap mindmap, @NotNull User collaborator);
|
||||||
|
|
||||||
@NotNull
|
|
||||||
LockInfo lock(@NotNull Mindmap mindmap, @NotNull User user, long session) throws WiseMappingException;
|
|
||||||
|
|
||||||
long generateSession();
|
long generateSession();
|
||||||
|
|
||||||
|
long verifyAndUpdateLock(@NotNull Mindmap mindmap, @NotNull User user, long session, long timestamp) throws LockException, SessionExpiredException;
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,35 +20,21 @@ package com.wisemapping.service;
|
||||||
|
|
||||||
import com.wisemapping.exceptions.AccessDeniedSecurityException;
|
import com.wisemapping.exceptions.AccessDeniedSecurityException;
|
||||||
import com.wisemapping.exceptions.LockException;
|
import com.wisemapping.exceptions.LockException;
|
||||||
import com.wisemapping.exceptions.WiseMappingException;
|
import com.wisemapping.exceptions.SessionExpiredException;
|
||||||
import com.wisemapping.model.CollaborationRole;
|
import com.wisemapping.model.CollaborationRole;
|
||||||
import com.wisemapping.model.Mindmap;
|
import com.wisemapping.model.Mindmap;
|
||||||
import com.wisemapping.model.User;
|
import com.wisemapping.model.User;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
/*
|
|
||||||
* Refresh page should not lost the lock.
|
|
||||||
* En caso que no sea posible grabar por que se perdio el lock, usar mensaje de error para explicar el por que...
|
|
||||||
* Mensaje modal explicando que el mapa esta siendo editado, por eso no es posible edilarlo....
|
|
||||||
* Internacionalizacion de los mensaje ...
|
|
||||||
* Logout limpiar las sessiones ...
|
|
||||||
*
|
|
||||||
* Casos:
|
|
||||||
* - Usuario pierde el lock:
|
|
||||||
* - Y grabo con la misma sessions y el timestap ok.
|
|
||||||
* - Y grabo con la misma session y el timestap esta mal
|
|
||||||
* - Y grabo con distinta sessions
|
|
||||||
* - Usuario pierde el lock, pero intenta grabar camio
|
|
||||||
*/
|
|
||||||
|
|
||||||
class LockManagerImpl implements LockManager {
|
class LockManagerImpl implements LockManager {
|
||||||
public static final int ONE_MINUTE_MILLISECONDS = 1000 * 60;
|
private static final int ONE_MINUTE_MILLISECONDS = 1000 * 60;
|
||||||
final Map<Integer, LockInfo> lockInfoByMapId;
|
private final Map<Integer, LockInfo> lockInfoByMapId;
|
||||||
final static Timer expirationTimer = new Timer();
|
private final static Timer expirationTimer = new Timer();
|
||||||
final private static Logger logger = Logger.getLogger(LockManagerImpl.class);
|
final private static Logger logger = Logger.getLogger(LockManagerImpl.class);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -123,17 +109,12 @@ class LockManagerImpl implements LockManager {
|
||||||
return System.nanoTime();
|
return System.nanoTime();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
@NotNull
|
@NotNull
|
||||||
public LockInfo lock(@NotNull Mindmap mindmap, @NotNull User user, long session) throws WiseMappingException {
|
private LockInfo lock(@NotNull Mindmap mindmap, @NotNull User user, long session) throws LockException {
|
||||||
if (isLocked(mindmap) && !isLockedBy(mindmap, user)) {
|
if (isLocked(mindmap) && !isLockedBy(mindmap, user)) {
|
||||||
throw new LockException("Invalid lock, this should not happen");
|
throw new LockException("Invalid lock, this should not happen");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!mindmap.hasPermissions(user, CollaborationRole.EDITOR)) {
|
|
||||||
throw new AccessDeniedSecurityException(mindmap.getId(), user);
|
|
||||||
}
|
|
||||||
|
|
||||||
LockInfo result = lockInfoByMapId.get(mindmap.getId());
|
LockInfo result = lockInfoByMapId.get(mindmap.getId());
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
// Update timeout only...
|
// Update timeout only...
|
||||||
|
@ -154,17 +135,59 @@ class LockManagerImpl implements LockManager {
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
|
|
||||||
logger.debug("Lock expiration scheduler started. Current locks:" + lockInfoByMapId.keySet());
|
synchronized (this) {
|
||||||
|
logger.debug("Lock expiration scheduler started. Current locks:" + lockInfoByMapId.keySet());
|
||||||
// Search for expired sessions and remove them ....
|
// Search for expired sessions and remove them ....
|
||||||
lockInfoByMapId.
|
lockInfoByMapId.
|
||||||
keySet().
|
keySet().
|
||||||
stream().
|
stream().
|
||||||
filter(mapId -> lockInfoByMapId.get(mapId).isExpired()).
|
filter(mapId -> lockInfoByMapId.get(mapId).isExpired()).
|
||||||
forEach(mapId -> unlock(mapId));
|
forEach(mapId -> unlock(mapId));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}, ONE_MINUTE_MILLISECONDS, ONE_MINUTE_MILLISECONDS);
|
}, ONE_MINUTE_MILLISECONDS, ONE_MINUTE_MILLISECONDS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long verifyAndUpdateLock(@NotNull Mindmap mindmap, @NotNull User user, @Nullable long session, @NotNull long timestamp) throws LockException, SessionExpiredException {
|
||||||
|
synchronized (this) {
|
||||||
|
// Could the map be updated ?
|
||||||
|
verifyLock(mindmap, user, session, timestamp);
|
||||||
|
|
||||||
|
// Update timestamp for lock ...
|
||||||
|
final LockInfo lockInfo = this.updateExpirationTimeout(mindmap, user);
|
||||||
|
return lockInfo.getTimestamp();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyLock(@NotNull Mindmap mindmap, @NotNull User user, long session, long timestamp) throws LockException, SessionExpiredException {
|
||||||
|
|
||||||
|
// The lock was lost, reclaim as the ownership of it.
|
||||||
|
final boolean lockLost = this.isLocked(mindmap);
|
||||||
|
if (!lockLost) {
|
||||||
|
this.lock(mindmap, user, session);
|
||||||
|
}
|
||||||
|
|
||||||
|
final LockInfo lockInfo = this.getLockInfo(mindmap);
|
||||||
|
if (lockInfo.getUser().identityEquality(user)) {
|
||||||
|
long savedTimestamp = mindmap.getLastModificationTime().getTimeInMillis();
|
||||||
|
final boolean outdated = savedTimestamp > timestamp;
|
||||||
|
|
||||||
|
if (lockInfo.getSession() == session) {
|
||||||
|
// Timestamp might not be returned to the client. This try to cover this case, ignoring the client timestamp check.
|
||||||
|
final User lastEditor = mindmap.getLastEditor();
|
||||||
|
boolean editedBySameUser = lastEditor == null || user.identityEquality(lastEditor);
|
||||||
|
if (outdated && !editedBySameUser) {
|
||||||
|
throw new SessionExpiredException("Map has been updated by " + (lastEditor.getEmail()) + ",Timestamp:" + timestamp + "," + savedTimestamp + ", User:" + lastEditor.getId() + ":" + user.getId() + ",Mail:'" + lastEditor.getEmail() + "':'" + user.getEmail(), lastEditor);
|
||||||
|
}
|
||||||
|
} else if (outdated) {
|
||||||
|
logger.warn("Sessions:" + session + ":" + lockInfo.getSession() + ",Timestamp: " + timestamp + ": " + savedTimestamp);
|
||||||
|
// @Todo: Temporally disabled to unblock save action. More research needed.
|
||||||
|
// throw new MultipleSessionsOpenException("Sessions:" + session + ":" + lockInfo.getSession() + ",Timestamp: " + timestamp + ": " + savedTimestamp);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new SessionExpiredException("Different Users.", lockInfo.getUser());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,7 +34,7 @@
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
var mapId = '${mindmap.id}';
|
var mapId = '${mindmap.id}';
|
||||||
var lockTimestamp = '${lockTimestamp}';
|
var lockTimestamp = '${lockTimestamp}';
|
||||||
var lockSession = '${lockSession}';
|
var lockSession = '${lockSession}';
|
||||||
var mindmapLocked = ${mindmapLocked};
|
var mindmapLocked = ${mindmapLocked};
|
||||||
var mindmapLockedMsg = '<spring:message code="MINDMAP_LOCKED" arguments="${lockInfo.user.fullName},${lockInfo.user.email}"/>';
|
var mindmapLockedMsg = '<spring:message code="MINDMAP_LOCKED" arguments="${lockInfo.user.fullName},${lockInfo.user.email}"/>';
|
||||||
var userOptions = ${mindmap.properties};
|
var userOptions = ${mindmap.properties};
|
||||||
|
|
Loading…
Reference in New Issue