Node.js 常见网络错误信息

jopen 8年前
 

Node.js 常见网络错误信息

1.介绍

熟悉 Node.js 的同学一定见过ETIMEDOUT、EADDRINUSE等错误提示,那么这些错误信息到底是什么呢?答案其实很简单,因为 Node.js 底层使用的是 glibc 库 ,这些错误信息都是 glibc 库在 socket 连接时使用的 connect 函数中定义的错误类型,当然,v8在使用glibc库时也会加入一些自定义的错误类型,但许多错误情况还是和glibc中的定义一致的。

connect函数的定义为:int connect (int socket, struct sockaddr *addr, socklen_t length)

connect函数会使用文件描述符(file descriptor)socket表示的 socket 发起连接,socket 地址通过addr和length这两个参数来指定。(这个 socket 一般是其他机器的 socket,而且必须已经配置成了服务器)。

一般情况下,connect函数会等待服务器响应请求才返回。当然也可以将socket设置为非阻塞模式来不等待响应就快速返回(可以参考下 nginx 是怎么使用 socket 的)。

2.错误类型

connect函数正常的返回值为 0,在有错误时会返回 -1。函数中定义了如下错误条件:

EBADF:

socket 不是有效的文件描述符(file descriptor)。

ENOTSOCK

文件描述符socket不是 socket 。

EADDRNOTAVAIL

指定的地址在远程机器上不可用。

EAFNOSUPPORT

socket 不支持addr的命名空间。

EISCONN

socket 已经连接。

ETIMEDOUT

连接尝试超时。

ECONNREFUSED

服务器主动拒绝建立连接。

ENETUNREACH

从本机到给定addr的网络不通。

EADDRINUSE

给定addr的 socket 地址已经在使用。(这种错误最常见,我们有时候本地已经启动了一个 Node.js 程序,再次启动会看到这个错误,也就是端口号已经被占用了)

EINPROGRESS

socket 是非阻塞的,连接不能立即建立。可以使用select来确定连接完全建立的时间。参考 等待I/O 。在连接完全建立前如果在相同的 socket 上调用 connect,会以EALREADY失败。

EALREADY

socket 是非阻塞的而且有一个挂起的连接(参考上面的EINPROGRESS)。

3.注意

connect 是作为多线程程序的取消点定义的,开发者需要确保线程取消后释放了占用的资源(例如内存、文件描述符、 semaphore )等)。

参考资料

libc: Making a Connection