Drives USB automount com systemd

27

Estamos atualizando nossos servidores de uma distribuição muito desatualizada para um sistema moderno baseado no Debian Jessie, incluindo lightdm / xfce e, claro, systemd (e udisks2). Um ponto de discórdia é a montagem automática de unidades USB. Costumávamos fazer isso com algumas regras do udev. As regras antigas quase ainda funcionam - o ponto de montagem é criado e a unidade é montada corretamente, mas após alguns segundos o systemd está fazendo algo que interrompe a montagem, portanto, tentativas de acesso subsequentes resultam em erros "O terminal de transporte não está conectado".

A montagem manual da unidade via linha de comando funciona bem. O mesmo acontece com deixar um gerenciador de arquivos (thunar e thunar-volman, que por sua vez usa o udisks2). Mas essas não são opções viáveis ​​- esses sistemas geralmente funcionam sem cabeça, portanto, thunar normalmente não está funcionando. Precisamos ser capazes de conectar unidades de disco para backups autônomos baseados em cron.

Eu pensei que modificar o script do udev para gerar um trabalho desanexado que aguarde alguns segundos antes de executar a montagem pode resolver o problema, mas o systemd parece se esforçar para evitar isso - de alguma forma ainda espera que o trabalho desanexado termine antes continuando.

Talvez fazer com que o script do udev faça cócegas no udisks2 de alguma forma é a abordagem correta? Estou perdida, então qualquer conselho muito apreciado.

Mike Blackwell
fonte
1
Apenas relacionado tangencialmente, mas ... Você está colocando o xfce em um servidor?
Parthian Shot
Ah, usei o termo "servidor" de maneira bastante vaga ... Toda a interação do usuário com o sistema é por meio de um aplicativo da Web, normalmente acessado por um navegador na rede. Como alguns clientes preferem uma solução que não seja de rede, executamos o Chrome no console em uma espécie de modo de quiosque. (Isso também é útil para depurar problemas de configuração de rede, você pode conectar um monitor / mouse / teclado e acessar as ferramentas básicas de diagnóstico no aplicativo Web sem precisar de credenciais de login no Linux). Há provavelmente uma solução mais leve do que LightDM / xfce, mas esta foi mais simples de configurar ...
Mike Blackwell
Para quem deseja que as regras systemd-udevd executem diretamente um script: eu tinha isso; funcionou por um tempo, mas em algum momento parou de executar o script se o udevd tivesse sido iniciado automaticamente. Pare e reinicie a partir da linha de comando, e tudo ficará bem. Além disso, nunca funcionou bem com o NTFS + FUSE porque o udev detectou que tinha um processo filho de longa duração (ntfs-3g) e o matou após os 60 anos. Conclusão: as regras do udev que executam diretamente um script são uma perda de tempo. Siga as regras do udev e um serviço systemd, como observado nas respostas. Então você não precisa lidar com espaços para nome (MountFlags = slave).
Mark
Eu tive um problema semelhante de um script iniciado pelo udev não poder fazer conexões de rede. A solução abaixo de usar o systemd também funcionou para isso - obrigado!
Quentin Stafford-Fraser

Respostas:

28

Depois de várias partidas falsas, eu descobri isso. A chave é adicionar um serviço de unidade systemd entre o udev e um script de montagem.

(Para o registro, eu não consegui fazer isso funcionar usando o udisks2 (via algo como udisksctl mount -b /dev/sdb1) chamado diretamente de uma regra do udev ou de um arquivo de unidade do systemd. Parece haver uma condição de corrida e o nó do dispositivo não está pronto , resultando em Error looking up object for device /dev/sdb1. Infelizmente, uma vez que o udisks2 pode cuidar de toda a bagunça do ponto de montagem ...)

O trabalho pesado é feito por um script de shell, que cuida da criação e remoção dos pontos de montagem e da montagem e desmontagem das unidades.

/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

O script, por sua vez, é chamado por um arquivo de unidade systemd. Usamos a sintaxe do nome de arquivo "@" para que possamos passar o nome do dispositivo como argumento.

/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

Finalmente, algumas regras do udev iniciam e param o serviço da unidade systemd no 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"

