Автоматическое подключение USB-накопителей с помощью systemd

27

Мы обновляем наши серверы с очень устаревшего дистрибутива на современную систему на основе Debian Jessie, включая lightdm / xfce и, конечно, systemd (и udisks2). Одним из препятствий является автоматическое подключение USB-накопителей. Мы привыкли делать это с помощью некоторых правил udev. Старые правила почти все еще работают - точка монтирования создается и диск монтируется нормально, но через несколько секунд systemd делает что-то, что нарушает монтирование, поэтому последующие попытки доступа приводят к ошибкам «Конечная точка транспорта не подключена».

Ручная установка привода через командную строку работает нормально. То же самое можно сказать и о файловом менеджере (thunar и thunar-volman, который, в свою очередь, использует udisks2). Но это не жизнеспособные варианты - эти системы в основном работают без головы, поэтому thunar обычно не работает. Нам нужно иметь возможность подключать диски для автоматического резервного копирования на основе cron.

Я подумал, что изменение сценария udev для запуска отдельного задания, которое ждет несколько секунд, прежде чем выполнять монтирование, может сработать, но systemd, похоже, старается изо всех сил, чтобы предотвратить это - он каким-то образом все еще ожидает завершения отдельного задания, прежде чем продолжается.

Возможно, наличие сценария udev щекочет udisks2 как-то правильный подход? Я в растерянности, поэтому любые советы очень ценятся.

Майк Блэквелл
источник
1
Только косвенно связаны, но ... Вы ставите xfce на сервер?
Парфянский выстрел
Ах, я использовал термин «сервер» довольно свободно ... Все взаимодействие пользователя с системой происходит через веб-приложение, доступ к которому обычно осуществляется через браузер по сети. Но некоторые клиенты предпочитают не сетевое решение, поэтому мы запускаем Chrome на консоли в своего рода режиме киоска. (Это также удобно для отладки проблем конфигурации сети, вы можете подключить монитор / мышь / клавиатуру и получить доступ к основным диагностическим инструментам в веб-приложении, не требуя учетных данных для входа в Linux). Вероятно, есть более легкое решение по весу, чем lightdm / xfce, но это было проще всего настроить ...
Майк Блэквелл,
Для тех, кому нужны правила systemd-udevd, непосредственно запускающие скрипт: у меня было это; Некоторое время он работал, но в какой-то момент скрипт прекратил работать, если udevd был запущен автоматически. Остановитесь и перезапустите из командной строки, и все будет хорошо. Кроме того, он никогда не работал хорошо с NTFS + FUSE, потому что udev обнаружил, что у него есть длительный дочерний процесс (ntfs-3g), и убил его после 60-х годов. Итог: правила udev, напрямую запускающие скрипт - пустая трата времени. Вместо этого используйте правила udev и службу systemd, как указано в ответах. Тогда вам также не придется иметь дело с пространствами имен (MountFlags = slave).
Марк
У меня была похожая проблема с сценарием, запущенным udev, который не мог установить сетевые подключения. Приведенное ниже решение использования systemd тоже сработало - спасибо!
Квентин Стаффорд-Фрейзер

Ответы:

28

После нескольких неудачных попыток я понял это. Ключ заключается в добавлении службы модуля systemd между udev и сценарием монтирования.

(Для справки, я не смог заставить это работать, используя 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 запускают и останавливают службу модуля 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 dir (может и не понадобиться, но не повредит ...).
  • systemctl daemon-reload после того, как вы измените системные файлы модуля.
Майк Блэквелл
источник
4
Вау. Это круто. Хотел бы я дать несколько голосов! Единственное, что мне пришлось изменить, это то, что в моей системе, blkidпохоже, не извлекается ID_FS_LABEL, поэтому я просто использовал DEVBASEвместо, а не LABELдля создания MOUNT_POINTвместо.
Трэвис Григгс
Можно ли изменить эту настройку для работы с устройствами ATA / SCSI? См .: serverfault.com/q/825779/297059
user339676
@Travis - Вы можете использовать udevadmвместо blkid. Это дает гораздо больше деталей, а также дополнительную информацию. (например, udevadm info --query=property --name=sda1)
user339676
это не работает при загрузке, если USB-устройство уже подключено. Любые идеи?
Михал Артазов
Когда нулевые блоки не установлены, при размонтировании очистка может выдать ошибку, подобную /usr/bin/find: '/media/*': No such file or directory. Очистка может использовать дополнительную проверку, как if [ "$f" != "/media/*" ]; thenперед запуском find.
Pro Backup
12

есть новая, краткая 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

а затем оба или один из следующих:

  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сделал трюк для меня
Филипп Гачу
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_
  • включены скрипты-обертки для правильного размещения файлов и отмены при необходимости

Поскольку @MikeBlackwell уже проделал большую часть тяжелой работы, я решил не переписывать ее; только что сделал необходимые изменения. Я признал его работу, обнаружив его имя и URI оригинального ответа.

Найдите его по адресу https://github.com/raamsri/automount-usb

шесть-к
источник
2

Используя подход 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 и спасибо тебе Майк.

Эрик В.
источник
0

Я бы пошел с ответом Уоррена Янга, у меня есть несколько изменений, которые я сделал в

Я добавил некоторую защиту пространства, так как она давала ошибки от оценки среды для накопителя.

Я добавил раздел 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
penguinjeff
источник
Вы можете описать, чем отличается первоначальный ответ от вашего, в нескольких словах, чтобы сделать его более полезным. PS: Уоррена Янга не было ответа; возможно, вы имели в виду ответ Майка Блэквелла, который был отредактирован?
Амир