Как получить файл с сервера через SFTP?

Я пытаюсь получить файл с сервера, используя SFTP (в отличие от FTPS), используя Java. Как я могу это сделать?

18.08.2008 13:43:48
16 ОТВЕТОВ
РЕШЕНИЕ

Другой вариант - рассмотреть библиотеку JSch . JSch кажется предпочтительной библиотекой для нескольких крупных проектов с открытым исходным кодом, включая Eclipse, Ant и Apache Commons HttpClient, среди прочих.

Он поддерживает как логин пользователя / прохода, так и логин на основе сертификатов, а также целый ряд других вкусных функций SSH2.

Вот простое удаленное получение файла через SFTP. Обработка ошибок оставлена ​​в качестве упражнения для читателя :-)

JSch jsch = new JSch();

String knownHostsFilename = "/home/username/.ssh/known_hosts";
jsch.setKnownHosts( knownHostsFilename );

Session session = jsch.getSession( "remote-username", "remote-host" );    
{
  // "interactive" version
  // can selectively update specified known_hosts file 
  // need to implement UserInfo interface
  // MyUserInfo is a swing implementation provided in 
  //  examples/Sftp.java in the JSch dist
  UserInfo ui = new MyUserInfo();
  session.setUserInfo(ui);

  // OR non-interactive version. Relies in host key being in known-hosts file
  session.setPassword( "remote-password" );
}

session.connect();

Channel channel = session.openChannel( "sftp" );
channel.connect();

ChannelSftp sftpChannel = (ChannelSftp) channel;

sftpChannel.get("remote-file", "local-file" );
// OR
InputStream in = sftpChannel.get( "remote-file" );
  // process inputstream as needed

sftpChannel.exit();
session.disconnect();
198
21.08.2008 09:48:25
Cheekysoft, я заметил - при использовании Jsch - удаление файлов на sftp-сервере не работает. Также не работает переименование файлов. Любые идеи, пожалуйста ??? Andy
user283062 31.01.2011 01:24:33
Извините, это не то, с чем я сейчас работаю. (Пожалуйста, попробуйте оставить такие ответы в виде комментариев - например, это сообщение - а не как новый ответ на исходный вопрос)
Cheekysoft 31.01.2011 01:24:33
Что это за блок кода после назначения сеанса? Это какой-то причудливый синтаксис Java, который я никогда не видел? Если так - что это делает, будучи написанным таким образом?
Michael Peterson 17.10.2014 14:19:30
@ p1x3l5 стандартный синтаксис Java позволяет вставлять блок где угодно; при желании его можно использовать для более точного управления областью действия переменной. Однако в этом случае это всего лишь наглядное пособие, помогающее указать два варианта реализации: либо использовать интерактивную версию, которая запрашивает пароль у пользователя, либо использовать жестко заданный пароль, не требующий вмешательства пользователя, но, возможно, дополнительный риск для безопасности.
Cheekysoft 5.11.2014 10:37:28

Это было решение, которое я придумал http://sourceforge.net/projects/sshtools/ (большая часть обработки ошибок для ясности опущена). Это выдержка из моего блога

SshClient ssh = new SshClient();
ssh.connect(host, port);
//Authenticate
PasswordAuthenticationClient passwordAuthenticationClient = new PasswordAuthenticationClient();
passwordAuthenticationClient.setUsername(userName);
passwordAuthenticationClient.setPassword(password);
int result = ssh.authenticate(passwordAuthenticationClient);
if(result != AuthenticationProtocolState.COMPLETE){
     throw new SFTPException("Login to " + host + ":" + port + " " + userName + "/" + password + " failed");
}
//Open the SFTP channel
SftpClient client = ssh.openSftpClient();
//Send the file
client.put(filePath);
//disconnect
client.quit();
ssh.disconnect();
23
18.08.2008 13:49:13
Я согласен (с опозданием), он работал нормально для оригинального сайта / загрузки, которую я требовал, но отказался работать для нового. Я нахожусь в процессе перехода на JSch
David Hayes 27.01.2011 14:09:09

Хорошая абстракция поверх Jsch - это Apache commons-vfs, который предлагает API виртуальной файловой системы, который делает доступ к SFTP-файлам и их запись практически прозрачным. Хорошо сработало для нас.

23
19.08.2008 20:12:51
Можно ли использовать предварительные общие ключи в сочетании с commons-vfs?
Benedikt Waldvogel 15.05.2009 13:46:16
Да это так. Если вам нужны нестандартные удостоверения, вы можете вызвать SftpFileSystemConfigBuilder.getInstance (). SetIdentities (...).
Russ Hayward 27.01.2011 11:05:54
Вы можете использовать предварительно общие ключи. Но эти ключи должны быть без пароля. OtrosLogViewer использует SSH-авторизацию ключа с VFS, но требует удалить ключевую фразу из ключа ( code.google.com/p/otroslogviewer/wiki/SftpAuthPubKey )
KrzyH 18.12.2012 13:05:07

Лучшее решение, которое я нашел, это Paramiko . Там есть версия на Java.

0
11.01.2009 15:02:20
github.com/terencehonles/jaramiko заброшен в пользу JSch (см. уведомление на github).
rü- 6.10.2015 15:04:20

У вас также есть JFileUpload с надстройкой SFTP (Java тоже): http://www.jfileupload.com/products/sftp/index.html

1
14.02.2009 11:38:05
JFileUpload - это апплет, а не библиотека. Лицензия коммерческая. Не выглядит активным, либо.
rü- 6.10.2015 15:06:38

Попробуйте edtFTPj / PRO , зрелую, надежную клиентскую библиотеку SFTP, которая поддерживает пулы соединений и асинхронные операции. Также поддерживает FTP и FTPS, так что все основы для безопасной передачи файлов покрыты.

2
1.06.2009 04:42:40

Я использую этот SFTP API под названием Zehon, он великолепен, поэтому его легко использовать с большим количеством примеров кода. Вот сайт http://www.zehon.com

1
18.06.2009 08:10:55
Зехон кажется мертвым. И где источник? Какая «лицензия» стоит за «бесплатной»?
rü- 6.10.2015 15:00:23

hierynomus / sshj имеет полную реализацию SFTP версии 3 (что реализует OpenSSH)

Пример кода из SFTPUpload.java

package net.schmizz.sshj.examples;

import net.schmizz.sshj.SSHClient;
import net.schmizz.sshj.sftp.SFTPClient;
import net.schmizz.sshj.xfer.FileSystemFile;

import java.io.File;
import java.io.IOException;

/** This example demonstrates uploading of a file over SFTP to the SSH server. */
public class SFTPUpload {

    public static void main(String[] args)
            throws IOException {
        final SSHClient ssh = new SSHClient();
        ssh.loadKnownHosts();
        ssh.connect("localhost");
        try {
            ssh.authPublickey(System.getProperty("user.name"));
            final String src = System.getProperty("user.home") + File.separator + "test_file";
            final SFTPClient sftp = ssh.newSFTPClient();
            try {
                sftp.put(new FileSystemFile(src), "/tmp");
            } finally {
                sftp.close();
            }
        } finally {
            ssh.disconnect();
        }
    }

}
7
3.11.2016 15:26:47
хорошая работа!! хотя пример на главной странице может быть полезным.
OhadR 4.05.2015 12:28:02

Ниже приведен пример использования Apache Common VFS:

FileSystemOptions fsOptions = new FileSystemOptions();
SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(fsOptions, "no");
FileSystemManager fsManager = VFS.getManager();
String uri = "sftp://user:password@host:port/absolute-path";
FileObject fo = fsManager.resolveFile(uri, fsOptions);
43
30.03.2010 20:46:43
Еще одна полезная вещь - установить время ожидания, чтобы, если удаленная система была отключена, вы не зависали там вечно. Вы можете сделать это так же, как это было сделано для отключения проверки ключа хоста: SftpFileSystemConfigBuilder.getInstance (). SetTimeout (fsOptions, 5000);
Scott Jones 6.09.2012 13:35:23
Как бы вы посоветовали закрыть это соединение при использовании нескольких клиентов SFTP одновременно?
2Big2BeSmall 7.08.2016 06:23:57
Что если мой пароль содержит символ @?
User3 22.06.2017 12:29:31

