Copiando volumes para sistemas de armazenamento por meio de um servidor Linux usando XCOPY

Acontece que você precisa obter uma cópia completa de um volume dentro de um sistema de armazenamento de dados (DSS), não um instantâneo, um clone, mas um volume completo. Mas nem sempre o sistema de armazenamento permite que isso seja feito internamente por meios próprios. Parece que a única opção é copiar através do servidor, mas neste caso todo o volume de dados será conduzido através do próprio servidor, da rede para o sistema de armazenamento e para as portas de armazenamento, carregando todos esses componentes. Mas existem comandos SCSI que podem permitir que você faça tudo dentro do próprio sistema de armazenamento, e se o seu sistema suportar VAAI do VMware, então é quase 100% que o comando XCOPY (EXTENDED COPY) é suportado, que informa ao array o que e onde copiar, sem envolver servidor de processo e rede.

Parece que tudo deveria ser simples, mas não consegui encontrar nenhum script pronto de imediato, então tive que reinventar a roda. Linux foi selecionado para o sistema operacional do servidor e o comando ddpt (http://sg.danny.cz/sg/ddpt.html) foi selecionado como ferramenta de cópia. Usando essa combinação, você pode copiar qualquer volume de qualquer sistema operacional, já que a cópia ocorre bloco por bloco no sistema de armazenamento. Como é necessário copiar bloco por bloco e o número de blocos deve ser contado, o comando blockdev foi utilizado para contar o número dessas iterações. O tamanho máximo do bloco foi obtido experimentalmente; na verdade, o ddpt não funcionou com um bloco grande. O resultado foi o seguinte script bastante simples:

#!/bin/bash
# first parameter = input device
# second parameter = output device
# device size must be the same
# changing bs variable can reduce speed, max speed should be at bs=32768. 32768 is max setting, lower settings should be calculated dividing by 2

set -o nounset
bs=32768
s=`blockdev --getsz $1`
i=0
while [ $i -le $s ]
do
ddpt of=$2 bs=512 oflag=xcopy,direct if=$1 iflag=xcopy,direct count=$bs verbose=-1 skip=$i seek=$i
i=$(( $i+$bs ))
done

Vamos fazer uma pequena verificação! Bem, por ser pequeno, um arquivo de 1 TB não foi criado e verificado rapidamente pelo md5sum :)

root@sales-demo-05:/home/vasilyk# blockdev --getsz /dev/mapper/mpathfs
2516582400
root@sales-demo-05:/home/vasilyk# blockdev --getsz /dev/mapper/mpathfr
2516582400
root@sales-demo-05:/home/vasilyk# mount /dev/mapper/mpathfs /xcopy_source/
mount: /xcopy_source: wrong fs type, bad option, bad superblock on /dev/mapper/mpathfs, missing codepage or helper program, or other error.
root@sales-demo-05:/home/vasilyk# mkfs /dev/mapper/mpathfs
mke2fs 1.44.1 (24-Mar-2018)
Discarding device blocks: done
Creating filesystem with 314572800 4k blocks and 78643200 inodes
Filesystem UUID: bed3ea00-c181-4b4e-b52e-d9bb498be756
Superblock backups stored on blocks:
        32768, 98304, 163840, 229376, 294912, 819200, 884736, 1605632, 2654208,
        4096000, 7962624, 11239424, 20480000, 23887872, 71663616, 78675968,
        102400000, 214990848

Allocating group tables: done
Writing inode tables: done
Writing superblocks and filesystem accounting information: done

root@sales-demo-05:/home/vasilyk# mount /dev/mapper/mpathfs /xcopy_source/
root@sales-demo-05:/home/vasilyk# ls -l /xcopy_source/
total 16
drwx------ 2 root root 16384 Aug 19 15:35 lost+found
root@sales-demo-05:/home/vasilyk# head -c 1T </dev/urandom > /xcopy_source/1TB_file
root@sales-demo-05:/home/vasilyk# ls -l /xcopy_source/
total 1074791444
-rw-r--r-- 1 root root 1099511627776 Aug 19 17:25 1TB_file
drwx------ 2 root root         16384 Aug 19 15:35 lost+found
root@sales-demo-05:/home/vasilyk# umount /xcopy_source
root@sales-demo-05:/home/vasilyk# mount /dev/mapper/mpathfr /xcopy_dest/
mount: /xcopy_dest: wrong fs type, bad option, bad superblock on /dev/mapper/mpathfr, missing codepage or helper program, or other error.
root@sales-demo-05:/home/vasilyk# cat xcopy.sh
#!/bin/bash
# first parameter = input device
# second parameter = output device
# device size must be the same
# changing bs variable can reduce speed, max speed should be at bs=32768. 32768 is max setting, lower settings should be calculated dividing by 2

bs=32768
s=`blockdev --getsz $1`
i=0
while [ $i -le $s ]
do
ddpt of=$2 bs=512 oflag=xcopy,direct if=$1 iflag=xcopy,direct count=$bs verbose=-1 skip=$i seek=$i
i=$(( $i+$bs ))
done
root@sales-demo-05:/home/vasilyk# time ./xcopy.sh /dev/mapper/mpathfs /dev/mapper/mpathfr
real    11m30.878s
user    2m3.000s
sys     1m11.657s

O que estava acontecendo no sistema de armazenamento naquele momento:

Copiando volumes para sistemas de armazenamento por meio de um servidor Linux usando XCOPY
Vamos continuar com o Linux.

root@sales-demo-05:/home/vasilyk# mount /dev/mapper/mpathfr /xcopy_dest/
root@sales-demo-05:/home/vasilyk# ls -l /xcopy_dest/
total 1074791444
-rw-r--r-- 1 root root 1099511627776 Aug 19 17:25 1TB_file
drwx------ 2 root root         16384 Aug 19 15:35 lost+found
root@sales-demo-05:/home/vasilyk# mount /dev/mapper/mpathfs /xcopy_source/
root@sales-demo-05:/home/vasilyk# md5sum /xcopy_source/1TB_file
53dc6dfdfc89f099c0d5177c652b5764  /xcopy_source/1TB_file
root@sales-demo-05:/home/vasilyk# md5sum /xcopy_dest/1TB_file
53dc6dfdfc89f099c0d5177c652b5764  /xcopy_dest/1TB_file
root@sales-demo-05:/home/vasilyk#

Deu tudo certo, mas teste e use por sua conta e risco! Como volume de origem, é melhor tirar instantâneos, para começar.

Fonte: habr.com

Adicionar um comentário