记一次php mysqli出现No such file or directory的错误排查

原创
2017/09/29 10:56
阅读数 2.9W
AI总结

最近公司的一台开发机器坏了,需要把部分工作相关的系统转移到另一台机器上,在转移mantis的过程中发现mysql连不上了,而且错误居然是No such file or directory,这个错误信息很明显告诉我文件不存在,但是我是通过网络连mysql,何故会提示文件不存在?于是写了一段测试代码。

$mysqli = new mysqli('localhost', 'root', '', 'bugtracker');
if ($mysqli->connect_error) {
    die('Connect Error (' . $mysqli->connect_errno . '): '. $mysqli->connect_error);
}

果然,还是报错!!!!

PHP Warning:  mysqli::mysqli(): (HY000/2002): No such file or directory in /root/test.php on line 2

为了确保参数没错,使用mysql客户端进行测试

mysql -uroot -hlocalhost

嗯,连上了,完全没有任何错误,那说明参数是没有问题的,可为什么相同的参数,mysql客户端连接正常,php连接不上?为了一探究竟,于是使用strace再次跑了一下这段代码,发现系统调用的connect居然不是使用tcp,而是unixdomain,文件路径是/tmp/mysql.sock,而这个文件确实不存在

connect(3, {sa_family=AF_FILE, path="/tmp/mysql.sock"}, 17) = -1 ENOENT (No such file or directory)

到这里,我已经知道错误的来源,是使用unixdomain去连接了一个不存的地址,导致报错,可为什么会使用unixdomain连接mysql呢?而这个地址又从哪里来的呢?于是又查阅了相关资(Bai)料(Du),根据史书记载,当使用localhost进行连接的时候,会使用unixdomain的方式,那这么说来mysql客户端也是使用unixdomain了?于是又对mysql使用了一次strace

connect(3, {sa_family=AF_FILE, path="/var/lib/mysql/mysql.sock"}, 110) = 0

发现connect确实是使用unixdomain,但是.......这个文件路径跟上面的完全不一样,而这个路径是存在的。。可上面/tmp/mysql.sock到底哪里来的?于是又马不停蹄的去翻阅php文档,文档明确指出,mysqli的构造函数socket参数(也就是上面提到的unixdomain的文件路径)默认从php.ini的mysqli.default_socket中取,那么再次检查参数

php -i|grep 'mysqli.default_socket'
mysqli.default_socket => no value => no value

嗯哼?没有值,闹呢这是。那这个货到底从哪里来的,为了探明事情真相,开始翻阅mysqli的源代码,果然,从源代码中找出了这么一段

if (host_len == sizeof("localhost") - 1 && !strncasecmp(host, "localhost", host_len)) {
    DBG_INF_FMT("socket=%s", socket_or_pipe? socket_or_pipe:"n/a");
    if (!socket_or_pipe) {
        socket_or_pipe = "/tmp/mysql.sock";
    }
    transport_len = mnd_sprintf(&transport, 0, "unix://%s", socket_or_pipe);
    unix_socket = TRUE;
}

如果使用localhost,就使用unixdomain,文件路径就是参数中的socket,但如果这个参数是空,那么奏写死/tmp/mysql.sock。这么说来上面报错不存在的地址就是在源代码中给写死了,所以需要自己手动设置为正确的值。于是修改mysqli的参数再次测试。

$mysqli = new mysqli('localhost', 'root', '', 'bugtracker', 3306, "/var/lib/mysql/mysql.sock");

果然好了,到这里,已经查明事情真相,连接不上就是因为使用了localhost并且mysqli.default_socket的值为空,而源码中提供的默认路径又不存在。到这里我不经要问,可否加两个判断呢?使用localhost并且socket不为空才使用unixdomain,而不提供默认路径,因为如果先装php后装mysql,那么这个default_socket就会是个空值。

当然解决这个问题其实很简单,网上有非常多的文章都有写如何解决,比如将mysqli.default_socket的值改为正确的路径,或者将localhost改成127.0.0.1等。主要还是想知道为什么不配置socket会出现错误,要找到问题的根源才好对症下药。

 

 

展开阅读全文
加载中
点击加入讨论🔥(5) 发布并加入讨论🔥
5 评论
9 收藏
1
分享
AI总结
返回顶部
顶部