web-dev-qa-db-ja.com

systemdでUSBドライブを自動マウント

サーバーを非常に古くなったディストリビューションからlightdm/xfce、もちろんsystemd(およびudisks2)を含む最新のDebian Jessieベースのシステムに更新しています。問題の1つは、USBドライブの自動マウントです。以前は、いくつかのudevルールでこれを実現していました。古いルールはほぼそのまま機能します-マウントポイントが作成され、ドライブが正常にマウントされますが、数秒後にsystemdがマウントを破壊する何かを実行しているため、その後のアクセス試行で「トランスポートエンドポイントが接続されていません」というエラーが発生します。

コマンドラインを介して手動でドライブをマウントすると問題なく動作します。ファイルマネージャー(thunarとthunar-volman、次にudisks2を使用)も同様です。しかし、それらは実行可能なオプションではありません-これらのシステムはほとんどヘッドレスで実行されるため、通常はthunarは実行されていません。 cronベースの無人バックアップのために、ディスクドライブをプラグインできる必要があります。

Udevスクリプトを変更して、マウントを実行する前に数秒待機するデタッチされたジョブを生成するのはうまくいくかもしれないと思いましたが、systemdはこれを防ぐために邪魔になりそうです-何らかの方法でデタッチされたジョブが完了する前に待機します続けます。

おそらく、udevスクリプトがudisks2をくすぐるのは、どういうわけか正しいアプローチでしょうか。私は途方に暮れているので、アドバイスは大歓迎です。

28
Mike Blackwell

いくつかの誤ったスタートの後、私はこれを理解しました。重要なのは、udevとマウントスクリプトの間にsystemdユニットサービスを追加することです。

(記録のために、udisks2を使用して(udisksctl mount -b /dev/sdb1などを介して)udevルールから直接またはsystemdユニットファイルから直接呼び出すことはできませんでした。競合状態とデバイスノードがあるようです準備が整っていないため、Error looking up object for device /dev/sdb1が発生します。残念ながら、udisks2はすべてのマウントポイントの煩雑さを処理できます...)

重い作業は、マウントポイントの作成と削除、ドライブのマウントとアンマウントを処理するシェルスクリプトによって行われます。

/ 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/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

最後に、いくつかのudevルールがhotplug/unplugで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ディレクトリのファイルを変更した後(必要ないかもしれませんが、害はありません...)。
  • systemdユニットファイルを変更した後のsystemctl daemon-reload
30
Mike Blackwell

新しい、簡潔な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

次に、以下の両方、または1つ:

  Sudo systemctl restart remote-fs.target
  Sudo systemctl restart local-fs.target

詳細については:

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

13

私はウォーレン・ヤングの答えに行きます私はいくつかの変更を加えました

ドライブの環境の評価からエラーが出るので、スペース保護を追加しました。

USBディスクをchmodするセクションを追加して、すべてのユーザーがntfsまたはvfat以外のディスクに完全にアクセスできるようにしました。

/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
    # added some sed's to avoid space issues
    eval $(/sbin/blkid -o udev ${DEVICE}|sed 's/=/="/'|sed 's/$/"/')

    # 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"
    #added a chmod checker for file systems that don't 
    #understand allow all to read write
    CHMOD=no
    # File system type specific mount options
    if [[ ${ID_FS_TYPE} == "vfat" ]]; then
        OPTS+=",users,gid=100,umask=000,shortname=mixed,utf8=1,flush"
    #added options I wanted on ntfs
    Elif [[ ${ID_FS_TYPE} == "ntfs" ]]; then
        OPTS+=",user,users,umask=000,allow_other"
    else
       CHMOD=yes
    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} ****"
    if [ "${CHMOD}" = "yes" ];then
        /usr/bin/find "${MOUNT_POINT}" -type f -exec chmod 0666 {} \;
        /usr/bin/find "${MOUNT_POINT}" -type d -exec chmod 0777 {} \;
    fi
}

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
2
penguinjeff

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

/ 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、マイクありがとうございます。

2
Eric V.

スクリプトを@MikeBlackwellから次のように変更しました。

  • /dev/sd[a-z]だけでなく/dev/sd[a-z]*も含む複数の文字にまたがるデバイス名を認識します。スピンドルの数が多いサーバーの場合がよくあります。
  • 自動マウントされたドライブのリストを/var/log/usb-mount.trackで追跡する
  • タグusb-mount.shを使用してアクションを/var/log/messagesに記録します
  • ラベルが割り当てられていないドライブの問題が発生しないようにマウントポイントが実行されないように、デバイス名の前にデバイスラベルを付けます(空?):/media/sdd2_usbtest/media/sdd2_
  • ファイルを適切に配置し、必要に応じて元に戻すラッパースクリプトが含まれています

@MikeBlackwellは大部分の面倒な作業をすでに行っているので、書き直さないことにしました。必要な変更を加えただけです。元の回答の名前とURIを確認した彼の作品を認めた。

https://github.com/raamsri/automount-usb で見つけてください

2
six-k