Commit d9357dd8 authored by Guus der Kinderen's avatar Guus der Kinderen

OF-1350: Be more flexible when dealing with MUC room passwords

XMPP defines two configuration fields for password-protecting a room:
- one that defines if the room is password protected.
- another one that defines the password.

Openfire, internally, only stores the password, if password-protection is required.

Some clients send only a password, without explicitly defining the password-protection boolean. In such cases, Openfire should be somewhat flexible, and process the configuration form as intended by the user.
parent fc0bee4a
...@@ -28,10 +28,7 @@ import org.jivesoftware.openfire.group.Group; ...@@ -28,10 +28,7 @@ import org.jivesoftware.openfire.group.Group;
import org.jivesoftware.openfire.group.GroupJID; import org.jivesoftware.openfire.group.GroupJID;
import org.jivesoftware.openfire.group.GroupManager; import org.jivesoftware.openfire.group.GroupManager;
import org.jivesoftware.openfire.group.GroupNotFoundException; import org.jivesoftware.openfire.group.GroupNotFoundException;
import org.jivesoftware.openfire.muc.CannotBeInvitedException; import org.jivesoftware.openfire.muc.*;
import org.jivesoftware.openfire.muc.ConflictException;
import org.jivesoftware.openfire.muc.ForbiddenException;
import org.jivesoftware.openfire.muc.MUCRole;
import org.jivesoftware.openfire.muc.cluster.RoomUpdatedEvent; import org.jivesoftware.openfire.muc.cluster.RoomUpdatedEvent;
import org.jivesoftware.util.JiveGlobals; import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.LocaleUtils; import org.jivesoftware.util.LocaleUtils;
...@@ -95,7 +92,8 @@ public class IQOwnerHandler { ...@@ -95,7 +92,8 @@ public class IQOwnerHandler {
* @throws ConflictException If the room was going to lose all of its owners. * @throws ConflictException If the room was going to lose all of its owners.
*/ */
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
public void handleIQ(IQ packet, MUCRole role) throws ForbiddenException, ConflictException, CannotBeInvitedException { public void handleIQ(IQ packet, MUCRole role) throws ForbiddenException, ConflictException, CannotBeInvitedException, NotAcceptableException
{
// Only owners can send packets with the namespace "http://jabber.org/protocol/muc#owner" // Only owners can send packets with the namespace "http://jabber.org/protocol/muc#owner"
if (MUCRole.Affiliation.owner != role.getAffiliation()) { if (MUCRole.Affiliation.owner != role.getAffiliation()) {
throw new ForbiddenException(); throw new ForbiddenException();
...@@ -158,7 +156,7 @@ public class IQOwnerHandler { ...@@ -158,7 +156,7 @@ public class IQOwnerHandler {
* @throws ConflictException If the room was going to lose all of its owners. * @throws ConflictException If the room was going to lose all of its owners.
*/ */
private void handleDataFormElement(MUCRole senderRole, Element formElement) private void handleDataFormElement(MUCRole senderRole, Element formElement)
throws ForbiddenException, ConflictException { throws ForbiddenException, ConflictException, NotAcceptableException {
DataForm completedForm = new DataForm(formElement); DataForm completedForm = new DataForm(formElement);
switch(completedForm.getType()) { switch(completedForm.getType()) {
...@@ -206,7 +204,8 @@ public class IQOwnerHandler { ...@@ -206,7 +204,8 @@ public class IQOwnerHandler {
* @throws ConflictException If the room was going to lose all of its owners. * @throws ConflictException If the room was going to lose all of its owners.
*/ */
private void processConfigurationForm(DataForm completedForm, MUCRole senderRole) private void processConfigurationForm(DataForm completedForm, MUCRole senderRole)
throws ForbiddenException, ConflictException { throws ForbiddenException, ConflictException, NotAcceptableException
{
List<String> values; List<String> values;
String booleanValue; String booleanValue;
FormField field; FormField field;
...@@ -320,23 +319,56 @@ public class IQOwnerHandler { ...@@ -320,23 +319,56 @@ public class IQOwnerHandler {
room.setCanOccupantsInvite(("1".equals(booleanValue))); room.setCanOccupantsInvite(("1".equals(booleanValue)));
} }
boolean passwordProtectionChanged = false;
boolean passwordChanged = false;
boolean updatedIsPasswordProtected = false;
String updatedPassword = null;
field = completedForm.getField("muc#roomconfig_passwordprotectedroom"); field = completedForm.getField("muc#roomconfig_passwordprotectedroom");
if (field != null) { if (field != null)
{
passwordProtectionChanged = true;
final String value = field.getFirstValue(); final String value = field.getFirstValue();
booleanValue = ((value != null ? value : "1")); booleanValue = ( ( value != null ? value : "1" ) );
boolean isPasswordProtected = "1".equals(booleanValue); updatedIsPasswordProtected = "1".equals( booleanValue );
if (isPasswordProtected) { }
// The room is password protected so set the new password
field = completedForm.getField("muc#roomconfig_roomsecret"); field = completedForm.getField("muc#roomconfig_roomsecret");
if (field != null) { if (field != null) {
final String secret = completedForm.getField("muc#roomconfig_roomsecret").getFirstValue(); passwordChanged = true;
room.setPassword(secret); updatedPassword = completedForm.getField("muc#roomconfig_roomsecret").getFirstValue();
} if ( updatedPassword != null && updatedPassword.isEmpty() )
{
updatedPassword = null;
} }
else { }
// The room is not password protected so remove any previous password
room.setPassword(null); if ( passwordProtectionChanged )
{
// The owner signifies that a change in password-protection status is desired.
if ( !updatedIsPasswordProtected )
{
// The owner lifts password protection.
room.setPassword( null );
} }
else if ( updatedPassword == null && room.getPassword() == null )
{
// The owner sets password-protection, but does not provide a password (and the room does not already have a password).
throw new NotAcceptableException( "Room is made password-protected, but is missing a password." );
}
else if ( updatedPassword != null )
{
// The owner sets password-protection and provided a new password.
room.setPassword( updatedPassword );
}
}
else if ( passwordChanged )
{
// The owner did not explicitly signal a password protection change, but did change the password value.
// This implies a change in password protection.
room.setPassword( updatedPassword );
} }
field = completedForm.getField("muc#roomconfig_whois"); field = completedForm.getField("muc#roomconfig_whois");
......
...@@ -407,6 +407,9 @@ public class LocalMUCUser implements MUCUser { ...@@ -407,6 +407,9 @@ public class LocalMUCUser implements MUCUser {
} }
} }
} }
catch (NotAcceptableException e) {
sendErrorPacket(packet, PacketError.Condition.not_acceptable );
}
catch (ForbiddenException e) { catch (ForbiddenException e) {
sendErrorPacket(packet, PacketError.Condition.forbidden); sendErrorPacket(packet, PacketError.Condition.forbidden);
} }
......
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