记一次php mysqli出现No such file or directory的错误排查
记一次php mysqli出现No such file or directory的错误排查
呵大官人 发表于7个月前
记一次php mysqli出现No such file or directory的错误排查
  • 发表于 7个月前
  • 阅读 211
  • 收藏 9
  • 点赞 1
  • 评论 4

【腾讯云】新注册用户域名抢购1元起>>>   

最近公司的一台开发机器坏了,需要把部分工作相关的系统转移到另一台机器上,在转移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会出现错误,要找到问题的根源才好对症下药。

 

 

  • 打赏
  • 点赞
  • 收藏
  • 分享
共有 人打赏支持
呵大官人
粉丝 117
博文 17
码字总数 15799
作品 1
评论 (4)
张乐1024
赞一个!
灵魂架构师
不看文章就知道是 默认sock文件没配置
dingdayu
虽然问题已在其他地方看到过,但是楼主解决问题的思路值得学习。
chdy
我也遇到过,但你解题思路清晰
×
呵大官人
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: