Renci.SshNet: ответ сервера не содержит идентификации протокола ssh

Я работаю с библиотекой Renci SSH.Net в приложении WPF, и у меня возникла проблема с использованием SFTP-клиента. Когда пользователь пытается подключиться для загрузки некоторых файлов с SFTP-сервера, он получает сообщение, показанное ниже:

Ответ сервера не содержит идентификатор протокола ssh

Похоже, это не связано с сервером, так как я могу нормально подключаться и загружать файлы на своем рабочем столе для разработки и на дополнительном ноутбуке. Это же приложение может подключаться через SSH и выполнять команды без проблем, проблема заключается только в соединении SFTP. Я ищу небольшое руководство относительно того, с чего начать устранение неполадок.

Код для SFTP показан ниже:

void DownloadPlogs()
    {
        try
        {
            SftpClient SFTP;
            if (GV.UseCustomPort && GV.CustomPort > 0 && GV.CustomPort < 65535)
            {
                SFTP = new SftpClient(GV.IpAddress, GV.CustomPort, GV.Username, GV.Password);
            }
            else
            {
                SFTP = new SftpClient(GV.IpAddress, 22, GV.Username, "");
            }
            SFTP.Connect();

            DownloadDirectory(SFTP, "/PLOG", Directory.GetCurrentDirectory() + @"\PLOG");
            ZipFile.CreateFromDirectory("PLOG", String.Format("{0} - {1} PLOGS.zip", GV.IpAddress, DateTime.Now.ToString("yyyyMMddHHmmss")));
            Directory.Delete(Directory.GetCurrentDirectory() + @"\PLOG", true);

            SFTP.Disconnect();
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.Message, "Error Getting PLOGS");
        }
    }

    void DownloadDirectory(SftpClient Client, string Source, string Destination)
    {
        var Files = Client.ListDirectory(Source);
        foreach (var File in Files)
        {
            if (!File.IsDirectory && !File.IsSymbolicLink)
            {
                DownloadFile(Client, File, Destination);
            }
            else if (File.IsSymbolicLink)
            {
                //Ignore
            }
            else if (File.Name != "." && File.Name != "..")
            {
                var Dir = Directory.CreateDirectory(System.IO.Path.Combine(Destination, File.Name));
                DownloadDirectory(Client, File.FullName, Dir.FullName);
            }
        }
    }

    void DownloadFile(SftpClient Client, Renci.SshNet.Sftp.SftpFile File, string Directory)
    {
        using (Stream FileStream = System.IO.File.OpenWrite(System.IO.Path.Combine(Directory, File.Name)))
        {
            Client.DownloadFile(File.FullName, FileStream);
        }
    }

Код для SSH ниже:

public SshConnection(string Host, int Port, string Username, string Password)
    {
        myClient = new SshClient(Host, Port, Username, Password);
        myClient.KeepAliveInterval = new TimeSpan(0, 0, 5);
        myClient.HostKeyReceived += myClient_HostKeyReceived;
        myClient.ErrorOccurred += myClient_ErrorOccurred;
    }

void myClient_ErrorOccurred(object sender, Renci.SshNet.Common.ExceptionEventArgs e)
    {
        MessageBox.Show(e.Exception.Message, "SSH Error Occurred");
    }

    void myClient_HostKeyReceived(object sender, Renci.SshNet.Common.HostKeyEventArgs e)
    {
        e.CanTrust = true;
    }

    public async void Connect()
    {
        Task T = new Task(() =>
        {
            try
            {
                myClient.Connect();
            }
            catch (System.Net.Sockets.SocketException)
            {
                MessageBox.Show("Invalid IP Address or Hostname", "SSH Connection Error");
            }
            catch (Renci.SshNet.Common.SshAuthenticationException ex)
            {
                MessageBox.Show(ex.Message, "SSH Authentication Error");
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.StackTrace, ex.Message);
                MessageBox.Show(ex.GetType().ToString());
                OnConnection(this, new ConnectEventArgs(myClient.IsConnected));
            }
        });

        T.Start();
        await T;
        if (T.IsCompleted)
        {
            OnConnection(this, new ConnectEventArgs(myClient.IsConnected));
        }
    }

    public void Disconnect()
    {
        try
        {
            myClient.Disconnect();
            OnConnection(this, new ConnectEventArgs(myClient.IsConnected));
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.StackTrace, ex.Message);
        }
    }

    public void SendData(string Data)
    {
        try
        {
            if (Data.EndsWith("\r\n"))
            {
                RunCommandAsync(Data, SshCommandRx);
            }
            else
            {
                RunCommandAsync(String.Format("{0}\r\n",Data), SshCommandRx);
            }
            //SshCommand Command = myClient.RunCommand(Data);
            //OnDataReceived(this, new DataEventArgs(Command.Result));
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.StackTrace, ex.Message);
        }
    }

    private async void RunCommandAsync(String Data, SshCommandCallback Callback)
    {
        Task<SshCommand> T = new Task<SshCommand>(() =>
        {
            try
            {
                SshCommand Command = myClient.RunCommand(Data);
                return Command;
            }

            catch (Exception ex)
            {
                MessageBox.Show(ex.Message, ex.GetType().ToString());
                return null;
            }
        });

        T.Start();
        await T;
        if (T.IsCompleted)
        {
            Callback(this, T.Result);
        }
    }

    private void SshCommandRx(SshConnection C, SshCommand Command)
    {
        if (Command != null)
        {
            string Rx = Command.Result;
            //if (Rx.StartsWith(Command.CommandText))
            //{
            //    Rx = Rx.Remove(0, Command.CommandText.Length);
            //}
            while (Rx.EndsWith("\r\n\r\n") == false)
            {
                Rx += "\r\n";
            }
            OnDataReceived(this, new DataEventArgs(Rx));
        }
    }

