Commit 3acba09c authored by Gaston Dombiak's avatar Gaston Dombiak Committed by gaston

Added ability to manually lock/unlock a room. JM-46


git-svn-id: http://svn.igniterealtime.org/svn/repos/messenger/trunk@723 b35dd754-fafc-0310-a699-88a17e54d16e
parent 76ec6476
...@@ -381,6 +381,15 @@ public interface MUCRoom extends ChatDeliverer { ...@@ -381,6 +381,15 @@ public interface MUCRoom extends ChatDeliverer {
*/ */
public boolean isLocked(); public boolean isLocked();
/**
* Returns true if the room is locked and it was locked by a room owner after the room was
* initially configured.
*
* @return true if the room is locked and it was locked by a room owner after the room was
* initially configured.
*/
public boolean isManuallyLocked();
/** /**
* An event callback fired whenever an occupant changes his nickname within the chatroom. * An event callback fired whenever an occupant changes his nickname within the chatroom.
* *
...@@ -767,14 +776,24 @@ public interface MUCRoom extends ChatDeliverer { ...@@ -767,14 +776,24 @@ public interface MUCRoom extends ChatDeliverer {
*/ */
public boolean canBroadcastPresence(String roleToBroadcast); public boolean canBroadcastPresence(String roleToBroadcast);
/**
* Locks the room so that users cannot join the room. Only the owner of the room can lock/unlock
* the room.
*
* @param senderRole the role of the occupant that locked the room.
* @throws ForbiddenException If the user is not an owner of the room.
*/
public void lock(MUCRole senderRole) throws ForbiddenException;
/** /**
* Unlocks the room so that users can join the room. The room is locked when created and only * Unlocks the room so that users can join the room. The room is locked when created and only
* the owner of the room can unlock it by sending the configuration form to the Multi-User Chat * the owner of the room can unlock it by sending the configuration form to the Multi-User Chat
* service. * service.
* *
* @param senderRole the role of the occupant that unlocked the room. * @param senderRole the role of the occupant that unlocked the room.
* @throws ForbiddenException If the user is not an owner of the room.
*/ */
public void unlockRoom(MUCRole senderRole); public void unlock(MUCRole senderRole) throws ForbiddenException;
/** /**
* Sends an invitation to a user. The invitation will be sent as if the room is inviting the * Sends an invitation to a user. The invitation will be sent as if the room is inviting the
......
...@@ -309,8 +309,8 @@ public class IQOwnerHandler { ...@@ -309,8 +309,8 @@ public class IQOwnerHandler {
} }
// If the room was locked, unlock it and send to the owner the "room is now unlocked" // If the room was locked, unlock it and send to the owner the "room is now unlocked"
// message // message
if (room.isLocked()) { if (room.isLocked() && !room.isManuallyLocked()) {
room.unlockRoom(senderRole); room.unlock(senderRole);
} }
} }
} }
......
...@@ -36,14 +36,6 @@ import org.dom4j.Element; ...@@ -36,14 +36,6 @@ import org.dom4j.Element;
*/ */
public class MUCRoomImpl implements MUCRoom { public class MUCRoomImpl implements MUCRoom {
/**
* The timeout period to unlock a room. If the period expired the default means that the default
* configuration was accepted for the room.
*/
// TODO Set this variable from a default configuration. Add setters and getters.
// Default value is 30 min ( 30(min) * 60(sec) * 1000(mill) )
public static long LOCK_TIMEOUT = 1800000;
/** /**
* The server hosting the room. * The server hosting the room.
*/ */
...@@ -108,14 +100,9 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -108,14 +100,9 @@ public class MUCRoomImpl implements MUCRoom {
private MUCRoomHistory roomHistory; private MUCRoomHistory roomHistory;
/** /**
* Flag that indicates whether a room is locked or not. * Time when the room was locked. A value of zero means that the room is unlocked.
*/ */
private boolean roomLocked; private long lockedTime;
/**
* The time when the room was locked.
*/
long lockedTime;
/** /**
* List of chatroom's owner. The list contains only bare jid. * List of chatroom's owner. The list contains only bare jid.
...@@ -273,7 +260,6 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -273,7 +260,6 @@ public class MUCRoomImpl implements MUCRoom {
this.iqOwnerHandler = new IQOwnerHandler(this, packetRouter); this.iqOwnerHandler = new IQOwnerHandler(this, packetRouter);
this.iqAdminHandler = new IQAdminHandler(this, packetRouter); this.iqAdminHandler = new IQAdminHandler(this, packetRouter);
// No one can join the room except the room's owner // No one can join the room except the room's owner
this.roomLocked = true;
this.lockedTime = startTime; this.lockedTime = startTime;
// Set the default roles for which presence is broadcast // Set the default roles for which presence is broadcast
rolesToBroadcastPresence.add("moderator"); rolesToBroadcastPresence.add("moderator");
...@@ -394,7 +380,7 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -394,7 +380,7 @@ public class MUCRoomImpl implements MUCRoom {
} }
boolean isOwner = owners.contains(user.getAddress().toBareJID()); boolean isOwner = owners.contains(user.getAddress().toBareJID());
// If the room is locked and this user is not an owner raise a RoomLocked exception // If the room is locked and this user is not an owner raise a RoomLocked exception
if (roomLocked) { if (isLocked()) {
if (!isOwner) { if (!isOwner) {
throw new RoomLockedException(); throw new RoomLockedException();
} }
...@@ -500,8 +486,8 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -500,8 +486,8 @@ public class MUCRoomImpl implements MUCRoom {
} }
if (joinRole != null) { if (joinRole != null) {
// It is assumed that the room is new based on the fact that it's locked and // It is assumed that the room is new based on the fact that it's locked and
// it has only one occupants (the owner). // that it was locked when it was created.
boolean isRoomNew = roomLocked && occupants.size() == 1; boolean isRoomNew = isLocked() && creationDate.getTime() == lockedTime;
try { try {
// Send the presence of this new occupant to existing occupants // Send the presence of this new occupant to existing occupants
Presence joinPresence = joinRole.getPresence().createCopy(); Presence joinPresence = joinRole.getPresence().createCopy();
...@@ -512,7 +498,6 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -512,7 +498,6 @@ public class MUCRoomImpl implements MUCRoom {
} }
joinPresence.setFrom(joinRole.getRoleAddress()); joinPresence.setFrom(joinRole.getRoleAddress());
broadcastPresence(joinPresence); broadcastPresence(joinPresence);
} }
catch (Exception e) { catch (Exception e) {
Log.error(LocaleUtils.getLocalizedString("admin.error"), e); Log.error(LocaleUtils.getLocalizedString("admin.error"), e);
...@@ -520,6 +505,15 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -520,6 +505,15 @@ public class MUCRoomImpl implements MUCRoom {
// If the room has just been created send the "room locked until configuration is // If the room has just been created send the "room locked until configuration is
// confirmed" message // confirmed" message
if (isRoomNew) { if (isRoomNew) {
Message message = new Message();
message.setType(Message.Type.groupchat);
message.setBody(LocaleUtils.getLocalizedString("muc.new"));
message.setFrom(role.getRoleAddress());
message.setTo(user.getAddress());
router.route(message);
}
else if (isLocked()) {
// Warn the owner that the room is locked but it's not new
Message message = new Message(); Message message = new Message();
message.setType(Message.Type.groupchat); message.setType(Message.Type.groupchat);
message.setBody(LocaleUtils.getLocalizedString("muc.locked")); message.setBody(LocaleUtils.getLocalizedString("muc.locked"));
...@@ -1231,11 +1225,11 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -1231,11 +1225,11 @@ public class MUCRoomImpl implements MUCRoom {
} }
public boolean isLocked() { public boolean isLocked() {
if (System.currentTimeMillis() - startTime > LOCK_TIMEOUT) { return lockedTime > 0;
// Unlock the room. The default configuration is assumed to be accepted by the owner.
roomLocked = false;
} }
return roomLocked;
public boolean isManuallyLocked() {
return lockedTime > 0 && creationDate.getTime() != lockedTime;
} }
public void nicknameChanged(String oldNick, String newNick) { public void nicknameChanged(String oldNick, String newNick) {
...@@ -1254,7 +1248,7 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -1254,7 +1248,7 @@ public class MUCRoomImpl implements MUCRoom {
} }
// Set the new subject to the room // Set the new subject to the room
subject = packet.getSubject(); subject = packet.getSubject();
MUCPersistenceManager.updateRoomSubject(this, subject); MUCPersistenceManager.updateRoomSubject(this);
// Notify all the occupants that the subject has changed // Notify all the occupants that the subject has changed
packet.setFrom(role.getRoleAddress()); packet.setFrom(role.getRoleAddress());
send(packet); send(packet);
...@@ -1578,11 +1572,6 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -1578,11 +1572,6 @@ public class MUCRoomImpl implements MUCRoom {
public void setSavedToDB(boolean saved) { public void setSavedToDB(boolean saved) {
this.savedToDB = saved; this.savedToDB = saved;
if (saved) {
// Unlock the room now. This is necessary only when a persistent room has been
// retrieved from the database.
this.roomLocked = false;
}
} }
public void setPersistent(boolean persistent) { public void setPersistent(boolean persistent) {
...@@ -1621,9 +1610,35 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -1621,9 +1610,35 @@ public class MUCRoomImpl implements MUCRoom {
return "none".equals(roleToBroadcast) || rolesToBroadcastPresence.contains(roleToBroadcast); return "none".equals(roleToBroadcast) || rolesToBroadcastPresence.contains(roleToBroadcast);
} }
public void unlockRoom(MUCRole senderRole) { public void lock(MUCRole senderRole) throws ForbiddenException {
roomLocked = false; if (MUCRole.OWNER != senderRole.getAffiliation()) {
this.lockedTime = 0; throw new ForbiddenException();
}
if (isLocked()) {
// Do nothing if the room was already locked
return;
}
setLocked(true);
if (senderRole.getChatUser() != null) {
// Send to the occupant that locked the room a message saying so
Message message = new Message();
message.setType(Message.Type.groupchat);
message.setBody(LocaleUtils.getLocalizedString("muc.locked"));
message.setFrom(getRole().getRoleAddress());
message.setTo(senderRole.getChatUser().getAddress());
router.route(message);
}
}
public void unlock(MUCRole senderRole) throws ForbiddenException {
if (MUCRole.OWNER != senderRole.getAffiliation()) {
throw new ForbiddenException();
}
if (!isLocked()) {
// Do nothing if the room was already unlocked
return;
}
setLocked(false);
if (senderRole.getChatUser() != null) { if (senderRole.getChatUser() != null) {
// Send to the occupant that unlocked the room a message saying so // Send to the occupant that unlocked the room a message saying so
Message message = new Message(); Message message = new Message();
...@@ -1635,6 +1650,40 @@ public class MUCRoomImpl implements MUCRoom { ...@@ -1635,6 +1650,40 @@ public class MUCRoomImpl implements MUCRoom {
} }
} }
private void setLocked(boolean locked) {
if (locked) {
this.lockedTime = System.currentTimeMillis();
}
else {
this.lockedTime = 0;
}
MUCPersistenceManager.updateRoomLock(this);
}
/**
* Sets the date when the room was locked. Initially when the room is created it is locked so
* the locked date is the creation date of the room. Afterwards, the room may be manually
* locked and unlocked so the locked date may be in these cases different than the creation
* date. A Date with time 0 means that the the room is unlocked.
*
* @param lockedTime the date when the room was locked.
*/
void setLockedDate(Date lockedTime) {
this.lockedTime = lockedTime.getTime();
}
/**
* Returns the date when the room was locked. Initially when the room is created it is locked so
* the locked date is the creation date of the room. Afterwards, the room may be manually
* locked and unlocked so the locked date may be in these cases different than the creation
* date. When the room is unlocked a Date with time 0 is returned.
*
* @return the date when the room was locked.
*/
Date getLockedDate() {
return new Date(lockedTime);
}
public List<Presence> addAdmins(List<String> newAdmins, MUCRole senderRole) public List<Presence> addAdmins(List<String> newAdmins, MUCRole senderRole)
throws ForbiddenException, ConflictException { throws ForbiddenException, ConflictException {
List<Presence> answer = new ArrayList<Presence>(newAdmins.size()); List<Presence> answer = new ArrayList<Presence>(newAdmins.size());
......
...@@ -356,8 +356,8 @@ public class MUCUserImpl implements MUCUser { ...@@ -356,8 +356,8 @@ public class MUCUserImpl implements MUCUser {
roles.put(group, role); roles.put(group, role);
// If the client that created the room is non-MUC compliant then // If the client that created the room is non-MUC compliant then
// unlock the room thus creating an "instant" room // unlock the room thus creating an "instant" room
if (room.isLocked() && mucInfo == null) { if (mucInfo == null && room.isLocked() && !room.isManuallyLocked()) {
room.unlockRoom(role); room.unlock(role);
} }
} }
catch (UnauthorizedException e) { catch (UnauthorizedException e) {
......
...@@ -276,7 +276,7 @@ public class MultiUserChatServerImpl extends BasicModule implements MultiUserCha ...@@ -276,7 +276,7 @@ public class MultiUserChatServerImpl extends BasicModule implements MultiUserCha
try { try {
// Try to load the room's configuration from the database (if the room is // Try to load the room's configuration from the database (if the room is
// persistent but was added to the DB after the server was started up) // persistent but was added to the DB after the server was started up)
MUCPersistenceManager.loadFromDB(room); MUCPersistenceManager.loadFromDB((MUCRoomImpl) room);
} }
catch (IllegalArgumentException e) { catch (IllegalArgumentException e) {
// The room does not exist so check for creation permissions // The room does not exist so check for creation permissions
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment