web-dev-qa-db-ja.com

C#FTPを介してすべてのファイルとサブディレクトリをダウンロードする

一般情報
私はまだC#を学習しています。自分を助けるために、すべてのローカルプロジェクトをFTPサーバー上のフォルダーと自動的に同期するプログラムを作成しようとしています。これにより、学校でも自宅でも、常に同じプロジェクトを利用できます。

Dropboxのようなプログラムが既に私のためにこれを行っていることは知っていますが、そのようなものを作成することで、その過程で多くのことを教えてくれると思いました。

問題
目標への私の最初のステップは、FTPサーバーからすべてのファイル、サブディレクトリ、およびサブファイルをダウンロードすることでした。以下のコードを使用して、ディレクトリからすべてのファイルをダウンロードすることができました。ただし、私のコードには、メインディレクトリ内のフォルダー名とファイルのみがリストされています。サブフォルダーとサブファイルが返されることもダウンロードされることもありません。それとは別に、フォルダをファイルのようにダウンロードしようとしているため、サーバーは550エラーを返します。私はこれを4時間以上続けていますが、これらの問題を修正して機能させる方法については何も見つかりません。そのため、皆さんが私を助けてくれることを期待しています:)

コード

public string[] GetFileList()
{
    string[] downloadFiles;
    StringBuilder result = new StringBuilder();
    WebResponse response = null;
    StreamReader reader = null;

    try
    {
        FtpWebRequest request = (FtpWebRequest)WebRequest.Create(url);
        request.UseBinary = true;
        request.Method = WebRequestMethods.Ftp.ListDirectory;
        request.Credentials = new NetworkCredential(ftpUserName, ftpPassWord);
        request.KeepAlive = false;
        request.UsePassive = false;
        response = request.GetResponse();
        reader = new StreamReader(response.GetResponseStream());
        string line = reader.ReadLine();
        while (line != null)
        {
            result.Append(line);
            result.Append("\n");
            line = reader.ReadLine();
        }
        result.Remove(result.ToString().LastIndexOf('\n'), 1);
        return result.ToString().Split('\n');
    }
    catch (Exception ex)
    {
        if (reader != null)
        {
            reader.Close();
        }
        if (response != null)
        {
            response.Close();
        }
        downloadFiles = null;
        return downloadFiles;
    }
}

private void Download(string file)
{
    try
    {
        string uri = url + "/" + file;
        Uri serverUri = new Uri(uri);
        if (serverUri.Scheme != Uri.UriSchemeFtp)
        {
            return;
        }
        FtpWebRequest request = (FtpWebRequest)WebRequest.Create(url + "/" + file);
        request.UseBinary = true;
        request.Method = WebRequestMethods.Ftp.DownloadFile;
        request.Credentials = new NetworkCredential(ftpUserName, ftpPassWord);
        request.KeepAlive = false;
        request.UsePassive = false;
        FtpWebResponse response = (FtpWebResponse)request.GetResponse();
        Stream responseStream = response.GetResponseStream();
        FileStream writeStream = new FileStream(localDestnDir + "\\" + file, FileMode.Create);                
        int Length = 2048;
        Byte[] buffer = new Byte[Length];
        int bytesRead = responseStream.Read(buffer, 0, Length);
        while (bytesRead > 0)
        {
            writeStream.Write(buffer, 0, bytesRead);
            bytesRead = responseStream.Read(buffer, 0, Length);
        }
        writeStream.Close();
        response.Close();
    }
    catch (WebException wEx)
    {
        MessageBox.Show(wEx.Message, "Download Error");
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.Message, "Download Error");
    }
}
16
icecub

FtpWebRequestは、再帰的なファイル操作(ダウンロードを含む)を明示的にサポートしていません。再帰を自分で実装する必要があります。

  • リモートディレクトリを一覧表示する
  • エントリを繰り返し、ファイルをダウンロードし、サブディレクトリに再帰します(再度リストするなど)。

トリッキーな部分は、サブディレクトリからファイルを識別することです。 FtpWebRequestを使用して移植可能な方法でそれを行う方法はありません。残念ながらFtpWebRequestMLSDコマンドをサポートしていません。これはFTPプロトコルのファイル属性でディレクトリ一覧を取得する唯一のポータブルな方法です。 FTPサーバー上のオブジェクトがファイルかディレクトリかを確認する も参照してください。

