2014-02-17 23:58:40 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Copyright the original author or authors
|
|
|
|
*
|
|
|
|
* 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.util;
|
|
|
|
|
|
|
|
import static org.junit.Assert.*;
|
|
|
|
|
|
|
|
import java.io.StringReader;
|
|
|
|
import java.io.StringWriter;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Queue;
|
|
|
|
|
|
|
|
import javax.xml.transform.OutputKeys;
|
|
|
|
import javax.xml.transform.Source;
|
|
|
|
import javax.xml.transform.Transformer;
|
|
|
|
import javax.xml.transform.TransformerFactory;
|
|
|
|
import javax.xml.transform.stream.StreamResult;
|
|
|
|
import javax.xml.transform.stream.StreamSource;
|
|
|
|
|
|
|
|
import org.jivesoftware.smack.packet.Packet;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class can be used in conjunction with a mocked XMPP connection (
|
|
|
|
* {@link ConnectionUtils#createMockedConnection(Protocol, String, String)}) to
|
|
|
|
* verify a XMPP protocol. This can be accomplished in the following was:
|
|
|
|
* <ul>
|
|
|
|
* <li>add responses to packets sent over the mocked XMPP connection by the
|
|
|
|
* method to test in the order the tested method awaits them</li>
|
|
|
|
* <li>call the method to test</li>
|
|
|
|
* <li>call {@link #verifyAll()} to run assertions on the request/response pairs
|
|
|
|
* </li>
|
|
|
|
* </ul>
|
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* <pre>
|
|
|
|
* <code>
|
|
|
|
* public void methodToTest() {
|
|
|
|
* Packet packet = new Packet(); // create an XMPP packet
|
|
|
|
* PacketCollector collector = connection.createPacketCollector(new PacketIDFilter());
|
|
|
|
* connection.sendPacket(packet);
|
|
|
|
* Packet reply = collector.nextResult();
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* public void testMethod() {
|
|
|
|
* // create protocol
|
|
|
|
* Protocol protocol = new Protocol();
|
|
|
|
* // create mocked connection
|
|
|
|
* Connection connection = ConnectionUtils.createMockedConnection(protocol, "user@xmpp-server", "xmpp-server");
|
|
|
|
*
|
|
|
|
* // add reply packet to protocol
|
|
|
|
* Packet reply = new Packet();
|
|
|
|
* protocol.add(reply);
|
|
|
|
*
|
|
|
|
* // call method to test
|
|
|
|
* methodToTest();
|
|
|
|
*
|
|
|
|
* // verify protocol
|
|
|
|
* protocol.verifyAll();
|
|
|
|
* }
|
|
|
|
* </code>
|
|
|
|
* </pre>
|
|
|
|
*
|
|
|
|
* Additionally to adding the response to the protocol instance you can pass
|
|
|
|
* verifications that will be executed when {@link #verifyAll()} is invoked.
|
|
|
|
* (See {@link Verification} for more details.)
|
|
|
|
* <p>
|
|
|
|
* If the {@link #printProtocol} flag is set to true {@link #verifyAll()} will
|
|
|
|
* also print out the XML messages in the order they are sent to the console.
|
|
|
|
* This may be useful to inspect the whole protocol "by hand".
|
|
|
|
*
|
|
|
|
* @author Henning Staib
|
|
|
|
*/
|
|
|
|
public class Protocol {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set to <code>true</code> to print XML messages to the console while
|
|
|
|
* verifying the protocol.
|
|
|
|
*/
|
|
|
|
public boolean printProtocol = false;
|
|
|
|
|
|
|
|
// responses to requests are taken form this queue
|
|
|
|
Queue<Packet> responses = new LinkedList<Packet>();
|
|
|
|
|
|
|
|
// list of verifications
|
|
|
|
List<Verification<?, ?>[]> verificationList = new ArrayList<Verification<?, ?>[]>();
|
|
|
|
|
|
|
|
// list of requests
|
|
|
|
List<Packet> requests = new ArrayList<Packet>();
|
|
|
|
|
|
|
|
// list of all responses
|
|
|
|
List<Packet> responsesList = new ArrayList<Packet>();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a responses and all verifications for the request/response pair to
|
|
|
|
* the protocol.
|
|
|
|
*
|
|
|
|
* @param response the response for a request
|
|
|
|
* @param verifications verifications for request/response pair
|
|
|
|
*/
|
|
|
|
public void addResponse(Packet response, Verification<?, ?>... verifications) {
|
|
|
|
responses.offer(response);
|
|
|
|
verificationList.add(verifications);
|
|
|
|
responsesList.add(response);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Verifies the request/response pairs by checking if their numbers match
|
|
|
|
* and executes the verification for each pair.
|
|
|
|
*/
|
|
|
|
@SuppressWarnings("unchecked")
|
|
|
|
public void verifyAll() {
|
|
|
|
assertEquals(requests.size(), responsesList.size());
|
|
|
|
|
|
|
|
if (printProtocol)
|
|
|
|
System.out.println("=================== Start ===============\n");
|
|
|
|
|
|
|
|
for (int i = 0; i < requests.size(); i++) {
|
|
|
|
Packet request = requests.get(i);
|
|
|
|
Packet response = responsesList.get(i);
|
|
|
|
|
|
|
|
if (printProtocol) {
|
|
|
|
System.out.println("------------------- Request -------------\n");
|
|
|
|
System.out.println(prettyFormat(request.toXML()));
|
|
|
|
System.out.println("------------------- Response ------------\n");
|
|
|
|
if (response != null) {
|
|
|
|
System.out.println(prettyFormat(response.toXML()));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
System.out.println("No response");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Verification<Packet, Packet>[] verifications = (Verification<Packet, Packet>[]) verificationList.get(i);
|
|
|
|
if (verifications != null) {
|
|
|
|
for (Verification<Packet, Packet> verification : verifications) {
|
|
|
|
verification.verify(request, response);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (printProtocol)
|
|
|
|
System.out.println("=================== End =================\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the responses queue.
|
|
|
|
*
|
|
|
|
* @return the responses queue
|
|
|
|
*/
|
|
|
|
protected Queue<Packet> getResponses() {
|
|
|
|
return responses;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of all collected requests.
|
|
|
|
*
|
|
|
|
* @return list of requests
|
|
|
|
*/
|
|
|
|
public List<Packet> getRequests() {
|
|
|
|
return requests;
|
|
|
|
}
|
|
|
|
|
|
|
|
private String prettyFormat(String input, int indent) {
|
|
|
|
try {
|
|
|
|
Source xmlInput = new StreamSource(new StringReader(input));
|
|
|
|
StringWriter stringWriter = new StringWriter();
|
|
|
|
StreamResult xmlOutput = new StreamResult(stringWriter);
|
|
|
|
Transformer transformer = TransformerFactory.newInstance().newTransformer();
|
|
|
|
transformer.setOutputProperty(OutputKeys.INDENT, "yes");
|
|
|
|
transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount",
|
|
|
|
String.valueOf(indent));
|
|
|
|
transformer.transform(xmlInput, xmlOutput);
|
|
|
|
return xmlOutput.getWriter().toString();
|
|
|
|
}
|
|
|
|
catch (Exception e) {
|
|
|
|
return "error while formatting the XML: " + e.getMessage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private String prettyFormat(String input) {
|
|
|
|
return prettyFormat(input, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|