【问题标题】:Indy error 10038 "Socket operation on non-socket" after 61 seconds of inactivity61 秒不活动后 Indy 错误 10038“非套接字上的套接字操作”
【发布时间】:2016-12-17 20:51:51
【问题描述】:

我想从 FTP 服务器下载一些大文件 (GB)。 第一个文件的下载始终有效。然后在尝试获取第二个文件时,我得到:

“套接字错误#10038。非套接字上的套接字操作。”

错误出现在“获取”上。在“获取”之后,我看到了这些消息(通过 FTP 状态事件):

   Starting FTP transfer
   Disconnecting.
   Disconnected.

代码是这样的:

{pseudo-code}
for 1 to AllFiles do 
 begin 
   if Connect2FTP then 
    begin
     FTP.Get(Name, GzFile, TRUE, FALSE);  <--- Socket operation on non-socket" error (I also get EIdConnClosedGracefully 'Connection Closed Gracefully' in IDE, F9 will resume execution without problems, but this is OK)
     Unpack(GzFile); <--- this takes more than 60 seconds
    end;
 end;
if FTP.Connected
then FTP.Disconnect;

--

function Connect2FTP(FTP: TIdFTP; RemoteFolder: string; Log: TRichLog): Boolean;      
begin
 Result:= FTP.Connected;
 if NOT Result then
  begin         { We are already connected }
   FTP.Host    := MyFTP;
   FTP.Username:= usr;
   FTP.Password:= psw;

   TRY
    FTP.Connect;
   EXCEPT
    on E: Exception DO

   Result:= FTP.Connected;
   if Result then FTP.ChangeDir(RemoteFolder); 
  end;
end;

完整代码在这里:http://pastebin.com/RScj86R8 (PAS) 或这里 https://ufile.io/26b54 (ZIP)

我认为在调用“Unpack”后会出现问题,这需要几分钟时间。

更新:确认:调用“解包”后出现问题。我删除了电话,一切都很好。在下载之间暂停(睡眠或断点)程序一段时间(我认为超过 60 秒)会产生同样的问题。

【问题讨论】:

  • 这值得商榷。该代码无法编译,您已经完成了其中的重要部分。我建议你花更多的时间在这个问题上。显然你有一个问题,但如果你在你的问题上投入更多的精力,你会得到更好的帮助。在这里问了 290 个问题后,我很难相信您不知道该怎么做,为什么需要提供minimal reproducible example。啊,现在我看到你还在添加更多,即使在上面的评论之后。无论如何,10038 表示你关闭了套接字,然后再次尝试使用它。
  • 现在你知道去哪里看
  • 既然你是唯一一个有repro和实际代码的人,我想你需要调试它。
  • 好吧,我猜那一切都很好,正在使用的套接字也很好,抛出10038时系统错误。

标签: delphi ftp indy delphi-xe7 indy10


【解决方案1】:

FTP 使用多个套接字连接,一个用于命令/响应,每个传输使用单独的连接。

套接字错误的最可能原因是位于TIdFTP 之间的 FTP 不感知代理/路由器/防火墙,并且 FTP 服务器在短时间不活动后关闭命令连接。在Unpack()(或手动暂停)期间,命令连接上没有传输任何命令/响应,它处于空闲状态,因此会因此类代理/路由器/防火墙上的超时而关闭。

在传输过程中,命令连接处于空闲状态,没有 FTP 命令/响应在其上传输(除非您中止传输),直到传输完成。在此期间,不支持 FTP 的代理/路由器/防火墙可能会过早关闭命令连接。

为避免这种情况,TIdFTP 有一个NATKeepAlive 属性,可以在命令连接处于空闲状态时启用 TCP keep-alives。这通常可以防止过早关闭。

但是,当 prgress 中没有传输时,如果 NATKeepAlive.UseKeepAlive 为 True,TIdFTP 会在命令连接上禁用 TCP keep-alives。 TIdFTP 仅在传输期间使用 TCP 保持活动,假设您不会在 FTP 命令之间执行长时间延迟。如果需要延迟一段时间,要么关闭 FTP 连接,要么定期发送 FTP 命令(例如在定时器/线程中调用TIdFTP.Noop())。

或者,您可以尝试在连接到服务器后手动启用 TCP keep-alives,并将 NATKeepAlive.UseKeepAlive 设置为 False,以便 TIdFTP 在每次传输后不会自动禁用 keep-alives,例如:

function Connect2FTP(FTP: TIdFTP; RemoteFolder: string; Log: TRichLog): Boolean;      
begin
  Result := FTP.Connected;
  if not Result then
  begin         { We are not already connected }
    FTP.Host    := MyFTP;
    FTP.Username:= usr;
    FTP.Password:= psw;

    try
      FTP.Connect;
      try
        FTP.ChangeDir(RemoteFolder); 

        // send a TCP keep-alive every 5 seconds after being idle for 10 seconds
        FTP.NATKeepAlive.UseKeepAlive := False; // False by default, but just in case...
        FTP.Socket.Binding.SetKeepAliveValues(True, 10000, 5000);
      except
        FTP.Disconnect(False);
        raise;
      end;
    except
      Exit;
    end;

    Result := True;
  end;
end;

请注意,虽然大多数平台都支持在每个连接的基础上启用 TCP 保持连接(保持连接是 TCP 规范的一部分),但设置保持连接间隔是特定于平台的。目前,Indy 支持将间隔设置为:

  • Windows 2000+ 和 WinCE 4.x+,用于 Win32 和 .NET
  • Linux,当 Indy 在 Kylix 中使用时
  • Unix/Linux 和 NetBSD,在 FreePascal 中使用 Indy 时。

否则,将使用操作系统默认间隔,对于这种情况,它的值可能太大也可能不会太大,具体取决于操作系统配置。

目前,Delphi FireMonkey 中无法自定义间隔,Windows 除外。

如果特定平台支持设置自定义 TCP keep-alive 间隔,但 Indy 未在 SetKeepAliveValues() 中实现它们,您可以使用 TIdFTP.Socket.Binding.SetSockOpt() 来根据需要手动设置值。许多平台确实支持TCP_KEEPIDLE/TCP_KEEPINTVL 或等效的套接字选项。

【讨论】:

  • 哇。这是一个很好且有用的答案!非常感谢。目前我通过不解压下载的文件直到我拥有它们来“解决”这个问题。但很快我将实施您的解决方案之一。下载后断开连接,解压前(仅当解压文件很大时)显然是最简单的,我认为也是最可靠的。与下载 1GB 数据所需的时间相比,断开/重新连接所增加的时间微不足道。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2017-06-10
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多