解決策 1:
いくつかの誤ったスタートの後、私はこれを理解しました。重要なのは、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/[email protected]
[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 [email protected]%k.service"
KERNEL=="sd[a-z][0-9]", SUBSYSTEMS=="usb", ACTION=="remove", RUN+="/bin/systemctl stop [email protected]%k.service"
これはトリックを行うようです!このようなものをデバッグするためのいくつかの便利なコマンド:
udevadm control -l debug
/var/log/syslog
への詳細ログをオンにします 何が起こっているかを確認できます。udevadm control --reload-rules
rules.d ディレクトリ内のファイルを変更した後 (必要ではないかもしれませんが、問題はありません...).systemctl daemon-reload
systemd ユニット ファイルを変更した後。
解決策 2:
新しい簡潔な systemd
があります fstab
で使用できる auto-mount オプション これにより、標準化されたすべてのマウント許可オプションを使用できます。これは次のようになります:
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:
pmount の使用 、systemd、および Mike Blackwell のアプローチにより、すべてを簡素化できます:
/etc/systemd/system/[email protected]
[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 [email protected]%k.service"
ACTION=="remove",KERNEL=="sd[a-z][0-9]*",SUBSYSTEMS=="usb",RUN+="/bin/systemctl stop [email protected]%k.service"
HTH とマイクに感謝します。
解決策 4:
スクリプトを @MikeBlackwell から次のように変更しました:
/dev/sd[a-z]
だけでなく、複数の文字にまたがるデバイス名を認識する しかし/dev/sd[a-z]*
;多くの場合、スピンドルの数が多いサーバーに当てはまります。- 自動マウントされたドライブのリストを
/var/log/usb-mount.track
で追跡します - アクションを
/var/log/messages
に記録します タグ付き usb-mount.sh - ラベルが割り当てられていないドライブ (空?) で問題が発生しないように、マウント ポイントのデバイス ラベルをデバイス名の前に付けます:
/media/sdd2_usbtest
、/media/sdd2_
- ファイルを適切に配置し、必要に応じて元に戻すためのラッパー スクリプトが含まれています
@MikeBlackwell はすでにほとんどの面倒な作業を行っているため、書き直さないことにしました。必要な変更を加えただけです。私は、彼の名前と元の回答の URI を見つけて彼の仕事を認めました。
https://github.com/raamsri/automount-usb で見つけてください
解決策 5:
私はウォーレン・ヤングの答えに行きます いくつか変更を加えました
ドライブの環境の評価からエラーが発生していたため、スペース保護を追加しました。
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