Проблема с org.apache.commons.net.ftp.FTPClient listFiles () - PullRequest
7 голосов
/ 05 апреля 2010

Метод listFiles() org.apache.commons.net.ftp.FTPClient прекрасно работает с сервером Filezilla на 127.0.0.1, но возвращает null в корневом каталоге общедоступных FTP-серверов, таких как belnet.be.

Существует идентичнаявопрос по ссылке ниже, но enterRemotePassiveMode(), похоже, не помогает. Apache Commons FTPClient.listFiles

Может ли быть проблема с анализом списка?Если так, как можно решить эту проблему?

Редактировать: Вот дамп кэша каталогов:

Дамп кэша каталогов FileZilla

Дампирование 1 кэшированных каталогов

Entry 1:
Path: /
Server: anonymous@ftp.belnet.be:21, type: 4096
Directory contains 7 items:
  lrw-r--r-- ftp ftp      D          28      2009-06-17   debian
  lrw-r--r-- ftp ftp      D          31      2009-06-17   debian-cd
  -rw-r--r-- ftp ftp                  0 2010-03-04 13:30  keepalive.txt
  drwxr-xr-x ftp ftp      D        4096 2010-02-18 14:22  mirror
  lrw-r--r-- ftp ftp      D           6      2009-06-17   mirrors
  drwxr-xr-x ftp ftp      D        4096      2009-06-23   packages
  lrw-r--r-- ftp ftp      D           1      2009-06-17   pub

Вот мой код, использующий созданную мной оболочку (тестирование внутри оболочки дает те же результаты):

public static void main(String[] args) {        
    FTPUtils ftpUtils = new FTPUtils();
    String ftpURL = "ftp.belnet.be";
    Connection connection = ftpUtils.getFTPClientManager().getConnection( ftpURL );

    if( connection == null ){
        System.out.println( "Could not connect" );
        return; 
    }

    FTPClientManager manager = connection.getFptClientManager();
    FTPClient client = manager.getClient();

    try {
        client.enterRemotePassiveMode();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

    if( connection != null ){
        System.out.println( "Connected to FTP" );
        connection.login("Anonymous", "Anonymous");
        if( connection.isLoggedIn() ){
            System.out.println( "Login successful" );
            LoggedInManager loggedin = connection.getLoggedInManager(); 
            System.out.println( loggedin );
            String[] fileList = loggedin.getFileList();

            System.out.println( loggedin.getWorkingDirectory() );

            if( fileList == null || fileList.length == 0 )
                System.out.println( "No files found" );
            else{
                for (String name : fileList ) {
                    System.out.println( name );
                }
            }

            connection.disconnect();

            if( connection.isDisconnected() )
                System.out.println( "Disconnection successful" );
            else
                System.out.println( "Error disconnecting" );
        }else{
            System.out.println( "Unable to login" );
        }
    } else {
        System.out.println( "Could not connect" );
    }
}

Выводит следующий вывод:

Connected to FTP
Login succesful
utils.ftp.FTPClientManager$Connection$LoggedInManager@156ee8e
null
No files found
Disconnection successful

Внутри оболочки (попытался использовать listNames() и listFiles()):

        public String[] getFileList() {
            String[] fileList = null;
            FTPFile[] ftpFiles = null;

            try {
                ftpFiles = client.listFiles();
                //fileList = client.listNames();
                //System.out.println( client.listNames() );
            } catch (IOException e) {
                return null;
            }

            fileList = new String[ ftpFiles.length ];

            for( int i = 0; i < ftpFiles.length; i++ ){
                fileList[ i ] = ftpFiles[ i ].getName();
            }

            return fileList;
        }

Что касается FTPClient, он обрабатывается следующим образом:

public class FTPUtils {

private FTPClientManager clientManager;

public FTPClientManager getFTPClientManager(){
    clientManager = new FTPClientManager();
    clientManager.setClient( new FTPClient() );

    return clientManager;
}

Ответы [ 3 ]

8 голосов
/ 05 апреля 2010

Каждый FTP-сервер имеет свою разметку списка файлов (да, это не является частью стандарта FTP, он тупой), и поэтому вы должны использовать правильный FTPFileEntryParser, либо указав его вручную, либо разрешив CommonsFTP автоматически -обнаружить это.

Автоопределение обычно работает нормально, но иногда это не так, и вы должны указать его явно, например,

FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_UNIX);

FTPClient client = FTPClient();
client.configure(conf);

Это явно устанавливает ожидаемый тип FTP-сервера в UNIX. Попробуйте различные типы, посмотрите, как это происходит. Я пытался выяснить сам, но ftp.belnet.be отказывается от моих соединений: (* ​​1008 *

2 голосов
/ 05 апреля 2010

Вы пробовали проверить, можете ли вы перечислять файлы с помощью обычного FTP-клиента? (По какой-то причине я даже не могу подключиться к порту FTP "belnet.be".)

РЕДАКТИРОВАТЬ

Согласно javadoc для listFiles () , анализ выполняется с использованием экземпляра FTPFileEntryParser , предоставленного фабрикой синтаксических анализаторов. Возможно, вам необходимо выяснить, какой из анализаторов соответствует выводу LIST сервера FTP, и соответствующим образом настроить фабрику.

1 голос
/ 10 апреля 2013

В более ранней версии apache-Commons-net возникла проблема с синтаксическим анализом, команда SYST , которая возвращает тип сервера, когда возвращает ноль (внезапно) не была обработана в parsingException. Попробуйте использовать последний jar apache-commons-net, это может решить вашу проблему.

...