我在 Debian 9.9 上安装了 Apache 2.4,除了提供一个 Roundcube 实例(目前是 1.4 RC-1)之外,基本上什么都不做。
Apache的文档根目录是/home/web
(因为原因),Roundcube实例安装在/home/web/mail
.
Apache 用户 ( www-data
) 具有对整个/home/web
目录树的完全读/写访问权限。
文档根目录有一个重定向到的索引文件,www.mydomain.co.uk/mail
我用mod_rewrite
它来将任何 HTTP 请求更改为 HTTPS。
今天我注意到(并且不知道它发生了多久)当我点击mydomain.co.uk
它时在 Apache 中记录了一个 401 错误:
192.168.10.79 - - [09/Jun/2019:18:49:59 +0100] "GET /mail/ HTTP/1.1" 401 2955 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:67.0)
Gecko/20100101 Firefox/67.0"
当从外部访问它时,这尤其是一个问题,因为它会触发 fail2ban 规则并禁止我的客户。
401 错误表示“身份验证失败”,但我尚未使用 Apache 在该目录上设置任何身份验证,AIUI 需要存在一个.htpasswd
文件 - 我已经检查了我的 Apache 文档根目录中是否存在该文件,但没有- 或者在虚拟主机文件中使用 auth 子句 - 其中也没有。
如何找到原因并防止出现此 401 错误?
来自 Fiddler 的一些调试信息:
GET https://www.mydomain.co.uk/mail/ HTTP/1.1
Accept: text/html, application/xhtml+xml, image/jxr, */*
Accept-Language: en-GB
User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64; Trident/7.0; rv:11.0) like Gecko
Accept-Encoding: gzip, deflate
Host: www.mydomain.co.uk
Connection: Keep-Alive
Cookie: roundcube_sessid=sadh0s1j2rn769rvebhpsv6m81
HTTP/1.0 401 Unauthorized
Date: Mon, 10 Jun 2019 08:27:28 GMT
Server: Apache/2.4.25 (Debian)
Strict-Transport-Security: max-age=63072000; includeSubdomains;
Expires: Mon, 10 Jun 2019 08:27:28 GMT
Cache-Control: private, must-revalidate
Pragma: private
Last-Modified: Mon, 10 Jun 2019 08:27:28 GMT
X-UA-Compatible: IE=edge
X-Frame-Options: sameorigin
Content-Language: en
Vary: Accept-Encoding
X-Robots-Tag: noindex, nofollow
Connection: close
Content-Type: text/html; charset=UTF-8
Content-Length: 5435
如果我在“Auth”选项卡下查看 Fiddler:
No Proxy-Authenticate Header is present.
No WWW-Authenticate Header is present.
来自 wget 的一些进一步的 degug 信息:
DEBUG output created by Wget 1.18 on linux-gnueabi.
Reading HSTS entries from /home/darren/.wget-hsts
URI encoding = ‘UTF-8’
Converted file name 'mail' (UTF-8) -> 'mail' (UTF-8)
--2019-06-10 13:34:43-- https://www.mydomain.co.uk/mail
Certificates loaded: 151
Resolving www.mydomain.co.uk (www.mydomain.co.uk)... XX.XX.XX.XX
Caching www.mydomain.co.uk => XX.XX.XX.XX
Connecting to www.mydomain.co.uk (www.mydomain.co.uk)|XX.XX.XX.XX|:443... connected.
Created socket 3.
Releasing 0x004bf2e0 (new refcount 1).
---request begin---
GET /mail HTTP/1.1
User-Agent: Wget/1.18 (linux-gnueabi)
Accept: */*
Accept-Encoding: identity
Host: www.mydomain.co.uk
Connection: Keep-Alive
---request end---
HTTP request sent, awaiting response...
---response begin---
HTTP/1.1 301 Moved Permanently
Date: Mon, 10 Jun 2019 12:34:43 GMT
Server: Apache/2.4.25 (Debian)
Strict-Transport-Security: max-age=63072000; includeSubdomains;
Location: https://www.mydomain.co.uk/mail/
Content-Length: 331
Keep-Alive: timeout=5, max=100
Connection: Keep-Alive
Content-Type: text/html; charset=iso-8859-1
---response end---
301 Moved Permanently
Registered socket 3 for persistent reuse.
Parsed Strict-Transport-Security max-age = 63072000, includeSubDomains = true
Added new HSTS host: www.mydomain.co.uk:443 (max-age: 63072000, includeSubdomains: true)
URI content encoding = ‘iso-8859-1’
Location: https://www.mydomain.co.uk/mail/ [following]
Skipping 331 bytes of body: [<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN">
<html><head>
<title>301 Moved Permanently</title>
</head><body>
<h1>Moved Permanently</h1>
<p>The document has moved <a hreSkipping 157 bytes of body: [f="https://www.mydomain.co.uk/mail/">here</a>.</p>
<hr>
<address>Apache/2.4.25 (Debian) Server at www.mydomain.co.uk Port 443</address>
</body></html>
] done.
URI content encoding = None
Converted file name 'mail' (UTF-8) -> 'mail' (UTF-8)
--2019-06-10 13:34:43-- https://www.mydomain.co.uk/mail/
Reusing existing connection to www.mydomain.co.uk:443.
Reusing fd 3.
---request begin---
GET /mail/ HTTP/1.1
User-Agent: Wget/1.18 (linux-gnueabi)
Accept: */*
Accept-Encoding: identity
Host: www.mydomain.co.uk
Connection: Keep-Alive
---request end---
HTTP request sent, awaiting response...
---response begin---
HTTP/1.0 401 Unauthorized
Date: Mon, 10 Jun 2019 12:34:43 GMT
Server: Apache/2.4.25 (Debian)
Strict-Transport-Security: max-age=63072000; includeSubdomains;
Set-Cookie: roundcube_sessid=9dd09n33d6k3f1kil69f90bkk6; path=/; secure; HttpOnly
Expires: Mon, 10 Jun 2019 12:34:43 GMT
Cache-Control: private, no-cache, no-store, must-revalidate, post-check=0, pre-check=0
Pragma: no-cache
Last-Modified: Mon, 10 Jun 2019 12:34:43 GMT
X-Frame-Options: sameorigin
Content-Language: en
Vary: Accept-Encoding
X-Robots-Tag: noindex, nofollow
Connection: close
Content-Type: text/html; charset=UTF-8
---response end---
401 Unauthorized
Stored cookie www.mydomain.co.uk -1 (ANY) / <session> <secure> [expiry none] roundcube_sessid 9dd09n33d6k3f1kil69f90bkk6
Disabling further reuse of socket 3.
Username/Password Authentication Failed.
Saving HSTS entries to /home/darren/.wget-hsts
401 Unauthorized Error 是一个 HTTP 响应状态代码,表示客户端发送的请求无法通过身份验证,但可能有多种原因。
我会建议:
发布完整的请求和响应文本供我们检查。特别感兴趣的是
WWW-Authenticate
响应标头,它包含一个或多个挑战字符串,每个字符串指示如何获得正确的身份验证以访问所请求的资源。检查 Apache 服务器日志(可能需要启用详细日志记录)。
如果客户端被禁止,错误代码通常是 401。
清除客户端浏览器上的所有 cookie 和浏览器缓存。
使用禁用所有扩展程序的浏览器。在 Firefox 中这是安全模式,在 Chrome 中是隐身模式。
如果您的应用程序设置为返回此错误代码,请在服务器上搜索。
这是 Roundcube 中的一个新“特性”。
资源
这些是 Roundcube index.php 中有问题的行:
我已经注释掉最后三行以阻止产生 401 错误。