10

备注:由于垃圾邮件预防机制,我被迫将 Uris 的开头从 ftp:// 替换为 ftp。

我有以下问题。我必须使用 C# ftp 方法上传文件,然后重命名它。容易,对吧?:)

好的,假设我的 ftp 主机是这样的:

ftp.contoso.com

登录后,当前目录设置为:

用户/名称

所以,我想要实现的是登录,将文件作为 file.ext.tmp 上传到当前目录,上传成功后,将文件重命名为 file.ext

正如我猜想的那样,整个困难在于正确设置 FtpWebRequest 的请求 Uri。

MSDN 状态:

URI 可以是相对的或绝对的。如果 URI 的格式为“ ftp://contoso.com/%2fpath”(%2f是转义的“/”),则 URI 是绝对的,当前目录是 /path。但是,如果 URI 的格式为“ ftp://contoso.com/path ”,则 .NET Framework 首先登录 FTP 服务器(使用 Credentials 属性设置的用户名和密码),然后是当前目录设置为 UserLoginDirectory/路径。

好的,所以我使用以下 URI 上传文件:

ftp.contoso.com/file.ext.tmp

太好了,文件位于我想要的位置:在目录“用户/名称”中

现在,我想重命名文件,所以我使用以下 Uri 创建 Web 请求:

ftp.contoso.com/file.ext.tmp

并将参数重命名为:

文件.ext

这给了我 550 错误:找不到文件,没有权限等。

我在 Microsoft Network Monitor 中跟踪了这一点,它给了我:

命令:RNFR,从命令参数重命名
:/file.ext.tmp
Ftp:响应端口 53724,'550 文件 /file.ext.tmp 未找到'

就好像它在根目录中寻找文件 - 而不是在当前目录中。

我使用 Total Commander 手动重命名了文件,唯一的区别是 CommandParameter 没有第一个斜杠:

命令参数:file.ext.tmp

我可以通过提供以下绝对 URI 成功重命名文件:

ftp.contoso.com/%2fusers/%2fname/file.ext.tmp

但我不喜欢这种方法,因为我必须知道当前用户目录的名称。它可能可以通过使用 WebRequestMethods.Ftp.PrintWorkingDirectory 来完成,但它增加了额外的复杂性(调用此方法来检索目录名称,然后组合路径以形成正确的 URI)。

我不明白为什么 URI ftp.contoso.com/file.ext.tmp 适合上传而不适合重命名?我在这里错过了什么吗?

该项目设置为 .NET 4.0,在 Visual Studio 2010 中编码。

编辑

好的,我放置代码片段。

请注意,ftp 主机、用户名和密码应填写。要使此示例正常工作 - 即产生错误 - 用户目录必须与 root 不同(“pwd”-命令应该返回不同于“/”的内容)

class Program
{
    private const string fileName = "test.ext";
    private const string tempFileName = fileName + ".tmp";
    private const string ftpHost = "127.0.0.1";
    private const string ftpUserName = "anonymous";
    private const string ftpPassword = "";
    private const int bufferSize = 524288;

    static void Main(string[] args)
    {
        try
        {
            string path = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.MyDocuments), fileName);

            if (!File.Exists(path))
                File.WriteAllText(path, "FTP RENAME SAMPLE");

            string requestUri = "ftp://" + ftpHost + "/" + tempFileName;

            //upload

            FtpWebRequest uploadRequest = (FtpWebRequest)WebRequest.Create(requestUri);
            uploadRequest.UseBinary = true;
            uploadRequest.UsePassive = true;
            uploadRequest.Credentials = new NetworkCredential(ftpUserName, ftpPassword);
            uploadRequest.KeepAlive = true;
            uploadRequest.Method = WebRequestMethods.Ftp.UploadFile;

            Stream requestStream = null;
            FileStream localFileStream = null;


            localFileStream = File.OpenRead(path);
            requestStream = uploadRequest.GetRequestStream();
            byte[] buffer = new byte[bufferSize];

            int readCount = localFileStream.Read(buffer, 0, bufferSize);
            long bytesSentCounter = 0;

