Automount USB ไดรฟ์พร้อม systemd


27

เรากำลังอัปเดตเซิร์ฟเวอร์ของเราจาก distro ที่ล้าสมัยไปเป็นระบบที่ใช้ Debian Jessie ที่ทันสมัยรวมถึง lightdm / xfce และแน่นอน systemd (และ udisks2) จุดต่อหนึ่งจุดคือการเมานท์ไดรฟ์ USB เราเคยทำสิ่งนี้ให้สำเร็จด้วยกฎของ udev กฎเก่ายังคงใช้งานได้ - จุดเมานต์จะถูกสร้างขึ้นและไดรฟ์ติดตั้งได้ดี แต่หลังจากไม่กี่วินาที systemd กำลังทำอะไรบางอย่างที่ทำให้เมาท์ดังนั้นความพยายามในการเข้าถึงที่ตามมาทำให้เกิดข้อผิดพลาด

การติดตั้งไดรฟ์ด้วยตนเองผ่านทางบรรทัดคำสั่งทำงานได้ดี ดังนั้นให้ตัวจัดการไฟล์ (thunar และ thunar-volman ซึ่งใช้ udisks2) แต่นั่นไม่ใช่ตัวเลือกที่ทำงานได้ - ระบบเหล่านี้ส่วนใหญ่ทำงานโดยไม่ใช้หัวดังนั้น thunar จึงไม่ได้ทำงานตามปกติ เราจำเป็นต้องสามารถเสียบดิสก์ไดรฟ์สำหรับการสำรองข้อมูล cron แบบอัตโนมัติ

ฉันคิดว่าการแก้ไขสคริปต์ udev เพื่อวางไข่งานเดี่ยวซึ่งรอสองสามวินาทีก่อนที่จะทำการเมานท์อาจทำเคล็ดลับได้ แต่ systemd ดูเหมือนจะออกนอกทางเพื่อป้องกันสิ่งนี้ อย่างต่อเนื่อง

บางทีการมีสคริปต์ udev tickle udisks2 เป็นวิธีการที่ถูกต้องหรือไม่? ฉันกำลังแพ้ดังนั้นคำแนะนำใด ๆ ก็ชื่นชมอย่างมาก


1
มีความเกี่ยวข้องกับสัมผัสเชิงสัมผัส แต่ ... คุณกำลังวาง xfce บนเซิร์ฟเวอร์
คู่ปรับ Shot

อ่าฉันใช้คำว่า "เซิร์ฟเวอร์" ค่อนข้างหลวม ... การโต้ตอบของผู้ใช้กับระบบคือผ่านเว็บแอปซึ่งโดยทั่วไปจะเข้าถึงผ่านเบราว์เซอร์ผ่านเครือข่าย แต่ลูกค้าบางคนชอบโซลูชันที่ไม่ใช่เครือข่ายเราจึงใช้งาน Chrome บนคอนโซลในโหมดคีออสก์ (นอกจากนี้ยังมีประโยชน์สำหรับการดีบักปัญหาการกำหนดค่าเครือข่ายคุณสามารถเชื่อมต่อจอภาพ / เมาส์ / คีย์บอร์ดและเข้าถึงเครื่องมือการวินิจฉัยขั้นพื้นฐานในเว็บแอปโดยไม่ต้องใช้ข้อมูลประจำตัวในการเข้าสู่ระบบ Linux) อาจมีวิธีแก้ปัญหาน้ำหนักเบากว่า lightdm / xfce แต่นี่เป็นวิธีที่ง่ายที่สุดในการตั้งค่า ...
Mike Blackwell

สำหรับผู้ที่ต้องการ systemd-udevd กฎโดยตรงเรียกใช้สคริปต์: ฉันมีสิ่งนี้; มันใช้งานได้ซักพัก แต่ในบางจุดก็หยุดเรียกใช้สคริปต์ถ้า udevd เริ่มทำงานโดยอัตโนมัติ หยุดและเริ่มต้นใหม่จากบรรทัดคำสั่งและมันก็ใช้ได้ ยิ่งไปกว่านั้นมันไม่เคยทำงานได้ดีกับ NTFS + FUSE เพราะ udev ตรวจพบว่ามีกระบวนการลูกที่ทำงานเป็นเวลานาน (ntfs-3g) และฆ่ามันหลังจากยุค 60 บรรทัดล่าง: กฎ udev โดยตรงที่เรียกใช้สคริปต์โดยตรงจะเสียเวลา ไปกับกฎ udev และบริการ systemd แทนดังที่ระบุไว้ในคำตอบ จากนั้นคุณไม่ต้องจัดการกับเนมสเปซ (MountFlags = slave) เช่นกัน
ทำเครื่องหมาย

