mirror of
https://github.com/vanitasvitae/Smack.git
synced 2024-11-23 04:22:05 +01:00
Improve privacy parsing and API. Add NumberUtil
Make 'order' an long Parse fall-through case's child elements (message, iq, presence-in, presence-out) Remove privacy.addExtension(new DefaultPacketExtension(parser.getName(), parser.getNamespace())); at the beginning of PrivacyProvider. Was there since day one for an unknown reason.
This commit is contained in:
parent
142f78c135
commit
c5db012fc8
6 changed files with 145 additions and 51 deletions
|
@ -0,0 +1,34 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* Copyright © 2015 Florian Schmaus
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.jivesoftware.smack.util;
|
||||||
|
|
||||||
|
public class NumberUtil {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the given long is within the range of an unsigned 32-bit integer, the XML type "xs:unsignedInt".
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
*/
|
||||||
|
public static void checkIfInUInt32Range(long value) {
|
||||||
|
if (value < 0) {
|
||||||
|
throw new IllegalArgumentException("unsigned 32-bit integers can't be negative");
|
||||||
|
}
|
||||||
|
if (value > ((1L << 32) - 1)) {
|
||||||
|
throw new IllegalArgumentException("unsigned 32-bit integers can't be greater then 2^32 - 1");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,6 +16,8 @@
|
||||||
*/
|
*/
|
||||||
package org.jivesoftware.smackx.privacy.packet;
|
package org.jivesoftware.smackx.privacy.packet;
|
||||||
|
|
||||||
|
import org.jivesoftware.smack.util.NumberUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A privacy item acts a rule that when matched defines if a packet should be blocked or not.
|
* A privacy item acts a rule that when matched defines if a packet should be blocked or not.
|
||||||
*
|
*
|
||||||
|
@ -40,8 +42,11 @@ public class PrivacyItem {
|
||||||
|
|
||||||
/** allow is the action associated with the item, it can allow or deny the communication. */
|
/** allow is the action associated with the item, it can allow or deny the communication. */
|
||||||
private final boolean allow;
|
private final boolean allow;
|
||||||
/** order is a non-negative integer that is unique among all items in the list. */
|
|
||||||
private final int order;
|
/**
|
||||||
|
* order is a unsigned 32-bit integer that is unique among all items in the list.
|
||||||
|
**/
|
||||||
|
private final long order;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Type defines if the rule is based on JIDs, roster groups or presence subscription types.
|
* Type defines if the rule is based on JIDs, roster groups or presence subscription types.
|
||||||
|
@ -75,7 +80,7 @@ public class PrivacyItem {
|
||||||
* @param allow true if this is an allow item
|
* @param allow true if this is an allow item
|
||||||
* @param order the order of this privacy item
|
* @param order the order of this privacy item
|
||||||
*/
|
*/
|
||||||
public PrivacyItem(boolean allow, int order) {
|
public PrivacyItem(boolean allow, long order) {
|
||||||
this(null, null, allow, order);
|
this(null, null, allow, order);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,7 +98,8 @@ public class PrivacyItem {
|
||||||
* @param allow true if this is an allow item
|
* @param allow true if this is an allow item
|
||||||
* @param order the order of this privacy item
|
* @param order the order of this privacy item
|
||||||
*/
|
*/
|
||||||
public PrivacyItem(Type type, String value, boolean allow, int order) {
|
public PrivacyItem(Type type, String value, boolean allow, long order) {
|
||||||
|
NumberUtil.checkIfInUInt32Range(order);
|
||||||
this.type = type;
|
this.type = type;
|
||||||
this.value = value;
|
this.value = value;
|
||||||
this.allow = allow;
|
this.allow = allow;
|
||||||
|
@ -191,7 +197,7 @@ public class PrivacyItem {
|
||||||
*
|
*
|
||||||
* @return the order number.
|
* @return the order number.
|
||||||
*/
|
*/
|
||||||
public int getOrder() {
|
public long getOrder() {
|
||||||
return order;
|
return order;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,8 +16,9 @@
|
||||||
*/
|
*/
|
||||||
package org.jivesoftware.smackx.privacy.provider;
|
package org.jivesoftware.smackx.privacy.provider;
|
||||||
|
|
||||||
import org.jivesoftware.smack.packet.DefaultPacketExtension;
|
import org.jivesoftware.smack.SmackException;
|
||||||
import org.jivesoftware.smack.provider.IQProvider;
|
import org.jivesoftware.smack.provider.IQProvider;
|
||||||
|
import org.jivesoftware.smack.util.ParserUtils;
|
||||||
import org.jivesoftware.smackx.privacy.packet.Privacy;
|
import org.jivesoftware.smackx.privacy.packet.Privacy;
|
||||||
import org.jivesoftware.smackx.privacy.packet.PrivacyItem;
|
import org.jivesoftware.smackx.privacy.packet.PrivacyItem;
|
||||||
import org.xmlpull.v1.XmlPullParser;
|
import org.xmlpull.v1.XmlPullParser;
|
||||||
|
@ -38,11 +39,8 @@ public class PrivacyProvider extends IQProvider<Privacy> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Privacy parse(XmlPullParser parser, int initialDepth)
|
public Privacy parse(XmlPullParser parser, int initialDepth)
|
||||||
throws XmlPullParserException, IOException {
|
throws XmlPullParserException, IOException, SmackException {
|
||||||
Privacy privacy = new Privacy();
|
Privacy privacy = new Privacy();
|
||||||
/* privacy.addExtension(PacketParserUtils.parsePacketExtension(parser
|
|
||||||
.getName(), parser.getNamespace(), parser)); */
|
|
||||||
privacy.addExtension(new DefaultPacketExtension(parser.getName(), parser.getNamespace()));
|
|
||||||
boolean done = false;
|
boolean done = false;
|
||||||
while (!done) {
|
while (!done) {
|
||||||
int eventType = parser.next();
|
int eventType = parser.next();
|
||||||
|
@ -78,7 +76,7 @@ public class PrivacyProvider extends IQProvider<Privacy> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse the list complex type
|
// Parse the list complex type
|
||||||
public void parseList(XmlPullParser parser, Privacy privacy) throws XmlPullParserException, IOException {
|
private static void parseList(XmlPullParser parser, Privacy privacy) throws XmlPullParserException, IOException, SmackException {
|
||||||
boolean done = false;
|
boolean done = false;
|
||||||
String listName = parser.getAttributeValue("", "name");
|
String listName = parser.getAttributeValue("", "name");
|
||||||
ArrayList<PrivacyItem> items = new ArrayList<PrivacyItem>();
|
ArrayList<PrivacyItem> items = new ArrayList<PrivacyItem>();
|
||||||
|
@ -100,59 +98,73 @@ public class PrivacyProvider extends IQProvider<Privacy> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse the list complex type
|
// Parse the list complex type
|
||||||
public PrivacyItem parseItem(XmlPullParser parser) throws XmlPullParserException, IOException {
|
private static PrivacyItem parseItem(XmlPullParser parser) throws XmlPullParserException, IOException, SmackException {
|
||||||
boolean done = false;
|
|
||||||
// Retrieves the required attributes
|
// Retrieves the required attributes
|
||||||
String actionValue = parser.getAttributeValue("", "action");
|
String actionValue = parser.getAttributeValue("", "action");
|
||||||
String orderValue = parser.getAttributeValue("", "order");
|
// Set the order number, this attribute is required
|
||||||
|
long order = ParserUtils.getLongAttribute(parser, "order");
|
||||||
|
|
||||||
|
// If type is not set, then it's the fall-through case
|
||||||
String type = parser.getAttributeValue("", "type");
|
String type = parser.getAttributeValue("", "type");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* According the action value it sets the allow status. The fall-through action is assumed
|
* According the action value it sets the allow status. The fall-through action is assumed
|
||||||
* to be "allow"
|
* to be "allow"
|
||||||
*/
|
*/
|
||||||
boolean allow = true;
|
boolean allow;
|
||||||
if ("allow".equalsIgnoreCase(actionValue)) {
|
switch (actionValue) {
|
||||||
allow = true;
|
case "allow":
|
||||||
} else if ("deny".equalsIgnoreCase(actionValue)) {
|
allow = true;
|
||||||
allow = false;
|
break;
|
||||||
|
case "deny":
|
||||||
|
allow = false;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new SmackException("Unkown action value '" + actionValue + "'");
|
||||||
}
|
}
|
||||||
// Set the order number
|
|
||||||
int order = Integer.parseInt(orderValue);
|
|
||||||
|
|
||||||
PrivacyItem item;
|
PrivacyItem item;
|
||||||
if (type != null) {
|
if (type != null) {
|
||||||
// If the type is not null, then we are dealing with a standard privacy item
|
// If the type is not null, then we are dealing with a standard privacy item
|
||||||
String value = parser.getAttributeValue("", "value");
|
String value = parser.getAttributeValue("", "value");
|
||||||
item = new PrivacyItem(PrivacyItem.Type.valueOf(type), value, allow, order);
|
item = new PrivacyItem(PrivacyItem.Type.valueOf(type), value, allow, order);
|
||||||
|
|
||||||
while (!done) {
|
|
||||||
int eventType = parser.next();
|
|
||||||
if (eventType == XmlPullParser.START_TAG) {
|
|
||||||
if (parser.getName().equals("iq")) {
|
|
||||||
item.setFilterIQ(true);
|
|
||||||
}
|
|
||||||
if (parser.getName().equals("message")) {
|
|
||||||
item.setFilterMessage(true);
|
|
||||||
}
|
|
||||||
if (parser.getName().equals("presence-in")) {
|
|
||||||
item.setFilterPresenceIn(true);
|
|
||||||
}
|
|
||||||
if (parser.getName().equals("presence-out")) {
|
|
||||||
item.setFilterPresenceOut(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (eventType == XmlPullParser.END_TAG) {
|
|
||||||
if (parser.getName().equals("item")) {
|
|
||||||
done = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// If the type is null, then we are dealing with the fall-through privacy item.
|
// If the type is null, then we are dealing with the fall-through privacy item.
|
||||||
item = new PrivacyItem(allow, order);
|
item = new PrivacyItem(allow, order);
|
||||||
}
|
}
|
||||||
|
parseItemChildElements(parser, item);
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void parseItemChildElements(XmlPullParser parser, PrivacyItem privacyItem) throws XmlPullParserException, IOException {
|
||||||
|
final int initialDepth = parser.getDepth();
|
||||||
|
|
||||||
|
outerloop: while (true) {
|
||||||
|
int eventType = parser.next();
|
||||||
|
switch (eventType) {
|
||||||
|
case XmlPullParser.START_TAG:
|
||||||
|
String name = parser.getName();
|
||||||
|
switch (name) {
|
||||||
|
case "iq":
|
||||||
|
privacyItem.setFilterIQ(true);
|
||||||
|
break;
|
||||||
|
case "message":
|
||||||
|
privacyItem.setFilterMessage(true);
|
||||||
|
break;
|
||||||
|
case "presence-in":
|
||||||
|
privacyItem.setFilterPresenceIn(true);
|
||||||
|
break;
|
||||||
|
case "presence-out":
|
||||||
|
privacyItem.setFilterPresenceOut(true);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case XmlPullParser.END_TAG:
|
||||||
|
if (parser.getDepth() == initialDepth) {
|
||||||
|
break outerloop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@ package org.jivesoftware.smackx.xdatavalidation.packet;
|
||||||
|
|
||||||
import org.jivesoftware.smack.packet.NamedElement;
|
import org.jivesoftware.smack.packet.NamedElement;
|
||||||
import org.jivesoftware.smack.packet.PacketExtension;
|
import org.jivesoftware.smack.packet.PacketExtension;
|
||||||
|
import org.jivesoftware.smack.util.NumberUtil;
|
||||||
import org.jivesoftware.smack.util.StringUtils;
|
import org.jivesoftware.smack.util.StringUtils;
|
||||||
import org.jivesoftware.smack.util.XmlStringBuilder;
|
import org.jivesoftware.smack.util.XmlStringBuilder;
|
||||||
import org.jivesoftware.smackx.xdata.FormField;
|
import org.jivesoftware.smackx.xdata.FormField;
|
||||||
|
@ -313,18 +314,18 @@ public abstract class ValidateElement implements PacketExtension {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The 'max' attribute specifies the maximum allowable number of selected/entered values. The 'min' attribute
|
* The 'max' attribute specifies the maximum allowable number of selected/entered values. The 'min' attribute
|
||||||
* specifies the minimum allowable number of selected/entered values. Both attributes are optional and must be a
|
* specifies the minimum allowable number of selected/entered values. Both attributes are optional, but at
|
||||||
* positive integer.
|
* least one must bet set, and the value must be within the range of a unsigned 32-bit integer.
|
||||||
*
|
*
|
||||||
* @param min
|
* @param min
|
||||||
* @param max
|
* @param max
|
||||||
*/
|
*/
|
||||||
public ListRange(Long min, Long max) {
|
public ListRange(Long min, Long max) {
|
||||||
if (min != null && min < 0) {
|
if (min != null) {
|
||||||
throw new IllegalArgumentException("min must not be negative");
|
NumberUtil.checkIfInUInt32Range(min);
|
||||||
}
|
}
|
||||||
if (max != null && max < 0) {
|
if (max != null) {
|
||||||
throw new IllegalArgumentException("max must not be negative");
|
NumberUtil.checkIfInUInt32Range(max);
|
||||||
}
|
}
|
||||||
if (max == null && min == null) {
|
if (max == null && min == null) {
|
||||||
throw new IllegalArgumentException("Either min or max must be given");
|
throw new IllegalArgumentException("Either min or max must be given");
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
package org.jivesoftware.smackx.privacy.provider;
|
package org.jivesoftware.smackx.privacy.provider;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
@ -62,4 +63,44 @@ public class PrivacyProviderTest extends InitExtensions {
|
||||||
assertEquals(true, second.isAllow());
|
assertEquals(true, second.isAllow());
|
||||||
assertEquals(2, second.getOrder());
|
assertEquals(2, second.getOrder());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void parsePrivacyListWithFallThroughInclChildElements() throws Exception {
|
||||||
|
// @formatter:off
|
||||||
|
final String xmlPrivacyList =
|
||||||
|
"<iq type='result' id='getlist2' to='romeo@example.net/orchard'>"
|
||||||
|
+ "<query xmlns='jabber:iq:privacy'>"
|
||||||
|
+ "<list name='public'>"
|
||||||
|
+ "<item type='jid'"
|
||||||
|
+ "value='tybalt@example.com'"
|
||||||
|
+ "action='deny'"
|
||||||
|
+ "order='1'/>"
|
||||||
|
+ "<item action='allow' order='2'>"
|
||||||
|
+ "<message/>"
|
||||||
|
+ "<presence-in/>"
|
||||||
|
+ "</item>"
|
||||||
|
+ "</list>"
|
||||||
|
+ "</query>"
|
||||||
|
+ "</iq>";
|
||||||
|
// @formatter:on
|
||||||
|
IQ iqPrivacyList = (IQ) PacketParserUtils.parseStanza(xmlPrivacyList);
|
||||||
|
assertTrue(iqPrivacyList instanceof Privacy);
|
||||||
|
|
||||||
|
Privacy privacyList = (Privacy) iqPrivacyList;
|
||||||
|
List<PrivacyItem> pl = privacyList.getPrivacyList("public");
|
||||||
|
|
||||||
|
PrivacyItem first = pl.get(0);
|
||||||
|
assertEquals(PrivacyItem.Type.jid, first.getType());
|
||||||
|
assertEquals("tybalt@example.com", first.getValue());
|
||||||
|
assertEquals(false, first.isAllow());
|
||||||
|
assertEquals(1, first.getOrder());
|
||||||
|
|
||||||
|
PrivacyItem second = pl.get(1);
|
||||||
|
assertTrue(second.isAllow());
|
||||||
|
assertEquals(2, second.getOrder());
|
||||||
|
assertTrue(second.isFilterMessage());
|
||||||
|
assertTrue(second.isFilterPresenceIn());
|
||||||
|
assertFalse(second.isFilterPresenceOut());
|
||||||
|
assertFalse(second.isFilterIQ());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,7 +54,7 @@ public class DataValidationHelperTest {
|
||||||
fail("No correct check on consistency");
|
fail("No correct check on consistency");
|
||||||
}
|
}
|
||||||
catch (IllegalArgumentException e) {
|
catch (IllegalArgumentException e) {
|
||||||
assertEquals("min must not be negative", e.getMessage());
|
assertEquals("unsigned 32-bit integers can't be negative", e.getMessage());
|
||||||
}
|
}
|
||||||
|
|
||||||
element.setListRange(new ListRange(10L, 100L));
|
element.setListRange(new ListRange(10L, 100L));
|
||||||
|
|
Loading…
Reference in a new issue