            while (readCount > 0)
            {
                requestStream.Write(buffer, 0, readCount);
                bytesSentCounter += readCount;
                readCount = localFileStream.Read(buffer, 0, bufferSize);
                System.Threading.Thread.Sleep(100);
            }

            localFileStream.Close();
            requestStream.Close();

            FtpWebResponse response = (FtpWebResponse)uploadRequest.GetResponse();
            FtpStatusCode code = response.StatusCode;
            string description = response.StatusDescription;
            response.Close();

            if (code == FtpStatusCode.ClosingData)
                Console.WriteLine("File uploaded successfully");

            //rename

            FtpWebRequest renameRequest = (FtpWebRequest)WebRequest.Create(requestUri);
            renameRequest.UseBinary = true;
            renameRequest.UsePassive = true;
            renameRequest.Credentials = new NetworkCredential(ftpUserName, ftpPassword);
            renameRequest.KeepAlive = true;
            renameRequest.Method = WebRequestMethods.Ftp.Rename;
            renameRequest.RenameTo = fileName;

            try
            {

                FtpWebResponse renameResponse = (FtpWebResponse)renameRequest.GetResponse();

                Console.WriteLine("Rename OK, status code: {0}, rename status description: {1}", response.StatusCode, response.StatusDescription);

                renameResponse.Close();
            }
            catch (WebException ex)
            {
                Console.WriteLine("Rename failed, status code: {0}, rename status description: {1}", ((FtpWebResponse)ex.Response).StatusCode, 
                    ((FtpWebResponse)ex.Response).StatusDescription);
            }

        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
        }
        finally
        {
            Console.ReadKey();
        }
    }
}
4

2 回答 2

8

我遇到了类似的问题。问题是 FtpWebRequest(错误地)在重命名请求之前添加了“/”,从这个日志(上传和重命名)可以看出:

URL: 
  http://127.0.0.1/Test.txt
FTP log:
  STOR Test.txt.part
  RNFR /Test.txt.part
  RNTO /Test.txt

请注意,仅当您上传到根目录时才会出现此问题。如果您将 URL 更改为http://127.0.0.1/path/Test.txt,则一切正常。

我对这个问题的解决方案是使用 %2E (dot) 作为路径:

URL:
  http://127.0.0.1/%2E/Test.txt
FTP log:
 STOR ./Test.txt.part
 RNFR ./Test.txt.part
 RNTO ./Test.txt

您必须对点进行 url 编码,否则 FtpWebRequest 会将路径“/./”简化为“/”。

于 2012-02-21T10:19:01.120 回答
4

C#

using System.Net;
using System.IO;

重命名 FTP 服务器功能上的文件名

C#

 private void RenameFileName(string currentFilename, string newFilename)
   {
       FTPSettings.IP = "DOMAIN NAME";
       FTPSettings.UserID = "USER ID";
       FTPSettings.Password = "PASSWORD";
       FtpWebRequest reqFTP = null;
       Stream ftpStream = null ;
       try
       {

           reqFTP = (FtpWebRequest)FtpWebRequest.Create(new Uri("ftp://" + FTPSettings.IP + "/" + currentFilename));
           reqFTP.Method = WebRequestMethods.Ftp.Rename;
           reqFTP.RenameTo = newFilename;
           reqFTP.UseBinary = true;
           reqFTP.Credentials = new NetworkCredential(FTPSettings.UserID, FTPSettings.Password);
           FtpWebResponse response = (FtpWebResponse)reqFTP.GetResponse();
           ftpStream = response.GetResponseStream();
           ftpStream.Close();
           response.Close();
       }
       catch (Exception ex)
       {
           if (ftpStream != null)
           {
               ftpStream.Close();
               ftpStream.Dispose();
           }
           throw new Exception(ex.Message.ToString());
       }
   }

   public static class FTPSettings
   {
       public static string IP { get; set; }
       public static string UserID { get; set; }
       public static string Password { get; set; }
   }
于 2011-05-21T07:32:03.367 回答