/** * $RCSfile$ * $Revision$ * $Date$ * * Copyright 2003-2007 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; import org.jivesoftware.smack.filter.PacketFilter; import org.jivesoftware.smack.packet.Packet; import java.io.IOException; import java.io.Writer; import java.util.ArrayList; import java.util.List; import java.util.Queue; import java.util.concurrent.ConcurrentLinkedQueue; /** * Writes packets to a XMPP server. Packets are sent using a dedicated thread. Packet * interceptors can be registered to dynamically modify packets before they're actually * sent. Packet listeners can be registered to listen for all outgoing packets. * * @author Matt Tucker */ class PacketWriter { private Thread writerThread; private Thread keepAliveThread; private Writer writer; private XMPPConnection connection; final private Queue queue; private boolean done; final protected List listeners = new ArrayList(); private boolean listenersDeleted; /** * Timestamp when the last stanza was sent to the server. This information is used * by the keep alive process to only send heartbeats when the connection has been idle. */ private long lastActive = System.currentTimeMillis(); /** * List of PacketInterceptor that will be notified when a new packet is about to be * sent to the server. These interceptors may modify the packet before it is being * actually sent to the server. */ final private List interceptors = new ArrayList(); /** * Flag that indicates if an interceptor was deleted. This is an optimization flag. */ private boolean interceptorDeleted = false; /** * Creates a new packet writer with the specified connection. * * @param connection the connection. */ protected PacketWriter(XMPPConnection connection) { this.queue = new ConcurrentLinkedQueue(); this.connection = connection; init(); } /** * Initializes the writer in order to be used. It is called at the first connection and also * is invoked if the connection is disconnected by an error. */ protected void init() { this.writer = connection.writer; listenersDeleted = false; interceptorDeleted = false; done = false; writerThread = new Thread() { public void run() { writePackets(this); } }; writerThread.setName("Smack Packet Writer (" + connection.connectionCounterValue + ")"); writerThread.setDaemon(true); } /** * Sends the specified packet to the server. * * @param packet the packet to send. */ public void sendPacket(Packet packet) { if (!done) { // Invoke interceptors for the new packet that is about to be sent. Interceptors // may modify the content of the packet. processInterceptors(packet); queue.add(packet); synchronized (queue) { queue.notifyAll(); } // Process packet writer listeners. Note that we're using the sending // thread so it's expected that listeners are fast. processListeners(packet); } } /** * Registers a packet listener with this writer. The listener will be * notified immediately after every packet this writer sends. A packet filter * determines which packets will be delivered to the listener. Note that the thread * that writes packets will be used to invoke the listeners. Therefore, each * packet listener should complete all operations quickly or use a different * thread for processing. * * @param packetListener the packet listener to notify of sent packets. * @param packetFilter the packet filter to use. */ public void addPacketListener(PacketListener packetListener, PacketFilter packetFilter) { synchronized (listeners) { listeners.add(new ListenerWrapper(packetListener, packetFilter)); } } /** * Removes a packet listener. * * @param packetListener the packet listener to remove. */ public void removePacketListener(PacketListener packetListener) { synchronized (listeners) { for (int i=0; i 0) { KeepAliveTask target = new KeepAliveTask(keepAliveInterval); keepAliveThread = new Thread(target); target.setThread(keepAliveThread); keepAliveThread.setDaemon(true); keepAliveThread.setName("Smack Keep Alive (" + connection.connectionCounterValue + ")"); keepAliveThread.start(); } } void setWriter(Writer writer) { this.writer = writer; } /** * Shuts down the packet writer. Once this method has been called, no further * packets will be written to the server. */ public void shutdown() { done = true; synchronized (queue) { queue.notifyAll(); } } /** * Returns the next available packet from the queue for writing. * * @return the next packet for writing. */ private Packet nextPacket() { Packet packet = null; // Wait until there's a packet or we're done. while (!done && (packet = queue.poll()) == null) { try { synchronized (queue) { queue.wait(); } } catch (InterruptedException ie) { // Do nothing } } return packet; } private void writePackets(Thread thisThread) { try { // Open the stream. openStream(); // Write out packets from the queue. while (!done && (writerThread == thisThread)) { Packet packet = nextPacket(); if (packet != null) { synchronized (writer) { writer.write(packet.toXML()); writer.flush(); // Keep track of the last time a stanza was sent to the server lastActive = System.currentTimeMillis(); } } } // Flush out the rest of the queue. try { synchronized (writer) { while (!queue.isEmpty()) { Packet packet = queue.remove(); writer.write(packet.toXML()); } writer.flush(); } } catch (Exception e) { e.printStackTrace(); } // Close the stream. try { writer.write(""); writer.flush(); } catch (Exception e) { // Do nothing } finally { try { writer.close(); } catch (Exception e) { // Do nothing } } } catch (IOException ioe){ if (!done) { done = true; connection.packetReader.notifyConnectionError(ioe); } } } /** * Process listeners. */ private void processListeners(Packet packet) { // Clean up null entries in the listeners list if the flag is set. List // removes are done seperately so that the main notification process doesn't // need to synchronize on the list. synchronized (listeners) { if (listenersDeleted) { for (int i=listeners.size()-1; i>=0; i--) { if (listeners.get(i) == null) { listeners.remove(i); } } listenersDeleted = false; } } // Notify the listeners of the new sent packet int size = listeners.size(); for (int i=0; i=0; i--) { if (interceptors.get(i) == null) { interceptors.remove(i); } } interceptorDeleted = false; } } // Notify the interceptors of the new packet to be sent int size = interceptors.size(); for (int i=0; i"); writer.write(stream.toString()); writer.flush(); } /** * A wrapper class to associate a packet filter with a listener. */ protected static class ListenerWrapper { private PacketListener packetListener; private PacketFilter packetFilter; public ListenerWrapper(PacketListener packetListener, PacketFilter packetFilter) { this.packetListener = packetListener; this.packetFilter = packetFilter; } public boolean equals(Object object) { if (object == null) { return false; } if (object instanceof ListenerWrapper) { return ((ListenerWrapper)object).packetListener.equals(this.packetListener); } else if (object instanceof PacketListener) { return object.equals(this.packetListener); } return false; } public void notifyListener(Packet packet) { if (packetFilter == null || packetFilter.accept(packet)) { packetListener.processPacket(packet); } } } /** * A wrapper class to associate a packet filter with an interceptor. */ private static class InterceptorWrapper { private PacketInterceptor packetInterceptor; private PacketFilter packetFilter; public InterceptorWrapper(PacketInterceptor packetInterceptor, PacketFilter packetFilter) { this.packetInterceptor = packetInterceptor; this.packetFilter = packetFilter; } public boolean equals(Object object) { if (object == null) { return false; } if (object instanceof InterceptorWrapper) { return ((InterceptorWrapper) object).packetInterceptor .equals(this.packetInterceptor); } else if (object instanceof PacketInterceptor) { return object.equals(this.packetInterceptor); } return false; } public void notifyListener(Packet packet) { if (packetFilter == null || packetFilter.accept(packet)) { packetInterceptor.interceptPacket(packet); } } } /** * A TimerTask that keeps connections to the server alive by sending a space * character on an interval. */ private class KeepAliveTask implements Runnable { private int delay; private Thread thread; public KeepAliveTask(int delay) { this.delay = delay; } protected void setThread(Thread thread) { this.thread = thread; } public void run() { try { // Sleep 15 seconds before sending first heartbeat. This will give time to // properly finish TLS negotiation and then start sending heartbeats. Thread.sleep(15000); } catch (InterruptedException ie) { // Do nothing } while (!done && keepAliveThread == thread) { synchronized (writer) { // Send heartbeat if no packet has been sent to the server for a given time if (System.currentTimeMillis() - lastActive >= delay) { try { writer.write(" "); writer.flush(); } catch (Exception e) { // Do nothing } } } try { // Sleep until we should write the next keep-alive. Thread.sleep(delay); } catch (InterruptedException ie) { // Do nothing } } } } }