/* * Copyright 2007 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.mockftpserver.stub.command; import java.io.IOException; import java.net.InetAddress; import org.apache.log4j.Logger; import org.mockftpserver.core.command.Command; import org.mockftpserver.core.command.CommandHandler; import org.mockftpserver.core.command.InvocationRecord; import org.mockftpserver.core.command.ReplyCodes; import org.mockftpserver.core.session.Session; import org.mockftpserver.core.util.Assert; /** * CommandHandler for the PASV (Passove Mode) command. Request the Session to switch to passive * data connection mode. Return a reply code of 227, along with response text of the form: * "Entering Passive Mode. (h1,h2,h3,h4,p1,p2)", where h1..h4 are the 4 * bytes of the 32-bit internet host address of the server, and p1..p2 are the 2 * bytes of the 16-bit TCP port address of the data connection on the server to which * the client must connect. See RFC959 for more information. *

* Each invocation record stored by this CommandHandler contains no data elements. * * @version $Revision$ - $Date$ * * @author Chris Mair */ public final class PasvCommandHandler extends AbstractStubCommandHandler implements CommandHandler { private static final Logger LOG = Logger.getLogger(PasvCommandHandler.class); /** * Constructor. Initialize the replyCode. */ public PasvCommandHandler() { setReplyCode(ReplyCodes.PASV_OK); } /** * @throws IOException * @see org.mockftpserver.core.command.CommandHandler#handleCommand(Command, Session, InvocationRecord) */ public void handleCommand(Command command, Session session, InvocationRecord invocationRecord) throws IOException { int port = session.switchToPassiveMode(); InetAddress server = session.getServerHost(); Assert.isTrue(port > -1, "The server-side port is invalid: " + port); LOG.debug("server=" + server + " port=" + port); String hostAndPort = "(" + convertHostAndPortToStringOfBytes(server, port) + ")"; sendReply(session, hostAndPort); } /** * Convert the InetAddess and port number to a comma-delimited list of byte values, * suitable for the response String from the PASV command. * @param host - the InetAddress * @param port - the port number * @return the comma-delimited list of byte values, e.g., "196,168,44,55,23,77" */ static String convertHostAndPortToStringOfBytes(InetAddress host, int port) { StringBuffer buffer = new StringBuffer(); byte[] address = host.getAddress(); for (int i = 0; i < address.length; i++) { int positiveValue = (address[i] >=0) ? address[i] : 256 + address[i]; buffer.append(positiveValue); buffer.append(","); } int p1 = port >> 8; int p2 = port % 256; buffer.append(String.valueOf(p1)); buffer.append(","); buffer.append(String.valueOf(p2)); return buffer.toString(); } }