如何模拟linux中socket/tcp编程的异常情况,比如终止连接的一端?

xgw*_*ang 6 c sockets linux tcp

我正在学习使用SO_SNDTIMEOSO_RCVTIMEO来检查超时.它易于使用读取插座.但是当我想检查写入超时时,它总是返回成功.这就是我所做的:(全部处于阻止模式)

  1. 关闭客户端读取套接字并在服务器开始写入之前退出
  2. 在服务器开始写入之前终止客户端
  3. 接受之后但在写入之前拔掉服务器的电缆

好吧,似乎所有这些案例写成只是成功回归.我认为原因应该是端口是由os管理的资源,并且在客户端,程序消失后,tcp连接仍然显示FIN_WAIT2状态.

那么,有没有方便的方法来模拟写入可以接收错误的一些情况,例如EPIPE,EAGAIN

Jay*_*Jay 5

如何获得错误EAGAIN?
要获得错误EAGAIN,您需要使用非阻塞套接字.与非阻塞套接字,你需要写大量的数据(并停止在对端接收数据),让你的内部TCP缓冲区得到填补,并返回该错误.

如何获得错误EPIPE?
要获得错误EPIPE,您需要在关闭对等端的套接字后发送大量数据.您可以从此SO 链接获得有关EPIPE错误的更多信息.我在提供的链接中询问了有关"断管错误"的问题,并且接受的答案给出了详细说明.重要的是要注意,要获得EPIPE错误,您应该将发送的flags参数设置为MSG_NOSIGNAL.没有它,异常发送可以生成SIGPIPE信号.

附加说明
请注意,很难模拟写入失败,因为TCP通常会将您尝试写入的数据存储到其内部缓冲区中.因此,如果内部缓冲区有足够的空间,那么您不会立即收到错误.最好的方法是尝试编写大量数据.您还可以尝试使用带有SO_SNDBUF选项的setsockopt函数为发送设置较小的缓冲区大小


Eri*_*ski 5

您可以使用故障注入来模拟错误.例如,libfiu是一个故障注入库,它带有一个示例项目,允许您模拟POSIX函数的错误.基本上它用于LD_PRELOAD在常规系统调用(包括write)周围注入一个包装器,然后包装器可以配置为传递给真正的系统调用,或者返回你喜欢的任何错误.