Вот полный исходный код примера, использующего JSch, не беспокоясь о проверке ключа ssh.

import com.jcraft.jsch.*;

public class TestJSch {
    public static void main(String args[]) {
        JSch jsch = new JSch();
        Session session = null;
        try {
            session = jsch.getSession("username", "127.0.0.1", 22);
            session.setConfig("StrictHostKeyChecking", "no");
            session.setPassword("password");
            session.connect();

            Channel channel = session.openChannel("sftp");
            channel.connect();
            ChannelSftp sftpChannel = (ChannelSftp) channel;
            sftpChannel.get("remotefile.txt", "localfile.txt");
            sftpChannel.exit();
            session.disconnect();
        } catch (JSchException e) {
            e.printStackTrace();  
        } catch (SftpException e) {
            e.printStackTrace();
        }
    }
}
109
18.02.2013 11:40:16
finallyБлок должен использоваться для включения канала код очистки, чтобы убедиться , что он всегда работает.
hotshot309 18.11.2013 17:47:46
Я получаю это исключение сейчас: com.jcraft.jsch.JSchException: Session.connect: java.security.InvalidAlgorithmParameterException: Prime size must be multiple of 64, and can only range from 512 to 2048 (inclusive)
anon58192932 19.01.2016 19:07:30
Я обнаружил, что у JSCH есть 0 или 1 дополнительная зависимость. Вы можете игнорировать зависимость JZLIB, если отключите сжатие. // отключить сжатие session.setConfig ("compress.s2c", "none"); session.setConfig ("compress.c2s", "none");
englebart 1.02.2019 19:19:32
Без строгой проверки хоста вы подвержены атаке «человек посередине».
rustyx 13.03.2019 14:16:18

Я нашел полный рабочий пример для SFTP в Java, используя JSCH API http://kodehelp.com/java-program-for-uploading-file-to-sftp-server/

2
1.10.2012 18:04:58

Энди, чтобы удалить файл в удаленной системе, вам нужно использовать (channelExec)JSch и передать команды unix / linux для его удаления.

3
12.11.2015 09:20:49

Хотя приведенные выше ответы были очень полезны, я потратил целый день, чтобы заставить их работать, сталкиваясь с различными исключениями, такими как «неработающий канал», «ключ rsa неизвестен» и «поврежден пакет».

Ниже приведен рабочий класс многоразового использования для загрузки / выгрузки файлов SFTP с использованием библиотеки JSch.

Загрузить использование:

SFTPFileCopy upload = new SFTPFileCopy(true, /path/to/sourcefile.png", /path/to/destinationfile.png");

Скачать использование:

SFTPFileCopy download = new SFTPFileCopy(false, "/path/to/sourcefile.png", "/path/to/destinationfile.png");

Код класса:

import com.jcraft.jsch.Channel;
import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.Session;
import com.jcraft.jsch.UIKeyboardInteractive;
import com.jcraft.jsch.UserInfo;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import javax.swing.JOptionPane;
import menue.Menue;

public class SFTPFileCopy1 {

    public SFTPFileCopy1(boolean upload, String sourcePath, String destPath) throws FileNotFoundException, IOException {
        Session session = null;
        Channel channel = null;
        ChannelSftp sftpChannel = null;
        try {
            JSch jsch = new JSch();
            //jsch.setKnownHosts("/home/user/.putty/sshhostkeys");
            session = jsch.getSession("login", "mysite.com", 22);
            session.setPassword("password");

            UserInfo ui = new MyUserInfo() {
                public void showMessage(String message) {

                    JOptionPane.showMessageDialog(null, message);

                }

                public boolean promptYesNo(String message) {

                    Object[] options = {"yes", "no"};

                    int foo = JOptionPane.showOptionDialog(null,
                            message,
                            "Warning",
                            JOptionPane.DEFAULT_OPTION,
                            JOptionPane.WARNING_MESSAGE,
                            null, options, options[0]);

                    return foo == 0;

                }
            };
            session.setUserInfo(ui);

            session.setConfig("StrictHostKeyChecking", "no");
            session.connect();
            channel = session.openChannel("sftp");
            channel.setInputStream(System.in);
            channel.setOutputStream(System.out);
            channel.connect();
            sftpChannel = (ChannelSftp) channel;

            if (upload) { // File upload.
                byte[] bufr = new byte[(int) new File(sourcePath).length()];
                FileInputStream fis = new FileInputStream(new File(sourcePath));
                fis.read(bufr);
                ByteArrayInputStream fileStream = new ByteArrayInputStream(bufr);
                sftpChannel.put(fileStream, destPath);
                fileStream.close();
            } else { // File download.
                byte[] buffer = new byte[1024];
                BufferedInputStream bis = new BufferedInputStream(sftpChannel.get(sourcePath));
                OutputStream os = new FileOutputStream(new File(destPath));
                BufferedOutputStream bos = new BufferedOutputStream(os);
                int readCount;
                while ((readCount = bis.read(buffer)) > 0) {
                    bos.write(buffer, 0, readCount);
                }
                bis.close();
                bos.close();
            }
        } catch (Exception e) {
            System.out.println(e);
        } finally {
            if (sftpChannel != null) {
                sftpChannel.exit();
            }
            if (channel != null) {
                channel.disconnect();
            }
            if (session != null) {
                session.disconnect();
            }
        }
    }

