/* * Copyright (C) 2005-2008 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.openfire; import org.dom4j.Element; import org.jivesoftware.openfire.multiplex.UnknownStanzaException; import org.jivesoftware.openfire.net.SASLAuthentication; import org.jivesoftware.openfire.session.LocalClientSession; import org.xmpp.packet.*; /** * Handles the routing of packets to a particular session. It will invoke all of the appropriate * interceptors, before and after having the server process the message. * * @author Alexander Wenckus */ public class SessionPacketRouter implements PacketRouter { protected LocalClientSession session; private PacketRouter router; private boolean skipJIDValidation = false; public SessionPacketRouter(LocalClientSession session) { this.session = session; router = XMPPServer.getInstance().getPacketRouter(); } /** * Sets if TO addresses of Elements being routed should be validated. Doing stringprep operations * is very expensive and sometimes we already validated the TO address so there is no need to * validate again the address. For instance, when using Connection Managers the validation * is done by the Connection Manager so we can just trust the TO address. On the other hand, * the FROM address is set by the server so there is no need to validate it.<p> * * By default validation is enabled. * * @param skipJIDValidation true if validation of TO address is enabled. */ public void setSkipJIDValidation(boolean skipJIDValidation) { this.skipJIDValidation = skipJIDValidation; } public void route(Element wrappedElement) throws UnknownStanzaException { String tag = wrappedElement.getName(); if ("auth".equals(tag) || "response".equals(tag)) { SASLAuthentication.handle(session, wrappedElement); } else if ("iq".equals(tag)) { route(getIQ(wrappedElement)); } else if ("message".equals(tag)) { route(new Message(wrappedElement, skipJIDValidation)); } else if ("presence".equals(tag)) { route(new Presence(wrappedElement, skipJIDValidation)); } else { throw new UnknownStanzaException(); } } private IQ getIQ(Element doc) { Element query = doc.element("query"); if (query != null && "jabber:iq:roster".equals(query.getNamespaceURI())) { return new Roster(doc); } else { return new IQ(doc, skipJIDValidation); } } @Override public void route(Packet packet) { // Security: Don't allow users to send packets on behalf of other users packet.setFrom(session.getAddress()); if(packet instanceof IQ) { route((IQ)packet); } else if(packet instanceof Message) { route((Message)packet); } else if(packet instanceof Presence) { route((Presence)packet); } } @Override public void route(IQ packet) { packet.setFrom(session.getAddress()); router.route(packet); session.incrementClientPacketCount(); } @Override public void route(Message packet) { packet.setFrom(session.getAddress()); router.route(packet); session.incrementClientPacketCount(); } @Override public void route(Presence packet) { packet.setFrom(session.getAddress()); router.route(packet); session.incrementClientPacketCount(); } }