mirror of
https://github.com/vanitasvitae/Smack.git
synced 2025-01-10 11:58:06 +01:00
535 lines
20 KiB
Java
535 lines
20 KiB
Java
/**
|
|
*
|
|
* Copyright 2003-2005 Jive Software.
|
|
*
|
|
* All rights reserved. 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.test;
|
|
|
|
import java.io.InputStream;
|
|
import java.net.URL;
|
|
import java.util.ArrayList;
|
|
import java.util.Enumeration;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
|
|
import javax.net.SocketFactory;
|
|
|
|
import junit.framework.TestCase;
|
|
|
|
import org.jivesoftware.smack.ConnectionConfiguration;
|
|
import org.jivesoftware.smack.XMPPConnection;
|
|
import org.jivesoftware.smack.XMPPException;
|
|
import org.jivesoftware.smack.util.ConnectionUtils;
|
|
import org.xmlpull.mxp1.MXParser;
|
|
import org.xmlpull.v1.XmlPullParser;
|
|
|
|
/**
|
|
* Base class for all the test cases which provides a pre-configured execution context. This
|
|
* means that any test case that subclassifies this base class will have access to a pool of
|
|
* connections and to the user of each connection. The maximum number of connections in the pool
|
|
* can be controlled by the message {@link #getMaxConnections()} which every subclass must
|
|
* implement.<p>
|
|
*
|
|
* This base class defines a default execution context (i.e. host, port, chat domain and muc
|
|
* domain) which can be found in the file "config/test-case.xml". However, each subclass could
|
|
* redefine the default configuration by providing its own configuration file (if desired). The
|
|
* name of the configuration file must be of the form <test class name>.xml (e.g. RosterTest.xml).
|
|
* The file must be placed in the folder "config". This folder is where the default configuration
|
|
* file is being held.
|
|
*
|
|
* @author Gaston Dombiak
|
|
*/
|
|
public abstract class SmackTestCase extends TestCase {
|
|
|
|
private String host = "localhost";
|
|
private String serviceName = "localhost";
|
|
private int port = 5222;
|
|
private String usernamePrefix = "user";
|
|
private String passwordPrefix;
|
|
private boolean testAnonymousLogin = false;
|
|
private Map<String, String> accountCreationParameters = new HashMap<String, String>();
|
|
private boolean samePassword;
|
|
private List<Integer> createdUserIdx = new ArrayList<Integer>();
|
|
private boolean compressionEnabled = false;
|
|
|
|
private String[] usernames;
|
|
private String[] passwords;
|
|
|
|
private String chatDomain = "chat";
|
|
private String mucDomain = "conference";
|
|
|
|
private XMPPConnection[] connections = null;
|
|
|
|
/**
|
|
* Constructor for SmackTestCase.
|
|
* @param arg0 arg for SmackTestCase
|
|
*/
|
|
public SmackTestCase(String arg0) {
|
|
super(arg0);
|
|
}
|
|
|
|
/**
|
|
* Returns the maximum number of connections to initialize for this test case. All the
|
|
* initialized connections will be connected to the server using a new test account for
|
|
* each conection.
|
|
*
|
|
* @return the maximum number of connections to initialize for this test case.
|
|
*/
|
|
protected abstract int getMaxConnections();
|
|
|
|
/**
|
|
* Returns a SocketFactory that will be used to create the socket to the XMPP server. By
|
|
* default no SocketFactory is used but subclasses my want to redefine this method.<p>
|
|
*
|
|
* A custom SocketFactory allows fine-grained control of the actual connection to the XMPP
|
|
* server. A typical use for a custom SocketFactory is when connecting through a SOCKS proxy.
|
|
*
|
|
* @return a SocketFactory that will be used to create the socket to the XMPP server.
|
|
*/
|
|
protected SocketFactory getSocketFactory() {
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Returns <code>false</code> if the connections initialized by the test case will be
|
|
* automatically connected to the XMPP server.
|
|
* Returns <code>true</code> if the connections initialized by the test case will
|
|
* NOT be connected to the XMPP server. To connect the connections invoke
|
|
* {@link #connectAndLogin(int)}.
|
|
* <p>
|
|
* Connections are connected by default.
|
|
* Overwrite this method if the test case needs unconnected connections.
|
|
*
|
|
* @return <code>true</code> if connections should NOT be connected automatically,
|
|
* <code>false</code> if connections should be connected automatically.
|
|
*/
|
|
protected boolean createOfflineConnections() {
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Returns the XMPPConnection located at the requested position. Each test case holds a
|
|
* pool of connections which is initialized while setting up the test case. The maximum
|
|
* number of connections is controlled by the message {@link #getMaxConnections()} which
|
|
* every subclass must implement.<p>
|
|
*
|
|
* If the requested position is greater than the connections size then an
|
|
* IllegalArgumentException will be thrown.
|
|
*
|
|
* @param index the position in the pool of the connection to look for.
|
|
* @return the XMPPConnection located at the requested position.
|
|
*/
|
|
protected XMPPConnection getConnection(int index) {
|
|
if (index > getMaxConnections()) {
|
|
throw new IllegalArgumentException("Index out of bounds");
|
|
}
|
|
return connections[index];
|
|
}
|
|
|
|
/**
|
|
* Creates a new XMPPConnection using the connection preferences. This is useful when
|
|
* not using a connection from the connection pool in a test case.
|
|
*
|
|
* @return a new XMPP connection.
|
|
*/
|
|
protected XMPPConnection createConnection() {
|
|
// Create the configuration for this new connection
|
|
ConnectionConfiguration config = new ConnectionConfiguration(host, port);
|
|
config.setCompressionEnabled(compressionEnabled);
|
|
config.setSendPresence(sendInitialPresence());
|
|
if (getSocketFactory() == null) {
|
|
config.setSocketFactory(getSocketFactory());
|
|
}
|
|
return new XMPPConnection(config);
|
|
}
|
|
|
|
/**
|
|
* Returns the name of the user (e.g. johndoe) that is using the connection
|
|
* located at the requested position.
|
|
*
|
|
* @param index the position in the pool of the connection to look for.
|
|
* @return the user of the user (e.g. johndoe).
|
|
*/
|
|
protected String getUsername(int index) {
|
|
return usernames[index];
|
|
}
|
|
|
|
/**
|
|
* Returns the password of the user (e.g. johndoe) that is using the connection
|
|
* located at the requested position.
|
|
*
|
|
* @param index the position in the pool of the connection to look for.
|
|
* @return the password of the user (e.g. johndoe).
|
|
*/
|
|
protected String getPassword(int index) {
|
|
return passwords[index];
|
|
}
|
|
|
|
/**
|
|
* Returns the bare XMPP address of the user (e.g. johndoe@jabber.org) that is
|
|
* using the connection located at the requested position.
|
|
*
|
|
* @param index the position in the pool of the connection to look for.
|
|
* @return the bare XMPP address of the user (e.g. johndoe@jabber.org).
|
|
*/
|
|
protected String getBareJID(int index) {
|
|
return getUsername(index) + "@" + getConnection(index).getServiceName();
|
|
}
|
|
|
|
/**
|
|
* Returns the full XMPP address of the user (e.g. johndoe@jabber.org/Smack) that is
|
|
* using the connection located at the requested position.
|
|
*
|
|
* @param index the position in the pool of the connection to look for.
|
|
* @return the full XMPP address of the user (e.g. johndoe@jabber.org/Smack).
|
|
*/
|
|
protected String getFullJID(int index) {
|
|
return getBareJID(index) + "/Smack";
|
|
}
|
|
|
|
protected String getHost() {
|
|
return host;
|
|
}
|
|
|
|
protected int getPort() {
|
|
return port;
|
|
}
|
|
|
|
protected String getServiceName() {
|
|
return serviceName;
|
|
}
|
|
|
|
/**
|
|
* Returns the default groupchat service domain.
|
|
*
|
|
* @return the default groupchat service domain.
|
|
*/
|
|
protected String getChatDomain() {
|
|
return chatDomain;
|
|
}
|
|
|
|
/**
|
|
* Returns the default MUC service domain.
|
|
*
|
|
* @return the default MUC service domain.
|
|
*/
|
|
protected String getMUCDomain() {
|
|
return mucDomain + "." + serviceName;
|
|
}
|
|
|
|
protected void setUp() throws Exception {
|
|
super.setUp();
|
|
init();
|
|
if (getMaxConnections() < 1) {
|
|
return;
|
|
}
|
|
connections = new XMPPConnection[getMaxConnections()];
|
|
usernames = new String[getMaxConnections()];
|
|
passwords = new String[getMaxConnections()];
|
|
|
|
try {
|
|
// Connect to the server
|
|
for (int i = 0; i < getMaxConnections(); i++) {
|
|
connections[i] = createConnection();
|
|
if (!createOfflineConnections())
|
|
connections[i].connect();
|
|
|
|
String currentPassword = usernamePrefix + (i+1);
|
|
String currentUser = currentPassword;
|
|
|
|
if (passwordPrefix != null)
|
|
currentPassword = (samePassword ? passwordPrefix : passwordPrefix + (i+1));
|
|
|
|
usernames[i] = currentUser;
|
|
passwords[i] = currentPassword;
|
|
}
|
|
// Use the host name that the server reports. This is a good idea in most
|
|
// cases, but could fail if the user set a hostname in their XMPP server
|
|
// that will not resolve as a network connection.
|
|
host = connections[0].getHost();
|
|
serviceName = connections[0].getServiceName();
|
|
|
|
if (!createOfflineConnections()) {
|
|
for (int i = 0; i < getMaxConnections(); i++) {
|
|
String currentUser = usernames[i];
|
|
String currentPassword = passwords[i];
|
|
|
|
try {
|
|
getConnection(i).login(currentUser, currentPassword, "Smack");
|
|
} catch (XMPPException e) {
|
|
// Create the test accounts
|
|
if (!getConnection(0).getAccountManager().supportsAccountCreation())
|
|
fail("Server does not support account creation");
|
|
|
|
// Create the account and try logging in again as the
|
|
// same user.
|
|
try {
|
|
createAccount(i, currentUser, currentPassword);
|
|
} catch (Exception e1) {
|
|
e1.printStackTrace();
|
|
fail("Could not create user: " + currentUser);
|
|
}
|
|
i--;
|
|
}
|
|
}
|
|
// Let the server process the available presences
|
|
Thread.sleep(150);
|
|
}
|
|
}
|
|
catch (Exception e) {
|
|
e.printStackTrace();
|
|
fail(e.getMessage());
|
|
}
|
|
}
|
|
|
|
protected void connectAndLogin(int connectionIndex) throws XMPPException
|
|
{
|
|
String password = usernamePrefix + (connectionIndex + 1);
|
|
|
|
if (passwordPrefix != null)
|
|
password = (samePassword ? passwordPrefix : passwordPrefix + (connectionIndex + 1));
|
|
|
|
XMPPConnection con = getConnection(connectionIndex);
|
|
|
|
if (!con.isConnected())
|
|
con.connect();
|
|
try {
|
|
con.login(usernamePrefix + (connectionIndex + 1), password, "Smack");
|
|
} catch (XMPPException e) {
|
|
createAccount(connectionIndex, usernamePrefix + (connectionIndex + 1), password);
|
|
con.login(usernamePrefix + (connectionIndex + 1), password, "Smack");
|
|
}
|
|
}
|
|
|
|
protected void disconnect(int connectionIndex) throws XMPPException
|
|
{
|
|
getConnection(connectionIndex).disconnect();
|
|
}
|
|
|
|
private void createAccount(int connectionIdx, String username, String password)
|
|
{
|
|
// Create the test account
|
|
try {
|
|
getConnection(connectionIdx).getAccountManager().createAccount(username, password);
|
|
createdUserIdx.add(connectionIdx);
|
|
} catch (XMPPException e) {
|
|
e.printStackTrace();
|
|
fail(e.getMessage());
|
|
}
|
|
}
|
|
|
|
protected void tearDown() throws Exception {
|
|
super.tearDown();
|
|
|
|
for (int i = 0; i < getMaxConnections(); i++)
|
|
{
|
|
if (createdUserIdx.contains(i))
|
|
{
|
|
try {
|
|
// If not connected, connect so that we can delete the account.
|
|
if (!getConnection(i).isConnected()) {
|
|
XMPPConnection con = getConnection(i);
|
|
con.connect();
|
|
con.login(getUsername(i), getUsername(i));
|
|
}
|
|
else if (!getConnection(i).isAuthenticated()) {
|
|
getConnection(i).login(getUsername(i), getUsername(i));
|
|
}
|
|
// Delete the created account for the test
|
|
getConnection(i).getAccountManager().deleteAccount();
|
|
}
|
|
catch (Exception e) {
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
if (getConnection(i).isConnected()) {
|
|
// Close the connection
|
|
getConnection(i).disconnect();
|
|
}
|
|
}
|
|
}
|
|
|
|
protected boolean sendInitialPresence() {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Initializes the context of the test case. We will first try to load the configuration from
|
|
* a file whose name is conformed by the test case class name plus an .xml extension
|
|
* (e.g RosterTest.xml). If no file was found under that name then we will try to load the
|
|
* default configuration for all the test cases from the file "config/test-case.xml".
|
|
*
|
|
*/
|
|
private void init() {
|
|
try {
|
|
boolean found = false;
|
|
// Try to load the configutation from an XML file specific for this test case
|
|
Enumeration<URL> resources =
|
|
ClassLoader.getSystemClassLoader().getResources(getConfigurationFilename());
|
|
while (resources.hasMoreElements()) {
|
|
found = parseURL(resources.nextElement());
|
|
}
|
|
// If none was found then try to load the configuration from the default configuration
|
|
// file (i.e. "config/test-case.xml")
|
|
if (!found) {
|
|
resources = ClassLoader.getSystemClassLoader().getResources("config/test-case.xml");
|
|
while (resources.hasMoreElements()) {
|
|
found = parseURL(resources.nextElement());
|
|
}
|
|
}
|
|
if (!found) {
|
|
System.err.println("File config/test-case.xml not found. Using default config.");
|
|
}
|
|
}
|
|
catch (Exception e) {
|
|
/* Do Nothing */
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns true if the given URL was found and parsed without problems. The file provided
|
|
* by the URL must contain information useful for the test case configuration, such us,
|
|
* host and port of the server.
|
|
*
|
|
* @param url the url of the file to parse.
|
|
* @return true if the given URL was found and parsed without problems.
|
|
*/
|
|
private boolean parseURL(URL url) {
|
|
boolean parsedOK = false;
|
|
InputStream systemStream = null;
|
|
try {
|
|
systemStream = url.openStream();
|
|
XmlPullParser parser = new MXParser();
|
|
parser.setFeature(XmlPullParser.FEATURE_PROCESS_NAMESPACES, true);
|
|
parser.setInput(systemStream, "UTF-8");
|
|
int eventType = parser.getEventType();
|
|
do {
|
|
if (eventType == XmlPullParser.START_TAG) {
|
|
if (parser.getName().equals("host")) {
|
|
host = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("port")) {
|
|
port = parseIntProperty(parser, port);
|
|
}
|
|
else if (parser.getName().equals("serviceName")) {
|
|
serviceName = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("chat")) {
|
|
chatDomain = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("muc")) {
|
|
mucDomain = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("username")) {
|
|
usernamePrefix = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("password")) {
|
|
samePassword = "true".equals(parser.getAttributeValue(0));
|
|
passwordPrefix = parser.nextText();
|
|
}
|
|
else if (parser.getName().equals("testAnonymousLogin")) {
|
|
testAnonymousLogin = "true".equals(parser.nextText());
|
|
}
|
|
else if (parser.getName().equals("accountCreationParameters")) {
|
|
int numAttributes = parser.getAttributeCount();
|
|
String key = null;
|
|
String value = null;
|
|
|
|
for (int i = 0; i < numAttributes; i++) {
|
|
key = parser.getAttributeName(i);
|
|
value = parser.getAttributeValue(i);
|
|
accountCreationParameters.put(key, value);
|
|
}
|
|
}
|
|
else if (parser.getName().equals("compressionEnabled")) {
|
|
compressionEnabled = "true".equals(parser.nextText());
|
|
}
|
|
}
|
|
eventType = parser.next();
|
|
}
|
|
while (eventType != XmlPullParser.END_DOCUMENT);
|
|
parsedOK = true;
|
|
}
|
|
catch (Exception e) {
|
|
e.printStackTrace();
|
|
}
|
|
finally {
|
|
try {
|
|
systemStream.close();
|
|
}
|
|
catch (Exception e) {
|
|
/* Do Nothing */
|
|
}
|
|
}
|
|
return parsedOK;
|
|
}
|
|
|
|
private static int parseIntProperty(XmlPullParser parser, int defaultValue) throws Exception {
|
|
try {
|
|
return Integer.parseInt(parser.nextText());
|
|
}
|
|
catch (NumberFormatException nfe) {
|
|
nfe.printStackTrace();
|
|
return defaultValue;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the name of the configuration file related to <b>this</b> test case. By default all
|
|
* the test cases will use the same configuration file. However, it's possible to override the
|
|
* default configuration by providing a file of the form <test case class name>.xml
|
|
* (e.g. RosterTest.xml).
|
|
*
|
|
* @return the name of the configuration file related to this test case.
|
|
*/
|
|
private String getConfigurationFilename() {
|
|
String fullClassName = this.getClass().getName();
|
|
int firstChar = fullClassName.lastIndexOf('.') + 1;
|
|
return "config/" + fullClassName.substring(firstChar) + ".xml";
|
|
}
|
|
|
|
/**
|
|
* Subscribes all connections with each other: They all become friends
|
|
*
|
|
* @throws XMPPException
|
|
*/
|
|
protected void letsAllBeFriends() throws XMPPException {
|
|
ConnectionUtils.letsAllBeFriends(connections);
|
|
}
|
|
|
|
/**
|
|
* Compares two contents of two byte arrays to make sure that they are equal
|
|
*
|
|
* @param message The message to show in the case of failure
|
|
* @param byteArray1 The first byte array.
|
|
* @param byteArray2 The second byte array.
|
|
*/
|
|
public static void assertEquals(String message, byte [] byteArray1, byte [] byteArray2) {
|
|
if(byteArray1.length != byteArray2.length) {
|
|
fail(message);
|
|
}
|
|
for(int i = 0; i < byteArray1.length; i++) {
|
|
assertEquals(message, byteArray1[i], byteArray2[i]);
|
|
}
|
|
}
|
|
|
|
public boolean isTestAnonymousLogin() {
|
|
return testAnonymousLogin;
|
|
}
|
|
|
|
public Map<String, String> getAccountCreationParameters() {
|
|
return accountCreationParameters;
|
|
}
|
|
}
|