c# - 如何从通过 FtpCommand 运行的 FTP 命令获取结果?

标签 c# winapi ftp pinvoke

我已经毫无问题地包装了大部分 wininet,但现在我被卡住了。我正在尝试 p/调用 FtpCommand来自 wininet.dll,但我运行的每个命令都返回“500 语法错误”。即使是像 dir 或 ls 这样的简单命令。如果我使用 ftp.exe 连接到同一台服务器,则命令可以正常工作并返回预期结果。

方法定义如下:

[DllImport("wininet.dll", CharSet = CharSet.Auto, SetLastError=true)]
extern public static int FtpCommand(
    [In]  IntPtr hConnect,      
    [In]  bool fExpectResponse, 
    [In]  int dwFlags,          
    [In]  string command,
    [In]  IntPtr dwContext,
    [In][Out]  ref int ftpCommand);

以及我调用它的代码:

    public string SendCommand(string cmd)
    {
        int result;
        IntPtr dataSocket = new IntPtr();
        switch(cmd)
        {
            case "PASV":
                result = WININET.FtpCommand(_hConnect, false, WININET.FTP_TRANSFER_TYPE_ASCII, cmd, IntPtr.Zero, ref dataSocket);
                break;
            default:
                result = WININET.FtpCommand(_hConnect, true, WININET.FTP_TRANSFER_TYPE_ASCII, cmd, IntPtr.Zero, ref dataSocket);
                break;
        }

        Console.WriteLine(InternetLastResponseInfo());

        int BUFFER_SIZE = 8192;

        if(result == 0){
            Error();
        }
        else if(dataSocket != IntPtr.Zero)
        {
            StringBuilder buffer = new StringBuilder(BUFFER_SIZE);
            int bytesRead = 0;

            do
            {
                result = WININET.InternetReadFile(dataSocket, buffer, BUFFER_SIZE, ref bytesRead);
            } while (result == 1 && bytesRead > 1);

            return buffer.ToString();

        }

        return "";
    }

其他一切正常,我可以下载文件、上传文件并使用 FtpFindFirstFile() 模拟 dir 命令,但我似乎无法使用上述方法发送命令。

编辑

我的 SendCommand 代码只是 SendCommand("DIR") 或 SendCommand("LS")。在阅读其中一个答案(编辑时看不到谁)后,我将其更改为 SendCommand("LIST") 并成功返回。

但是,我的问题是如何读取结果,我用什么来读取从 LIST 命令返回的数据,以便我可以以可读的格式输出它?

我已经更新了 SendCommand 方法以显示我计划如何读取返回的数据,但我总是得到 0 for bytesRead。我还尝试在 FtpCommand 调用中传递 dataSocket 接收到的句柄,但如果我这样做,应用程序只会退出而不会出现错误。

编辑2

我正在使用 InternetReadFile 从调用 FtpCommand 时返回的数据套接字句柄中读取数据。我使用的 InternetReadFile 方法签名是这样的:

[DllImport("wininet.dll", CharSet = CharSet.Ansi, SetLastError = true)]
extern public static int InternetReadFile(
    [In] IntPtr hConnect,
    [In][Out] StringBuilder buffer,
    [In] int buffCount,
    [In][Out] ref int bytesRead);

最终编辑

我发布了这个code on codeplex :

最佳答案

为什么会出现 500 语法错误

请记住,与 FTP 客户端(如 ftp.exe)不同,实际的 FTP protocol不接受像 ls 这样的命令,它接受像 LISTNLST

这样的命令

参见 Raw FTP Command ListRFC 959 FTP 接受的命令列表

如何获得 LISTNLST 命令的结果?

FtpCommand 的最后一个参数是

A pointer to a handle that is created if a valid data socket is opened. The fExpectResponse parameter must be set to TRUE for phFtpCommand to be filled.

您可以使用此句柄通过 InternetReadFile 读出列表.

注意事项:

  • 您可以考虑使用 WireShark 之类的东西查看 ftp.exe 发送的实际网络流量并将其与您的代码发送的内容进行比较
  • 您也可以考虑调查 FtpWebRequest ;如果它满足您的需求,它可能更易于使用。

关于c# - 如何从通过 FtpCommand 运行的 FTP 命令获取结果?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/677949/

相关文章:

c# - 在 C# 中使用 CsvHelper 写入文件

c# - 如何通过 FastMember 自动将枚举转换为 int?

c# - 使用 pythonnet 从 .Net Core 调用 python 脚本

c# - 安卓/Xamarin : How to get the label value of current activity?

c# - 给定一个窗口,我如何确定它是否是 winforms 应用程序的一部分?

c++ - 如何使用离屏 DC 一次渲染所有位图的汇编?

linux - 设置 Emacs TRAMP 以存储本地备份

C++ Win32 窗口背景不重绘

java - 在 30 秒内处理来自 FTP 服务器的 100 MB 以上文件?

python - 使用 FTP 后端的 Django 文件上传