本人出现问题的php报错信息:
GuzzleHttp\Exception\ConnectException: cURL error 6: Could not resolve host: api.mch.weixin.qq.com; Name or service not known

Guzzle是一个PHP的HTTP客户端,用来轻而易举地发送请求

系统版本:阿里云ECS CentOS7
PHP版本:5.5

先放结论:

如果您的Linux机器通过cURL,wget或类似的URL传输工具缓慢解析DNS,可能是因为ipv4 / ipv6解析冲突。

CentOS 6 和 CentOS 7 的 DNS 解析机制中,发送 IPV4 DNS 和 IPV6 DNS 请求使用了相同的网络五元组,此时应开启 single-request-reopen 配置,一旦出现同一 socket 发送的两次请求处理,解析端发送第一次请求后会关闭 socket,并在发送第二次请求前打开新的 socket。配置成功后不需要重启实例即可生效。

ipv4(A)和ipv6(AAAA)使用相同的套接字和端口来解析DNS,然后它只能发回一个响应,然后cURL或wget,或者你使用的任何东西,都会等待(可能)来自DNS解析的第二个答案的第一次查找超时。如果存在冲突,此选项会使您的计算机重新打开新套接字。


本人环境和说明:

  1. 查看本机适用的DNS服务器:
# cat /etc/resolv.conf

options timeout:1 attempts:1 rotate 
nameserver 100.100.2.136
nameserver 100.100.2.138
以上是购买ECS服务器后的默认值,第一行是可选参数
第二行应该是阿里云的内部DNS Server IP
  1. 本机CURL版本(因为PHP代用的是系统的curl库)

    # rpm -qa|grep curl
    curl-7.29.0-51.el7.x86_64
    libcurl-7.29.0-51.el7.x86_64
    libcurl-devel-7.29.0-51.el7.x86_64
  2. 手动用dig和host命令查询解析都是正常的
# dig @100.100.2.136 api.mch.weixin.qq.com

; <<>> DiG 9.9.4-RedHat-9.9.4-50.el7_3.1 <<>> @100.100.2.136 api.mch.weixin.qq.com
; (1 server found)
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 48831
;; flags: qr rd ra; QUERY: 1, ANSWER: 3, AUTHORITY: 0, ADDITIONAL: 0

;; QUESTION SECTION:
;api.mch.weixin.qq.com.     IN  A

;; ANSWER SECTION:
api.mch.weixin.qq.com.  37  IN  CNAME   forward.weixin.qq.com.
forward.weixin.qq.com.  41  IN  A   140.207.69.102
forward.weixin.qq.com.  41  IN  A   140.207.69.101

用100.100.2.138查也是同2个IP
  1. 经过G字母的搜索引擎查找资料,其他人的解决方法:
在/etc/resolv.conf的可选参数后里加上:single-request-reopen
第一行内容既改为:
options timeout:1 attempts:1 rotate single-request-reopen

本文参考出处:

  1. https://aarvik.dk/disable-ipv6/
  2. https://help.aliyun.com/knowledge_detail/59344.html
  3. http://blog.sina.com.cn/s/blog_7f2122c50101uk9m.html