Мы ' повторное обновление наших серверов с очень устаревшего дистрибутива до современной системы на основе Debian Jessie, включая lightdm / xfce и, конечно же, systemd (и udisks2). Одна из проблем - автоматическое монтирование USB-накопителей. Раньше мы добивались этого с помощью некоторых правил udev. Старые правила почти все еще работают - точка монтирования создается и диск монтируется нормально, но через несколько секунд systemd делает что-то, что прерывает монтирование, поэтому последующие попытки доступа приводят к ошибкам «Транспортная конечная точка не подключена».
Установка диска вручную через командную строку работает нормально. То же самое и с файловым менеджером (thunar и thunar-volman, который, в свою очередь, использует udisks2). Но это нежизнеспособные варианты - эти системы в основном работают без головы, поэтому thunar обычно не работает. Нам нужно иметь возможность подключать диски для автоматического резервного копирования на основе cron.
Я подумал, что изменение сценария udev для создания отдельного задания, которое ждет несколько секунд перед выполнением монтирования, может помочь, но systemd, похоже, изо всех сил старается предотвратить это - он каким-то образом все еще ждет отдельного задания чтобы закончить, прежде чем продолжить.
Возможно, скрипт udev щекочет udisks2 как-то правильно? Я в растерянности, поэтому буду благодарен за любой совет.
После нескольких фальстартов я понял это. Ключ состоит в том, чтобы добавить службу модуля systemd между udev и сценарием монтирования.
(Для записи, мне не удалось заставить это работать с помощью udisks2 (через что-то вроде udisksctl mount -b / dev / sdb1
) вызывается либо непосредственно из правила udev, либо из файла модуля systemd. Похоже, возникла гонка, и узел устройства не совсем готов, что привело к Ошибка поиска объекта для устройства / 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 на 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. появилась новая, краткая 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
Я изменил сценарий из @MikeBlackwell, чтобы:
/ dev / sd [az]
, но / dev / sd [аз] *
; часто бывает с серверами с большим количеством шпинделей. /var/log/usb-mount.track
/ var / log / сообщения
с тегом usb-mount.sh / media / sdd2_usbtest
, / media / sdd2_
Поскольку @MikeBlackwell уже выполнил большую часть тяжелой работы, я решил не переписать; просто внес необходимые изменения. Я признал его работу по обнаружению его имени и URI исходного ответа.
Найдите его на https://github.com/raamsri/automount-usb
Я бы пошел с ответом Уоррена Янга. У меня есть несколько изменений, которые я внес в
Я добавил некоторую защиту пространства, так как он давал ошибки из-за оценки среды для диска.
Я добавил раздел для 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
Используя pmount , systemd и подход Майка Блэквелла, вы можете упростить все (скрытое) и спасибо, Майк.