我们正在将服务器从非常过时的发行版更新为基于现代 Debian Jessie 的系统,包括 lightdm / xfce,当然还有 systemd(和 udisks2)。一个症结是自动挂载 USB 驱动器。我们过去使用一些 udev 规则来实现这一点。旧规则几乎仍然有效 - 挂载点已创建并且驱动器已正常挂载,但几秒钟后 systemd 正在执行一些破坏挂载的操作,因此后续访问尝试会导致“传输端点未连接”错误。
通过命令行手动安装驱动器效果很好。使用文件管理器(thunar 和 thunar-volman,后者又使用 udisks2)也可以。但这些都不是可行的选择 - 这些系统大多以无头方式运行,因此 thunar 通常不会运行。我们需要能够插入磁盘驱动器以进行无人值守的基于 cron 的备份。
我认为修改 udev 脚本以生成一个分离作业,该作业在执行挂载之前等待几秒钟可能会解决问题,但 systemd 似乎不遗余力地阻止这种情况 - 它仍然以某种方式等待分离作业完成后再继续。
也许让 udev 脚本以某种方式触发 udisks2 是正确的方法?我不知所措,所以非常感谢任何建议。
答案1
经过几次失败的尝试,我终于找到了答案。关键是在 udev 和挂载脚本之间添加一个 systemd 单元服务。
(据记录,我无法使用 udisks2(通过类似的方法udisksctl mount -b /dev/sdb1
)直接从 udev 规则或 systemd 单元文件调用来使其工作。似乎存在竞争条件并且设备节点尚未准备好,导致Error looking up object for device /dev/sdb1
。不幸的是,因为 udisks2 可以处理所有挂载点混乱......)
繁重的工作由 shell 脚本完成,该脚本负责创建和删除挂载点以及挂载和卸载驱动器。
/usr/local/bin/usb-mount.sh
#!/bin/bash
# This script is called from our systemd unit file to mount or unmount
# a USB drive.
usage()
{
echo "Usage: $0 {add|remove} device_name (e.g. sdb1)"
exit 1
}
if [[ $# -ne 2 ]]; then
usage
fi
ACTION=$1
DEVBASE=$2
DEVICE="/dev/${DEVBASE}"
# See if this drive is already mounted, and if so where
MOUNT_POINT=$(/bin/mount | /bin/grep ${DEVICE} | /usr/bin/awk '{ print $3 }')
do_mount()
{
if [[ -n ${MOUNT_POINT} ]]; then
echo "Warning: ${DEVICE} is already mounted at ${MOUNT_POINT}"
exit 1
fi
# Get info for this drive: $ID_FS_LABEL, $ID_FS_UUID, and $ID_FS_TYPE
eval $(/sbin/blkid -o udev ${DEVICE})
# Figure out a mount point to use
LABEL=${ID_FS_LABEL}
if [[ -z "${LABEL}" ]]; then
LABEL=${DEVBASE}
elif /bin/grep -q " /media/${LABEL} " /etc/mtab; then
# Already in use, make a unique one
LABEL+="-${DEVBASE}"
fi
MOUNT_POINT="/media/${LABEL}"
echo "Mount point: ${MOUNT_POINT}"
/bin/mkdir -p ${MOUNT_POINT}
# Global mount options
OPTS="rw,relatime"
# File system type specific mount options
if [[ ${ID_FS_TYPE} == "vfat" ]]; then
OPTS+=",users,gid=100,umask=000,shortname=mixed,utf8=1,flush"
fi
if ! /bin/mount -o ${OPTS} ${DEVICE} ${MOUNT_POINT}; then
echo "Error mounting ${DEVICE} (status = $?)"
/bin/rmdir ${MOUNT_POINT}
exit 1
fi
echo "**** Mounted ${DEVICE} at ${MOUNT_POINT} ****"
}
do_unmount()
{
if [[ -z ${MOUNT_POINT} ]]; then
echo "Warning: ${DEVICE} is not mounted"
else
/bin/umount -l ${DEVICE}
echo "**** Unmounted ${DEVICE}"
fi
# Delete all empty dirs in /media that aren't being used as mount
# points. This is kind of overkill, but if the drive was unmounted
# prior to removal we no longer know its mount point, and we don't
# want to leave it orphaned...
for f in /media/* ; do
if [[ -n $(/usr/bin/find "$f" -maxdepth 0 -type d -empty) ]]; then
if ! /bin/grep -q " $f " /etc/mtab; then
echo "**** Removing mount point $f"
/bin/rmdir "$f"
fi
fi
done
}
case "${ACTION}" in
add)
do_mount
;;
remove)
do_unmount
;;
*)
usage
;;
esac
反过来,该脚本由 systemd 单元文件调用。我们使用“@”文件名语法,以便我们可以将设备名称作为参数传递。
/etc/systemd/系统/[电子邮件保护]
[Unit]
Description=Mount USB Drive on %i
[Service]
Type=oneshot
RemainAfterExit=true
ExecStart=/usr/local/bin/usb-mount.sh add %i
ExecStop=/usr/local/bin/usb-mount.sh remove %i
最后,一些 udev 规则在热插拔时启动和停止 systemd 单元服务:
/etc/udev/rules.d/99-local.rules
KERNEL=="sd[a-z][0-9]", SUBSYSTEMS=="usb", ACTION=="add", RUN+="/bin/systemctl start usb-mount@%k.service"
KERNEL=="sd[a-z][0-9]", SUBSYSTEMS=="usb", ACTION=="remove", RUN+="/bin/systemctl stop usb-mount@%k.service"
这似乎很管用!下面是一些有用的命令,可用于调试以下内容:
udevadm control -l debug
打开详细日志记录/var/log/syslog
以便您可以看到正在发生的事情。udevadm control --reload-rules
在您修改 rules.d 目录中的文件之后(可能没有必要,但不会造成损害......)。systemctl daemon-reload
修改 systemd 单元文件后。
答案2
有一个新的、简洁的systemd
自动挂载选项可以与它一起使用,fstab
它允许您使用所有标准化的挂载权限选项,它看起来像这样:
x-systemd.automount
一行中的示例fstab
:
/dev/sdd1 /mnt/hitachi-one auto noauto,x-systemd.automount 0 2
该noauto
选项意味着它不会像旧软件一样在启动时尝试安装autofs
。
添加新x-systemd.automount
行后fstab
您需要运行:
sudo systemctl daemon-reload
然后执行以下两项或其中一项:
sudo systemctl restart remote-fs.target
sudo systemctl restart local-fs.target
了解更多信息:
https://wiki.archlinux.org/index.php/Fstab#Automount_with_systemd
答案3
使用挂载、systemd 和 Mike Blackwell 的方法,你可以简化整个事情:
/etc/systemd/系统/[电子邮件保护]
[Unit]
Description=Mount USB Drive on %i
[Service]
Type=oneshot
RemainAfterExit=true
ExecStart=/usr/bin/pmount --umask 000 /dev/%i /media/%i
ExecStop=/usr/bin/pumount /dev/%i
/etc/udev/rules.d/99-usb-mount.rules
ACTION=="add",KERNEL=="sd[a-z][0-9]*",SUBSYSTEMS=="usb",RUN+="/bin/systemctl start usb-mount@%k.service"
ACTION=="remove",KERNEL=="sd[a-z][0-9]*",SUBSYSTEMS=="usb",RUN+="/bin/systemctl stop usb-mount@%k.service"
HTH,谢谢你,Mike。
答案4
我已将@MikeBlackwell 的脚本修改为:
- 识别跨越多个字符的设备名称,不仅仅是
/dev/sd[a-z]
;/dev/sd[a-z]*
通常情况下,服务器具有大量的主轴。 - 跟踪自动安装驱动器列表
/var/log/usb-mount.track
/var/log/messages
使用标签记录操作usb 挂载命令- 在设备名称前加上设备标签,这样挂载点就不会遇到没有分配标签(空?)的驱动器的问题:
/media/sdd2_usbtest
,/media/sdd2_
- 包含包装脚本以适当地放置文件并在需要时撤消
由于@MikeBlackwell 已经完成了大部分繁重的工作,我选择不重写它;只是做了必要的更改。我已经确认了他的工作,并提供了他的名字和原始答案的 URI。