    public static abstract class MyUserInfo
            implements UserInfo, UIKeyboardInteractive {

        public String getPassword() {
            return null;
        }

        public boolean promptYesNo(String str) {
            return false;
        }

        public String getPassphrase() {
            return null;
        }

        public boolean promptPassphrase(String message) {
            return false;
        }

        public boolean promptPassword(String message) {
            return false;
        }

        public void showMessage(String message) {
        }

        public String[] promptKeyboardInteractive(String destination,
                String name,
                String instruction,
                String[] prompt,
                boolean[] echo) {

            return null;
        }
    }
}
2
1.08.2013 22:28:42

Библиотека Apache Commons SFTP

Файл общих свойств java для всех примеров

АдресСерверы = 111.222.333.444

= myUserId идентификатор пользователя

пароль = MyPassword

remoteDirectory = продукты /

localDirectory = импорт /

Загрузить файл на удаленный сервер с использованием SFTP

import java.io.File;
import java.io.FileInputStream;
import java.util.Properties;

import org.apache.commons.vfs2.FileObject;
import org.apache.commons.vfs2.FileSystemOptions;
import org.apache.commons.vfs2.Selectors;
import org.apache.commons.vfs2.impl.StandardFileSystemManager;
import org.apache.commons.vfs2.provider.sftp.SftpFileSystemConfigBuilder;

public class SendMyFiles {

 static Properties props;

 public static void main(String[] args) {

  SendMyFiles sendMyFiles = new SendMyFiles();
  if (args.length < 1)
  {
   System.err.println("Usage: java " + sendMyFiles.getClass().getName()+
     " Properties_file File_To_FTP ");
   System.exit(1);
  }

  String propertiesFile = args[0].trim();
  String fileToFTP = args[1].trim();
  sendMyFiles.startFTP(propertiesFile, fileToFTP);

 }

 public boolean startFTP(String propertiesFilename, String fileToFTP){

  props = new Properties();
  StandardFileSystemManager manager = new StandardFileSystemManager();

  try {

   props.load(new FileInputStream("properties/" + propertiesFilename));
   String serverAddress = props.getProperty("serverAddress").trim();
   String userId = props.getProperty("userId").trim();
   String password = props.getProperty("password").trim();
   String remoteDirectory = props.getProperty("remoteDirectory").trim();
   String localDirectory = props.getProperty("localDirectory").trim();

   //check if the file exists
   String filepath = localDirectory +  fileToFTP;
   File file = new File(filepath);
   if (!file.exists())
    throw new RuntimeException("Error. Local file not found");

   //Initializes the file manager
   manager.init();

   //Setup our SFTP configuration
   FileSystemOptions opts = new FileSystemOptions();
   SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(
     opts, "no");
   SftpFileSystemConfigBuilder.getInstance().setUserDirIsRoot(opts, true);
   SftpFileSystemConfigBuilder.getInstance().setTimeout(opts, 10000);

   //Create the SFTP URI using the host name, userid, password,  remote path and file name
   String sftpUri = "sftp://" + userId + ":" + password +  "@" + serverAddress + "/" + 
     remoteDirectory + fileToFTP;

   // Create local file object
   FileObject localFile = manager.resolveFile(file.getAbsolutePath());

   // Create remote file object
   FileObject remoteFile = manager.resolveFile(sftpUri, opts);

   // Copy local file to sftp server
   remoteFile.copyFrom(localFile, Selectors.SELECT_SELF);
   System.out.println("File upload successful");

  }
  catch (Exception ex) {
   ex.printStackTrace();
   return false;
  }
  finally {
   manager.close();
  }

  return true;
 }


}