Isso parece fazer o truque! Alguns comandos úteis para depurar coisas como esta:

  • udevadm control -l debugativa o log detalhado para /var/log/syslogque você possa ver o que está acontecendo.
  • udevadm control --reload-rules depois de modificar os arquivos no diretório rules.d (pode não ser necessário, mas não pode prejudicar ...).
  • systemctl daemon-reload depois de modificar os arquivos da unidade systemd.
Mike Blackwell
fonte
4
Uau. Isso é incrível. Gostaria de poder dar vários votos positivos! A única coisa que precisei modificar foi que, no meu sistema, blkidnão parece extrair um ID_FS_LABEL, então apenas usei o em DEVBASEvez de LABELconstruí-lo MOUNT_POINT.
Travis Griggs
Esta configuração pode ser modificada para funcionar com dispositivos ATA / SCSI? Veja: serverfault.com/q/825779/297059
user339676
@ Travis - Você pode usar em udevadmvez de blkid. Dá muito mais detalhes, além de informações adicionais. (eg, udevadm info --query=property --name=sda1)
user339676
isso não funciona bem na inicialização, se um dispositivo USB já estiver conectado. Alguma ideia?
Michal Artazov
Quando nullglobs não estão definidos, ao desmontar, a limpeza pode gerar um erro como /usr/bin/find: '/media/*': No such file or directory. A limpeza pode usar uma verificação adicional como if [ "$f" != "/media/*" ]; thenantes da execução find.
Pro Backup
12

existe uma nova e sucinta systemdopção de montagem automática, que pode ser usada com a fstabqual você pode usar todas as opções de permissão de montagem padronizadas, e fica assim:

  x-systemd.automount

um exemplo disso em uma fstablinha:

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

a noautoopção significará que não tentará ser montado na inicialização, como acontece com o software mais antigo autofs.

Depois de adicionar uma nova x-systemd.automountlinha, fstabvocê precisará executar:

  sudo systemctl daemon-reload

e então ambos, ou um, do seguinte:

  sudo systemctl restart remote-fs.target
  sudo systemctl restart local-fs.target

para mais informações sobre isso:

https://wiki.archlinux.org/index.php/Fstab#Automount_with_systemd

etc-infinito
fonte
sudo systemctl restart local-fs.targetfez o truque para mim
Philippe Gachoud
2

Modifiquei o script de @MikeBlackwell para:

  • reconhecer nomes de dispositivos que abrangem vários personagens, não apenas /dev/sd[a-z], mas /dev/sd[a-z]*; geralmente é o caso de servidores com maior número de eixos.
  • acompanhe a lista de unidades montadas automaticamente em /var/log/usb-mount.track
  • registre as ações /var/log/messagescom a tag usb-mount.sh
  • nome do dispositivo prefixo com o rótulo de dispositivo para o ponto de montagem para não correr em problemas com unidades que não tenham sido atribuídos um rótulo (esvaziar?): /media/sdd2_usbtest,/media/sdd2_
  • scripts de wrapper incluídos para colocar os arquivos adequadamente e desfazer, se necessário

Como o @MikeBlackwell já fez a maior parte do trabalho pesado, escolhi não reescrevê-lo; acabou de fazer as alterações necessárias. Eu reconheci seu trabalho vendo seu nome e URI da resposta original.

Encontre-o em https://github.com/raamsri/automount-usb

six-k
fonte
2

Usando a abordagem pmount , systemd e Mike Blackwell, você pode simplificar tudo:

/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 usb-mount@%k.service"
ACTION=="remove",KERNEL=="sd[a-z][0-9]*",SUBSYSTEMS=="usb",RUN+="/bin/systemctl stop usb-mount@%k.service"

HTH e obrigado Mike.

Eric V.
fonte
0

Eu iria com a resposta de Warren Young. Tenho algumas mudanças que fiz para

Adicionei alguma proteção de espaço, pois estava dando erros na avaliação do ambiente para a unidade.

Adicionei uma seção ao chmod de um disco USB para que todos os usuários tenham acesso total a discos não NTFS ou 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
fonte
Você pode descrever o que é diferente entre a resposta original e a sua em poucas palavras, para torná-la mais útil. PS: não houve resposta de Warren Young; talvez você quis dizer a resposta de Mike Blackwell que foi editada?
Amir