使用 systemd 自动挂载 USB 驱动器

Mik*_*ell 38 linux usb-flash-drive systemd

我们正在将服务器从非常过时的发行版更新为基于现代 Debian Jessie 的系统,包括 lightdm / xfce,当然还有 systemd(和 udisks2)。一个症结是自动挂载 USB 驱动器。我们过去常常通过一些 udev 规则来实现这一点。旧规则几乎仍然有效 - 安装点被创建并且驱动器安装正常,但几秒钟后 systemd 正在做一些破坏安装的事情,因此随后的访问尝试导致“传输端点未连接”错误。

通过命令行手动安装驱动器工作正常。让文件管理器(thunar 和 thunar-volman,反过来使用 udisks2)也是如此。但这些都不是可行的选择——这些系统大多是无头运行的,所以 thunar 通常不会运行。我们需要能够插入磁盘驱动器以进行无人值守的基于 cron 的备份。

我认为修改 udev 脚本以生成一个在执行挂载前等待几秒钟的分离作业可能会奏效,但 systemd 似乎竭尽全力防止这种情况发生 - 它仍然以某种方式等待分离作业完成之前继续。

也许让 udev 脚本以某种方式使 udisks2 发痒是正确的方法?我不知所措,所以非常感谢任何建议。

Mik*_*ell 42

在几次错误的开始之后,我发现了这一点。关键是在 udev 和挂载脚本之间添加一个 systemd 单元服务。

(作为记录,我无法使用udisksctl mount -b /dev/sdb1直接从 udev 规则或 systemd 单元文件调用的udisks2(通过类似)使其工作。似乎存在竞争条件,设备节点还没有完全准备好,导致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
Run Code Online (Sandbox Code Playgroud)

该脚本又由 systemd 单元文件调用。我们使用“@”文件名语法,因此我们可以将设备名称作为参数传递。

/etc/systemd/system/usb-mount@.service

[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
Run Code Online (Sandbox Code Playgroud)

最后,一些 udev 规则在热插拔/拔出时启动和停止 systemd 单元服务:

/etc/udev/rules.d/99-local.rules

[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
Run Code Online (Sandbox Code Playgroud)

这似乎可以解决问题!几个有用的命令来调试这样的东西:

  • udevadm control -l debug打开详细日志记录, /var/log/syslog以便您可以查看正在发生的事情。
  • udevadm control --reload-rules 在修改 rules.d 目录中的文件后(可能没有必要,但不会伤害......)。
  • systemctl daemon-reload 修改 systemd 单元文件后。

  • 哇。这太棒了。希望我能多投几票!我唯一需要修改的是在我的系统上,`blkid` 似乎没有提取 `ID_FS_LABEL`,所以我只是使用了 `DEVBASE` 而不是 `LABEL` 来构建 `MOUNT_POINT`。 (5认同)
  • 如果已经连接了 USB 设备,这在启动时效果不佳。有任何想法吗? (2认同)

inf*_*era 15

有一个新的、简洁的systemd自动挂载选项,可以使用fstab它允许您使用所有标准化的挂载权限选项,它看起来像这样:

  x-systemd.automount
Run Code Online (Sandbox Code Playgroud)

fstab一行中的一个例子:

  /dev/sdd1   /mnt/hitachi-one     auto     noauto,x-systemd.automount     0 2
Run Code Online (Sandbox Code Playgroud)

noauto选项意味着它不会像旧软件一样尝试在启动时安装autofs

x-systemd.automountfstab您添加新行后,需要运行:

  sudo systemctl daemon-reload
Run Code Online (Sandbox Code Playgroud)

然后是以下两项或一项:

  sudo systemctl restart remote-fs.target
  sudo systemctl restart local-fs.target
Run Code Online (Sandbox Code Playgroud)

有关它的更多信息:

https://wiki.archlinux.org/index.php/Fstab#Automount_with_systemd


小智 6

使用pmount、 systemd 和 Mike Blackwell 的方法,您可以简化整个事情:

/etc/systemd/system/usb-mount@.service

[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
Run Code Online (Sandbox Code Playgroud)

/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"
Run Code Online (Sandbox Code Playgroud)

谢谢,谢谢迈克。