Скачать файл с удаленного сервера, используя SFTP

import java.io.File;
import java.io.FileInputStream;
import java.util.Properties;

import org.apache.commons.vfs2.FileObject;
import org.apache.commons.vfs2.FileSystemOptions;
import org.apache.commons.vfs2.Selectors;
import org.apache.commons.vfs2.impl.StandardFileSystemManager;
import org.apache.commons.vfs2.provider.sftp.SftpFileSystemConfigBuilder;

public class GetMyFiles {

 static Properties props;

 public static void main(String[] args) {

  GetMyFiles getMyFiles = new GetMyFiles();
  if (args.length < 1)
  {
   System.err.println("Usage: java " + getMyFiles.getClass().getName()+
   " Properties_filename File_To_Download ");
   System.exit(1);
  }

  String propertiesFilename = args[0].trim();
  String fileToDownload = args[1].trim();
  getMyFiles.startFTP(propertiesFilename, fileToDownload);

 }

 public boolean startFTP(String propertiesFilename, String fileToDownload){

  props = new Properties();
  StandardFileSystemManager manager = new StandardFileSystemManager();

  try {

   props.load(new FileInputStream("properties/" + propertiesFilename));
   String serverAddress = props.getProperty("serverAddress").trim();
   String userId = props.getProperty("userId").trim();
   String password = props.getProperty("password").trim();
   String remoteDirectory = props.getProperty("remoteDirectory").trim();
   String localDirectory = props.getProperty("localDirectory").trim();


   //Initializes the file manager
   manager.init();

   //Setup our SFTP configuration
   FileSystemOptions opts = new FileSystemOptions();
   SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(
     opts, "no");
   SftpFileSystemConfigBuilder.getInstance().setUserDirIsRoot(opts, true);
   SftpFileSystemConfigBuilder.getInstance().setTimeout(opts, 10000);

   //Create the SFTP URI using the host name, userid, password,  remote path and file name
   String sftpUri = "sftp://" + userId + ":" + password +  "@" + serverAddress + "/" + 
     remoteDirectory + fileToDownload;

   // Create local file object
   String filepath = localDirectory +  fileToDownload;
   File file = new File(filepath);
   FileObject localFile = manager.resolveFile(file.getAbsolutePath());

   // Create remote file object
   FileObject remoteFile = manager.resolveFile(sftpUri, opts);

   // Copy local file to sftp server
   localFile.copyFrom(remoteFile, Selectors.SELECT_SELF);
   System.out.println("File download successful");

  }
  catch (Exception ex) {
   ex.printStackTrace();
   return false;
  }
  finally {
   manager.close();
  }

  return true;
 }

}

Удалить файл на удаленном сервере, используя SFTP

import java.io.FileInputStream;
import java.util.Properties;

import org.apache.commons.vfs2.FileObject;
import org.apache.commons.vfs2.FileSystemOptions;
import org.apache.commons.vfs2.impl.StandardFileSystemManager;
import org.apache.commons.vfs2.provider.sftp.SftpFileSystemConfigBuilder;

public class DeleteRemoteFile {

 static Properties props;

 public static void main(String[] args) {

  DeleteRemoteFile getMyFiles = new DeleteRemoteFile();
  if (args.length < 1)
  {
   System.err.println("Usage: java " + getMyFiles.getClass().getName()+
   " Properties_filename File_To_Delete ");
   System.exit(1);
  }

  String propertiesFilename = args[0].trim();
  String fileToDownload = args[1].trim();
  getMyFiles.startFTP(propertiesFilename, fileToDownload);

 }

