kitoy.me/content/blog/sauvegarde orangepi.md

543 lines
16 KiB
Markdown
Raw Normal View History

2020-04-19 09:06:04 +02:00
+++
template = "articles.html"
2021-01-13 01:55:57 +01:00
title = "Sauvegarde système de l'Orangepi sous debian/yunohost."
description = "Sauvegarder le système de sa carte ARM OrangePi avec debian sur clé usb"
2020-04-19 09:06:04 +02:00
date = 2019-12-09
+++
### Le problème
Ou plutôt les problèmes, car il peut y en avoir plusieurs; par exemple le
système sur la carte ne boot plus. On a fait une bêtise, on la débranché car on
a voulu la déplacer ou le fil s'est débranché par accident, et aller cette fois
là c'est la galère. C'est généralement long et fastidieux de récupérer les
données sur sa carte SD ou autre, j'ai donc fait un script qui fait une
sauvegarde globale du système sur un disque ou une clé usb que l'on devra
autoriser auparavant, histoire que dès que l'on branche une clé usb, ça n'envoie
pas les données dessus. Il y a bien évidement des solutions de sauvegarde en
ligne mais ayant une connexion avec peu de débit, ça peut prendre beaucoup de
temps, la connexion peut couper et alors c'est des complications, pour moi ce n'était pas très adapté.
### Le principe
Le principe c'est que le script a deux fonctionnalitées
2021-01-13 01:55:57 +01:00
* Une pour ajouter les supports de stockages où la sauvegarde sera envoyé. Le script formate la partition et il ajoute l'UUID à la liste des clés ou disques autorisé a recevoir les données.
Pour l'utiliser on lancera la commande **saveonmedia create**
* Une pour écouter avec udev les medias branchés et réaliser la sauvegarde quand elle reconnaît L'UUID d'un disque.
c'est la commande **saveonmedia make** qui fera ça.
Pour la liste, un bête fichier texte suffit pour ajouter les supports de stockage ligne par ligne, on se basera sur les UUID des partitions sur le disque qui sont normalement identifier de manière unique. Pour réaliser la sauvegarde le script lancera un script bash qui est généralement plus pratique si l'on veut ajouter des taches à effectuer ou personnaliser le truc un peu comme on veut.
2020-04-19 09:06:04 +02:00
``` python
#!/usr/bin/python3
# -*- coding: utf-8 -*-
import pyudev
import subprocess
import sys
import os
import socket
from email.mime.text import MIMEText
from subprocess import Popen, PIPE
def print_help():
print ("""
##### saveonusb Sauvergarde sur support USB ####
! A executer avec l'utilisateur avec des droits root (sudo) ou en root !
Pour autoriser un disque (clé usb, disque usb)
Lancez la commande :
./saveonusb create
Pour effectuer l'ecoute puis la sauvegarde:
./saveonusb make
""")
def send_msg(subject, message):
# Permet d'envoyer des mail a root via sendmail.
msg = MIMEText(message)
msg["From"] = "sauvegarde@"+socket.gethostname()
msg["To"] = "root@"+socket.gethostname()
msg["Subject"] = subject
p = Popen(["/usr/sbin/sendmail", "-t", "-oi"], stdin=PIPE, universal_newlines=True)
p.communicate(msg.as_string())
def clean_df_out(df_out):
# Ici on tri les enleve les partitions de la sortir de df qui ne sont pas des supports
# de stockages
#
df_out = df_out.split('\n')
lines=[]
for line in df_out:
if line[0:4] == '/dev':
lines.append(line)
line_clean = list()
lines_clean = list()
# On crée la liste des champs de df
for line in lines:
line = line.split(' ')
for champ in line:
if champ != '':
line_clean.append(champ)
lines_clean.append(line_clean.copy())
line_clean.clear()
return lines_clean
def getMinSpaceRequirement():
# df -k A ce schema de sorti
# Filesystem | 1K-blocks | Used | Avail | Capacity | Mounted on
#
# Pour obtenir l'espace minimum on suppose que l'on veut sauvegarder toutes les
# données de tout les supports de stockages.
df = subprocess.check_output(['/bin/df', '-k']).decode('utf8')
df_list = clean_df_out(df)
print (df_list)
space_used = 0
for line in df_list:
space_used = space_used + int(line[2])
return space_used
def get_uuid(device):
cmd = subprocess.check_output(['sudo', 'blkid', device])
cmd = cmd.decode('utf-8')
for line in cmd.split(' '):
if 'UUID=' == line[0:5]:
return line[6:len(line)-1]
return ''
def make_save(save_directory, storages):
context = pyudev.Context()
monitor = pyudev.Monitor.from_netlink(context)
monitor.filter_by('block')
done = False
with open (storages, 'r') as f:
medias_save = f.read().splitlines()
while not(done):
device = monitor.poll()
if 'ID_FS_TYPE' in device and device.action == "add":
print ("device: {0} medias: {1}".format(device.get('ID_FS_UUID'), medias_save))
if "ynh_save" == device.get('ID_FS_LABEL') and device.get('ID_FS_UUID') in medias_save:
print ("La sauvegarde a commencé")
send_msg("Une sauvegarde a été lancé","Ne retirez pas votre disque ou clé usb. Merçi.")
retcode = subprocess.call("mount " + device.device_node + " "+ save_directory, shell=True)
if retcode != 0:
subject = "Bon ... ça s'est mal passé"
2021-01-13 01:55:57 +01:00
message = """Impossible de monter le disque ou la clé USB :( """
2020-04-19 09:06:04 +02:00
send_msg(subject, message)
retcode = subprocess.call( save_script + " " + save_directory, shell=True)
if retcode != 0:
subject = "Bon ... ça s'est mal passé"
2021-01-13 01:55:57 +01:00
message = """Le script de sauvegarde a remonté une erreur, comment dire ... je le ferai autrement moi """
2020-04-19 09:06:04 +02:00
send_msg(subject, message)
retcode = subprocess.call("umount " + device.device_node, shell=True)
if retcode != 0:
subject = "Bon ... ça s'est a peu près bien passé"
2021-01-13 01:55:57 +01:00
message = """Ça a monté le support et tout, le script pas de problèmes, mais ca ne démonte pas la partition correctement le script ne peut rien faire seul vous pouvez voir ce qu'il se passe en vous connectant en admin à votre machine"""
2020-04-19 09:06:04 +02:00
else:
subject = "Confirmation de sauvegarde"
2021-01-13 01:55:57 +01:00
message = """Votre sauvegarde est prête vous pouvez retirer votre support de stockage en toute sécurité. Rebranchez ce même support quand vous souhaitez effectuer une sauvergarde. Attention! Toutes modifications de la partiton rendra le support invalide"""
2020-04-19 09:06:04 +02:00
send_msg(subject, message)
def create_key_save(storages):
context = pyudev.Context()
monitor = pyudev.Monitor.from_netlink(context)
monitor.filter_by('block')
with open (storages, 'r') as f:
medias_save = f.read().splitlines()
done = False
while not(done):
device = monitor.poll()
if device.device_type == "partition" and device.action == "add":
if "ynh_save" == device.get('ID_FS_LABEL') and device.get('ID_FS_UUID') in medias_save:
subject = "Ce support de stockage est déjà initialisé"
message = "Il n'est pas nécéssaire d'initialisé ce périphérique"
send_msg(subject, message)
else:
size = subprocess.check_output("fdisk -s " +
device.device_node,
shell=True)
size = int(size)
size_requirement = getMinSpaceRequirement()
if size > size_requirement:
done = True
retcode = subprocess.call("mkfs.ext4" + " -F -L ynh_save" +
" "+ device.device_node, shell=True)
if retcode != 0:
subject = "Impossible de formatter la clé"
2021-01-13 01:55:57 +01:00
message = """ Il y a eu un problème durant le formatage du support usb"""
2020-04-19 09:06:04 +02:00
send_msg(subject, message)
# On récupere le nouvel uuid
with open (storages, 'a') as f:
f.write(get_uuid(device.device_node)+ '\n')
subject = "Autorisation d'un support de sauvegarde"
message = """ Vous venez de créer un support pour votre sauvegarde.
Cela s'est deroulé normalement. Pour effectuer une sauvegarde maintenant
2021-01-13 01:55:57 +01:00
rebrancher votre support de stockage; un mail vous signalera quand vous pourrez enlevez votre support en toute sécurité."""
2020-04-19 09:06:04 +02:00
send_msg(subject, message)
done = True
else:
subject = "Votre support de stockage est de taille insufisante !"
2021-01-13 01:55:57 +01:00
message = """Il n'y a pas assez d'espace disque sur le support que
vous venez de brancher; essayer avec une media de plus grande capacité (minimum """ +
2020-04-19 09:06:04 +02:00
str(size_requirement/1000000) + "Gb)"
send_msg(subject, message)
save_directory = "/media/save"
save_script = "/etc/saveonusb/scripts/save.sh"
conf_path = "/etc/saveonusb/"
storages = conf_path+"storages"
if os.geteuid() != 0:
raise OSError("Ce programme doit s'executer avec les droits root")
if not(os.path.exists(save_directory)):
raise OSError("Le repertoire "+ save_directory +" n'existe pas")
if not(os.path.exists(conf_path)):
raise OSError("Le repertoire "+ conf_path +" n'existe pas")
if not(os.path.isfile(save_script)):
raise OSError("Le fichier "+ save_script +" n'existe pas")
if not(os.path.isfile(storages)):
raise OSError("Le fichier "+ storages +" n'existe pas")
if len(sys.argv) > 1:
if sys.argv[1] == 'create':
create_key_save(storages)
if sys.argv[1] == 'make':
make_save(save_directory, storages)
if sys.argv[1] == '--help' or sys.argv[1] == '-h':
print_help()
else:
print_help()
```
On installe pyudev:
``` bash
apt install python3-pyudev
```
On créé les répertoires et les fichiers nécessaires:
```bash
mkdir /media/save
2021-01-13 01:55:57 +01:00
mkdir -p /etc/saveonusb/scripts
2020-04-19 09:06:04 +02:00
touch /etc/saveonusb/storages && touch /etc/saveonusb/scripts/save.sh
chmod +x /etc/saveonusb/scripts/save.sh
```
Je me suis fait tout d'abord un petit script afin de générer
2021-01-13 01:55:57 +01:00
une image prête a être flasher sur une carte SD
2020-04-19 09:06:04 +02:00
``` bash
#!/bin/bash
set -e
#set -x
#Implementer un TRAC error avec exit qui clean quand ca foire
# Implementer des logs
#Fonction pour check les programmes nécéssaire.
#lance apt install quand les programmes manques
# modprobe loop
# Test les medias disponibles et passer en parametre le disk a monter pour mettre l'image
# Demande confirmation pour commencer tout les étapes ou a toutes les étapes
_apt_install_dep()
{
echo "on passe ici avec $1 comme argument"
case $1 in
truncate)
apt-get -y install --no-install-recommends coreutils
;;
mkfs.ext4)
apt-get -y install --no-install-recommends e2fsprogs
;;
*)
apt-get -y install --no-install-recommends $1
;;
esac
}
check_dep()
{
bins=(dd truncate rsync parted mkfs.ext4 )
for i in "${bins[@]}"; do
if ! which "${i}" &> /dev/null; then
echo "${i} command is required"
_apt_install_dep ${i}
fi
done
echo "check_dep ok"
}
create_img() {
local folder=$1
local image=$2
# Espace utilisé sur /
local usage=$(df -BM | grep ^/dev | head -1 | awk '{print $3}' | tr -cd '[0-9]. \n')
# Espace libre
local avaible=$(df -BM $folder | grep ^/dev | head -1 | awk '{print $4}' | tr -cd '[0-9]. \n')
if [[ $usage -gt $avaible ]]; then
echo "No space left"
echo "Required: $usage MB "
echo "Free space in $folder : $avaible MB "
exit 1
fi
#On laisse 1024M d'espace libre sur l'image
local image_size=$(($usage+1024))'M'
truncate -s $image_size $folder/$image
}
prepare_part() {
local loop=$1
local folder=$2
local image=$3
local mount_image=$4
losetup $loop $folder/$image
parted -s $loop -- mklabel msdos
parted -s $loop -- mkpart primary ext4 8192s -1s
#La carte ne boot pas avec la ligne en dessous
#parted -s $loop align-check optimal 1
partprobe $loop
mkfs.ext4 $loop"p1"
mkdir $mount_image
mount $loop"p1" $mount_image
}
copy_files() {
# Stopper les services avant de copie
# Vérifier que les partitions soient montées
# Afficher le point de montage de la partition
# On redirige la sortie du programme vers /dev/null et la sortie erreur (2) sur la sortie standard
# On ne verra que les erreurs des fichiers ou la copie a posé problème
local mount_image=$1
rsync -avrltD --delete --exclude={/dev/*,/proc/*,/sys/*,/media/*,/mnt/*,/run/*,/tmp/*} / $mount_image > /dev/null 2>&1
}
#Modify fstab
change_fstab() {
local mount_image=$1
local old_uuid=$(blkid -o export `mount | grep -w / | awk '{print $1}'` | grep ^UUID)
local new_uuid=$(blkid -o export `mount | grep -w /media/$IMAGE | awk '{print $1}'` | grep ^UUID)
echo "old_uuid: $old_uuid"
echo "new_uuid: $new_uuid"
sed -i s/$old_uuid/$new_uuid/ $MOUNT_IMAGE/etc/fstab
if [ -e $MOUNT_IMAGE/boot/armbianEnv.txt ]; then
sed -i s/$old_uuid/$new_uuid/ $MOUNT_IMAGE/boot/armbianEnv.txt
fi
}
write_mbr() {
# On cherche le uboot_sunxi_with_spl.bin selon si on est sur armbian ou non
local $uboot_mbr
local $loop
loop=$1
uboot_mbr=$2
2021-01-13 01:55:57 +01:00
2020-04-19 09:06:04 +02:00
if [ ! -f $uboot_mbr ]; then
exit 1;
fi
# on Efface la zone pour le mbr
dd if=/dev/zero of=$loop bs=1k count=1023 seek=1 status=noxfer > /dev/null 2>&1;
# on flash le mbr
dd if=$uboot_mbr of=$loop bs=1024 seek=8 status=noxfer > /dev/null 2>&1;
}
clean() {
local loop=$1
local mount_image=$2
sync
umount $mount_image
losetup -d $loop
rmdir $mount_image
}
## Début du script
show_usage() {
cat << EOF
Ce script génère une image qui peut être flasher sur une carte SD pour les cartes
Olimex LIME LIME2 et orange pi pc (plus)
$(basename $0)
-d Dossier où sera stocké l'image
-u uboot_sunxi_with_spl.bin
EOF
exit 1
}
if [ ! $(id -u) = 0 ]; then
echo "L'utilisateur doit être root"
exit 1
fi
## Possibilité d'aller chercher le uboot_sunxi_with_spl.bin sur une adresse http
#Check arguments
if [ $# -eq 0 ]; then
show_usage
fi
# Gérer les options https://www.tutorialspoint.com/unix_commands/getopt.htm
while getopts ":d:u:h" opt; do
case $opt in
d)
FOLDER=$OPTARG
;;
u)
UBOOT_MBR=$OPTARG
;;
h | *)
show_usage;
;;
esac
done
if [ -z $FOLDER ]; then
show_usage
fi
if [ ! -d $FOLDER ]; then
echo "$FOLDER isn't a directory";
#echo "Le dossier $FOLDER n'existe pas"
show_usage
fi
if [ -z $UBOOT_MBR ]; then
show_usage
fi
if [ ! -f $UBOOT_MBR ]; then
echo "This file $UBOOT_MBR doesn't exist";
show_usage
fi
IMAGE=snapshot_`date +%Y%m%d`.img
LOOP=$(losetup -f)
MOUNT_IMAGE="/media/"$IMAGE
check_dep
echo "check_dep : OK"
#On crée un image de la taille minimale avec 1024M de place disponible.
create_img $FOLDER $IMAGE
echo "create_img: OK"
#On crée l'interface Loop et les partitions qui vont bien
prepare_part $LOOP $FOLDER $IMAGE $MOUNT_IMAGE
echo "prepare_part: OK"
#On copie la /
copy_files $MOUNT_IMAGE
echo "copy_file: OK"
#
change_fstab $MOUNT_IMAGE
echo "change_fstab: OK"
#On écrit le MBR
write_mbr $LOOP $UBOOT_MBR
echo "write_uboot: OK"
#demonte l'image et detache l'interface loop
clean $LOOP $MOUNT_IMAGE
```
je copie ce script dans /usr/local/bin/make_img
2021-01-13 01:55:57 +01:00
``` bash
chmod +x /usr/local/bin/make_img
```
2020-04-19 09:06:04 +02:00
Puis dans mon fichier /etc/saveonusb/script/save.sh
```bash
DIR_SAVE=$1
2021-01-13 01:55:57 +01:00
if [ ! -d "$DIR_SAVE" ]; then
echo "Le repertoire $DIR_SAVE n'existe pas" 1>&2;
exit 1;
2020-04-19 09:06:04 +02:00
fi
/usr/local/bin/make_img -f $DIR_SAVE -u /usr/lib/linux-u-boot-next-orangepipcplus_5.90_armhf/u-boot-sunxi-with-spl.bin
if [ $? -eq 1 ]; then
echo "Il y a eu une erreur durant la copie des fichiers" 1>&2;
exit 1;
fi
2021-01-13 01:55:57 +01:00
```
2020-04-19 09:06:04 +02:00
Version pour les personnes qui sont sous yunohost:
2021-01-13 01:55:57 +01:00
``` bash
2020-04-19 09:06:04 +02:00
#!/bin/bash
DIR_SAVE=$1
if [ ! -d "$DIR_SAVE" ]; then
echo "Le repertoire $DIR_SAVE n'existe pas" 1>&2;
exit 1;
fi
/usr/bin/yunohost backup create -o $DIR_SAVE
if [ $? -eq 1 ]; then
echo "Il y a eu une erreur pendant l'exécution de yunohost backup" 1>&2;
exit 1;
fi
```
2021-01-13 01:55:57 +01:00
Il y a biensur des choses à améliorer, mais c'est fonctionnel. On pourrait par exemple avoir plusieurs scripts ou vérifier si le support de sauvergarde a encore la place suffisante. Mais pour mes besoins ça me suffit. Une fonction pour la restauration ça serait pas mal aussi :).