オプションは次のとおりです。

  • ファイルに対して失敗することが確実で、ディレクトリに対して成功する(またはその逆)ファイル名に対して操作を実行します。つまり「名前」をダウンロードしてみてください。それが成功した場合、それはファイルであり、それが失敗した場合、それはディレクトリです。
  • あなたは幸運かもしれませんし、特定のケースでは、ファイル名でディレクトリからファイルを伝えることができます(つまり、すべてのファイルには拡張子がありますが、サブディレクトリにはありません)
  • 長いディレクトリリスト(LIST command = ListDirectoryDetailsメソッド)を使用して、サーバー固有のリストを解析しようとします。多くのFTPサーバーは* nixスタイルのリストを使用します。このリストでは、エントリの最初にあるdでディレクトリを識別します。しかし、多くのサーバーは異なる形式を使用しています。次の例では、このアプローチを使用しています(* nix形式を想定)
void DownloadFtpDirectory(string url, NetworkCredential credentials, string localPath)
{
    FtpWebRequest listRequest = (FtpWebRequest)WebRequest.Create(url);
    listRequest.Method = WebRequestMethods.Ftp.ListDirectoryDetails;
    listRequest.Credentials = credentials;

    List<string> lines = new List<string>();

    using (FtpWebResponse listResponse = (FtpWebResponse)listRequest.GetResponse())
    using (Stream listStream = listResponse.GetResponseStream())
    using (StreamReader listReader = new StreamReader(listStream))
    {
        while (!listReader.EndOfStream)
        {
            lines.Add(listReader.ReadLine());
        }
    }

    foreach (string line in lines)
    {
        string[] tokens =
            line.Split(new[] { ' ' }, 9, StringSplitOptions.RemoveEmptyEntries);
        string name = tokens[8];
        string permissions = tokens[0];

        string localFilePath = Path.Combine(localPath, name);
        string fileUrl = url + name;

        if (permissions[0] == 'd')
        {
            if (!Directory.Exists(localFilePath))
            {
                Directory.CreateDirectory(localFilePath);
            }

            DownloadFtpDirectory(fileUrl + "/", credentials, localFilePath);
        }
        else
        {
            FtpWebRequest downloadRequest = (FtpWebRequest)WebRequest.Create(fileUrl);
            downloadRequest.Method = WebRequestMethods.Ftp.DownloadFile;
            downloadRequest.Credentials = credentials;

            using (FtpWebResponse downloadResponse =
                      (FtpWebResponse)downloadRequest.GetResponse())
            using (Stream sourceStream = downloadResponse.GetResponseStream())
            using (Stream targetStream = File.Create(localFilePath))
            {
                byte[] buffer = new byte[10240];
                int read;
                while ((read = sourceStream.Read(buffer, 0, buffer.Length)) > 0)
                {
                    targetStream.Write(buffer, 0, read);
                }
            }
        }
    }
}

次のような関数を使用します。

NetworkCredential credentials = new NetworkCredential("user", "mypassword");
string url = "ftp://ftp.example.com/directory/to/download/";
DownloadFtpDirectory(url, credentials, @"C:\target\directory");

サーバー固有のディレクトリリスト形式の解析に関する問題を回避する場合は、MLSDコマンドをサポートするサードパーティライブラリを使用するか、さまざまなLISTリスト形式の解析を行います。再帰的なダウンロード。

たとえば、 WinSCP .NET Assembly を使用すると、 Session.GetFiles

// Setup session options
SessionOptions sessionOptions = new SessionOptions
{
    Protocol = Protocol.Ftp,
    HostName = "ftp.example.com",
    UserName = "user",
    Password = "mypassword",
};

using (Session session = new Session())
{
    // Connect
    session.Open(sessionOptions);

    // Download files
    session.GetFiles("/directory/to/download/*", @"C:\target\directory\*").Check();
}

サーバーでサポートされている場合、内部的に、WinSCPはMLSDコマンドを使用します。そうでない場合は、LISTコマンドを使用し、多数の異なるリスト形式をサポートします。

Session.GetFiles method はデフォルトで再帰的です。

(私はWinSCPの著者です)

32
Martin Prikryl