หลังจากการผิดพลาดหลายครั้งฉันก็รู้เรื่องนี้ กุญแจสำคัญคือการเพิ่มบริการหน่วย systemd ระหว่าง udev และสคริปต์การติดตั้ง
(สำหรับเร็กคอร์ดฉันไม่สามารถใช้งานได้โดยใช้ udisks2 (ผ่านบางอย่างudisksctl mount -b /dev/sdb1
) ที่เรียกว่าโดยตรงจากกฎ udev หรือจากไฟล์ systemd unit ดูเหมือนว่าจะมีสภาพการแข่งขันและโหนดอุปกรณ์ยังไม่พร้อม ทำให้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 unit เราใช้ไวยากรณ์ชื่อไฟล์ "@" เพื่อให้เราสามารถส่งชื่ออุปกรณ์เป็นอาร์กิวเมนต์
/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 บางกฎเริ่มต้นและหยุดบริการ systemd unit บน hotplug / unplug:
/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 dir (อาจไม่จำเป็น แต่ไม่สามารถทำร้าย ... )
systemctl daemon-reload
หลังจากคุณแก้ไขไฟล์ systemd unit