person Chris    schedule 26.08.2017    source источник
comment
проволочная акула. Если это не сработает до того, как ваше зашифрованное SSH-соединение будет установлено, это определенно будет понятно в wirehark.   -  person Marcus Müller    schedule 26.08.2017
comment
Согласно исходному коду renci эта ошибка возникает, если он не получает строку версии с сервера. Я предполагаю, что вы получаете это или, возможно, это.   -  person Kenster    schedule 27.08.2017
comment
Одно и то же приложение может без проблем подключаться через SSH и выполнять команды, проблема заключается только в SFTP-подключении. — На том же сервере? Или другой сервер для SSH и SFTP?   -  person Martin Prikryl    schedule 27.08.2017
comment
@MartinPrikryl Да, это против того же сервера. Я также могу без проблем подключиться к двум другим клиентским машинам с одним и тем же сервером, поэтому я думаю, что это что-то специфичное для клиентской машины.   -  person Chris    schedule 27.08.2017
comment
Это не имеет никакого смысла. Потому что ваше SFTP-соединение терпит неудачу в какой-то момент, когда создается только SSH-соединение, а SFTP еще не задействовано. Покажите нам свой код, как для SSH, так и для SFTP.   -  person Martin Prikryl    schedule 27.08.2017
comment
@MartinPrikryl Вы правы, что это не имеет никакого смысла. Это довольно сбивает с толку. Я обновил, чтобы включить код. Единственное, что я заметил, это то, что я не регистрируюсь на событие HostKeyReceived для SFTP. Однако это не требуется, поскольку на других машинах он отлично работает без него.   -  person Chris    schedule 28.08.2017
comment
минимальный воспроизводимый пример, пожалуйста!   -  person Martin Prikryl    schedule 28.08.2017
comment
Я извиняюсь @MartinPrikryl. Я новичок на сайте и думал, что предоставил минимальные, полные и поддающиеся проверке сведения. Что-то пропало?   -  person Chris    schedule 28.08.2017
comment
1) Большая часть кода неактуальна. Нас интересует только часть соединения. 2) Вы не показали нам, как вы создаете myClient. 3) Нам нужны реальные примеры, а не общий код вроде new SftpClient(GV.IpAddress, GV.CustomPort, GV.Username, GV.Password);   -  person Martin Prikryl    schedule 28.08.2017
comment
Добавлено создание myClient. Что касается GV.IpAddress и т. д., это фактический код. У меня просто есть поля WPF, привязанные к классу, содержащему эти данные.   -  person Chris    schedule 28.08.2017
comment
И именно это делает его неполным примером. Прочтите статью по ссылке в моем предыдущем комментарии.   -  person Martin Prikryl    schedule 28.08.2017


Ответы (2)


Я решаю это для себя только с повторными попытками подключения. Не нашел, в чем именно проблема, но эта проблема с подключением возникала много раз.

Пример:

int attempts = 0;
            do
            {
                try
                {
                    client.Connect();
                }
                catch (Renci.SshNet.Common.SshConnectionException e)
                {
                    attempts++;
                }
            } while (attempts < _connectiontRetryAttempts && !client.IsConnected);
person grinay    schedule 20.02.2018
comment
Если это любой другой тип исключения, завершится ли когда-нибудь do-while? - person Brett Bim; 04.03.2019
comment
Да. Если в этом конкретном фрагменте кода произойдет какое-либо другое исключение, выполнение do while будет прервано. Вы должны позаботиться и о других исключениях. - person grinay; 05.03.2019

Я столкнулся с таким же странным сообщением об ошибке при попытке подключения к SFTP-серверу при использовании библиотеки SSH.NET в программе на сервере. Проблема не появлялась при тестировании с моей машины для разработки.

Решение состояло в том, чтобы наша серверная команда добавила IP-адрес сервера в файл hosts.allow на SFTP-сервере Linux.

person ThePeter    schedule 18.09.2018
comment
У меня возникла аналогичная проблема. У поставщика есть белый список IP-адресов, к которым можно подключиться. Эта ошибка была вызвана отключением VPN; однако он отлично работал через VPN. - person David Yates; 28.04.2020