阿帕奇人htpasswd
文件不支持任何影子功能。因此,您必须阻止用户访问您的网络服务器,以使他们远离密码文件。因此,唯一的解决方案是基于 SSH 的方法或任何其他远程解决方案。下面的描述将解释如何编写 SSH 命令脚本来仅在用户知道其旧密码的情况下更改密码。主要问题是,Apache 不提供命令行工具来验证密码htpasswd
文件。但这可以手工完成。
以下描述假设Web服务器用户是www-data
用户的主目录是/var/www
.
首先,您必须创建一个可由 Web 服务器用户写入的 htpasswd 文件:
# ls -la .htpasswd
-rw-r--r-- 1 www-data root 18 10. Mai 16:30 .htpasswd
然后您必须将所有用户的密钥添加到authorized_keys
Web 服务器用户的文件。您必须在每一行前面加上前缀command
option.
# cat .ssh/authorized_keys
command="/var/www/.htpasswd.sh" ssh-rsa AAAA... user@host
每当用户使用他的密钥连接时,仅.htpasswd.sh
被处决。用户没有任何对 Web 服务器的 shell 访问权限。
这是更改密码的脚本:
#! /bin/bash
HTPASSWD=/var/www/.htpasswd
die () { echo "$*" >&2 ; exit 1 ; }
read -p 'Enter user name: ' USER
read -s -p 'Old password: ' OLDPW ; echo
read -s -p 'New password: ' NEWPW0 ; echo
read -s -p 'Re-type new password: ' NEWPW1 ; echo
if LINE=$(grep ^"$USER": "$HTPASSWD")
then
echo "$LINE" | sed 's/.*:\(..\)\(.\+\)/\1 \2/' | {
read SALT CRYPT
if [[ "$SALT$CRYPT" = $(echo "$OLDPW" | mkpasswd -sS "$SALT") ]] ; then
if [ "$NEWPW0" != "$NEWPW1" ] ; then
die "Password verification error!"
fi
PWS=$(grep -v ^"$USER:" "$HTPASSWD")
{
echo "$PWS"
echo -n "$USER:"
echo "$NEWPW0" | mkpasswd -s
} > "$HTPASSWD"
echo "Updating password for user $USER."
else
die "Password verification error!"
fi
}
else
die "Password verification error!"
fi
棘手的部分是密码验证。它是通过读取旧盐并用旧盐加密旧密码来完成的。将结果与旧的加密密码进行比较htpasswd
file.
现在用户可以连接到 Web 服务器来更改密码:
$ ssh www-data@localhost
Enter user name: szi
Old password:
New password:
Re-type new password:
Updating password for user szi.
Connection to localhost closed.
每个人只能更改自己的密码,没有人可以访问其他用户的加密密码。该解决方案还有一个关于使用原始解决方案的额外好处htpasswd
shell 脚本中的程序,因为密码从不用作命令行参数。这是不可能的htpasswd
,因为它无法从 stdin 读取密码,例如mkpasswd
.