 public boolean startFTP(String propertiesFilename, String fileToDownload){

  props = new Properties();
  StandardFileSystemManager manager = new StandardFileSystemManager();

  try {

   props.load(new FileInputStream("properties/" + propertiesFilename));
   String serverAddress = props.getProperty("serverAddress").trim();
   String userId = props.getProperty("userId").trim();
   String password = props.getProperty("password").trim();
   String remoteDirectory = props.getProperty("remoteDirectory").trim();


   //Initializes the file manager
   manager.init();

   //Setup our SFTP configuration
   FileSystemOptions opts = new FileSystemOptions();
   SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(
     opts, "no");
   SftpFileSystemConfigBuilder.getInstance().setUserDirIsRoot(opts, true);
   SftpFileSystemConfigBuilder.getInstance().setTimeout(opts, 10000);

   //Create the SFTP URI using the host name, userid, password,  remote path and file name
   String sftpUri = "sftp://" + userId + ":" + password +  "@" + serverAddress + "/" + 
     remoteDirectory + fileToDownload;

   //Create remote file object
   FileObject remoteFile = manager.resolveFile(sftpUri, opts);

   //Check if the file exists
   if(remoteFile.exists()){
    remoteFile.delete();
    System.out.println("File delete successful");
   }

  }
  catch (Exception ex) {
   ex.printStackTrace();
   return false;
  }
  finally {
   manager.close();
  }

  return true;
 }

}
13
24.09.2013 07:51:25
AZ_ 24.09.2013 07:51:35
как настроить при наличии ssh-ключа (открытого ключа) для копирования файлов на сервер. Потому что мне нужно сделать ssh_trust между моим сервером и удаленным сервером.
M S Parmar 14.08.2015 12:37:27

Есть хорошее сравнение трех зрелых библиотек Java для SFTP: Commons VFS, SSHJ и JSch

Подводя итог, можно сказать, что у SSHJ самый ясный API, и лучше всего, если вам не нужна поддержка других хранилищ, предоставляемая Commons VFS.

Вот отредактированный пример SSHJ из github :

final SSHClient ssh = new SSHClient();
ssh.loadKnownHosts(); // or, to skip host verification: ssh.addHostKeyVerifier(new PromiscuousVerifier())
ssh.connect("localhost");
try {
    ssh.authPassword("user", "password"); // or ssh.authPublickey(System.getProperty("user.name"))
    final SFTPClient sftp = ssh.newSFTPClient();
    try {
        sftp.get("test_file", "/tmp/test.tmp");
    } finally {
        sftp.close();
    }
} finally {
    ssh.disconnect();
}
19
6.04.2016 14:01:44
Есть ли способ получить файл как InputStream?
Johan 7.12.2018 05:46:16
sshj в 2019 году по-прежнему в хорошем состоянии и используется проектом Alpakka (Akka)
Maxence 24.01.2019 14:35:52

Библиотека JSch - это мощная библиотека, которую можно использовать для чтения файлов с SFTP-сервера. Ниже тестируемый код для чтения файла из SFTP-местоположения построчно

JSch jsch = new JSch();
        Session session = null;
        try {
            session = jsch.getSession("user", "127.0.0.1", 22);
            session.setConfig("StrictHostKeyChecking", "no");
            session.setPassword("password");
            session.connect();

            Channel channel = session.openChannel("sftp");
            channel.connect();
            ChannelSftp sftpChannel = (ChannelSftp) channel;

            InputStream stream = sftpChannel.get("/usr/home/testfile.txt");
            try {
                BufferedReader br = new BufferedReader(new InputStreamReader(stream));
                String line;
                while ((line = br.readLine()) != null) {
                    System.out.println(line);
                }

            } catch (IOException io) {
                System.out.println("Exception occurred during reading file from SFTP server due to " + io.getMessage());
                io.getMessage();

            } catch (Exception e) {
                System.out.println("Exception occurred during reading file from SFTP server due to " + e.getMessage());
                e.getMessage();

            }

            sftpChannel.exit();
            session.disconnect();
        } catch (JSchException e) {
            e.printStackTrace();
        } catch (SftpException e) {
            e.printStackTrace();
        }

Пожалуйста, обратитесь к блогу для всей программы.

4
4.10.2017 07:59:19