ฉันมีปัญหาคล้ายกันของสคริปต์ที่เริ่มโดย udev ไม่สามารถทำการเชื่อมต่อเครือข่ายได้ วิธีแก้ปัญหาด้านล่างของการใช้ systemd ก็ใช้ได้เช่นกัน - ขอบคุณ!
เควนติน Stafford-Fraser

คำตอบ:


28

หลังจากการผิดพลาดหลายครั้งฉันก็รู้เรื่องนี้ กุญแจสำคัญคือการเพิ่มบริการหน่วย 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

4
ว้าว. นี่มันเจ๋งมาก. หวังว่าฉันจะให้ upvotes หลายรายการ! สิ่งเดียวที่ฉันต้องแก้ไขคือในระบบของฉันblkidดูเหมือนจะไม่แตกไฟล์ID_FS_LABELดังนั้นฉันจึงใช้DEVBASEแทนที่จะLABELสร้างสิ่งก่อสร้างMOUNT_POINTแทน
Travis Griggs

การตั้งค่านี้สามารถปรับเปลี่ยนให้ทำงานกับอุปกรณ์ ATA / SCSI ได้หรือไม่ โปรดดู: serverfault.com/q/825779/297059
user339676

@Travis - คุณสามารถใช้แทนudevadm blkidมันให้รายละเอียดมากขึ้นรวมทั้งข้อมูลเพิ่มเติม (เช่น, udevadm info --query=property --name=sda1)
user339676

สิ่งนี้ใช้งานไม่ได้กับการบู๊ตหากอุปกรณ์ usb เชื่อมต่ออยู่แล้ว ความคิดใด ๆ
Michal Artazov

เมื่อ nullglobs /usr/bin/find: '/media/*': No such file or directoryไม่ได้ตั้งบนยกเลิกการทำความสะอาดสามารถสร้างข้อผิดพลาดเหมือน Cleanup สามารถใช้การตรวจสอบเพิ่มเติมเช่นการทำงานก่อนif [ "$f" != "/media/*" ]; then find
สำรองข้อมูล Pro

12

มีsystemdตัวเลือกการเมาท์อัตโนมัติแบบใหม่ที่กระชับซึ่งสามารถใช้งานได้fstabซึ่งอนุญาตให้คุณใช้ตัวเลือกการอนุญาตเมาท์มาตรฐานทั้งหมดและดูเหมือนว่า:

  x-systemd.automount

ตัวอย่างของมันในfstabบรรทัด:

  /dev/sdd1   /mnt/hitachi-one     auto     noauto,x-systemd.automount     0 2

ตัวเลือกที่จะหมายความว่ามันจะไม่พยายามที่จะติดตั้งที่บูตเช่นเดียวกับซอฟแวร์ที่มีอายุมากกว่าnoautoautofs

หลังจากเพิ่ม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


sudo systemctl restart local-fs.targetทำกลอุบายให้ฉัน
Philippe Gachoud

2

ฉันได้แก้ไขสคริปต์จาก @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_
  • รวมสคริปต์ wrapper เพื่อวางไฟล์อย่างเหมาะสมและยกเลิกถ้าจำเป็น

ตั้งแต่ @MikeBlackwell ได้ทำการยกที่หนักที่สุดมาแล้วฉันจึงเลือกที่จะไม่เขียนมันใหม่ เพิ่งทำการเปลี่ยนแปลงที่จำเป็น ฉันยอมรับว่างานของเขาเห็นชื่อและ URI ของคำตอบดั้งเดิม

ค้นหาได้ที่https://github.com/raamsri/automount-usb


2

เมื่อใช้จำนวนเงิน , 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 และขอบคุณไมค์


0

ฉันต้องการคำตอบของ Warren Young ฉันมีการเปลี่ยนแปลงเล็กน้อยที่ฉันทำ

ฉันได้เพิ่มการป้องกันพื้นที่บางส่วนเนื่องจากมีข้อผิดพลาดจากสภาพแวดล้อมของไดรฟ์

ฉันเพิ่มส่วนหนึ่งลงใน chmod ดิสก์ usb เพื่อให้ผู้ใช้ทุกคนสามารถเข้าถึงดิสก์ที่ไม่ใช่ 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

คุณอาจต้องการอธิบายความแตกต่างระหว่างคำตอบดั้งเดิมกับคำของคุณเพื่อให้มีประโยชน์มากขึ้น PS: ไม่มีคำตอบจากวอร์เรนยัง; บางทีคุณอาจหมายถึงคำตอบของ Mike Blackwell ที่ถูกแก้ไข?
อาเมียร์
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.