It would be a good idea to schedule a daily backup of all the MySQL databases on your Synology. Those will indeed be wiped out if by accident, you have to reset your NAS (reinstall the firmware), ex.: if you lose a disk in your RAID 0.
That can be done using a shell script and a cron job as described here after.
How to Backup
- Go to the web administration interface (DSM) of your NAS.
- Install phpMYAdmin via the “Package Center” if not yet done.
- Open phpMyAdmin via the “Start Menu” (“Main Menu”).
- I am using the login ‘root’ with the password of Syno’s ‘admin’ user
- Open the “Users” tab in phpMyAdmin
- Click “Add User” (bellow the list of existing users)
- Enter the User Name. Ex.: backup
- Select “localhost” as an Host.
- Enter a password. Ex.: mysql
- Keep “none” as “Database for user”
- Grant the user with Global privileges: Select (in Data) and Reload, Show Databases and Lock Tables (in Administration)
- Click “Add User” at the botton of the page
Next, create a shell scripts in a Shared Folder of the NAS (Ex.: \volume1\backup\backupMySql.sh). Working in a Shared Folder will make it easier for you to copy/open later the backups from your PC). Don’t forget to create a “Unix” file, either using the touch command in a Console or saving a file As a “Unix shell script” within Notepad++. Notice that a script created with NotePad++ and saved on the Shared Folder will belong to the user account accessing that Shared Folder (Most probably your Windows Account if like me you simply created a user on your NAS with the same login and password). A script created with “touch” in a Console will belong to the user accessing the NAS via telnet/SSH (Most probably the “root” account).
[shell]#!/bin/sh # DIR=/volume1/backup/sqlbackup/ DATESTAMP=$(date +%Y%m%d%H%M%S) DB_USER=backup DB_PASS=mysql # create backup dir if it does not exist mkdir -p ${DIR} # remove backups older than $DAYS_KEEP #DAYS_KEEP=30 #find ${DIR}* -mtime +$DAYS_KEEP -exec rm -f {} \; 2> /dev/null # remove all backups except the $KEEP latest KEEP=5 BACKUPS=`find ${DIR} -name 'mysqldump-*.gz' | wc -l | sed 's/\ //g'` while [ $BACKUPS -ge $KEEP ] do ls -tr1 ${DIR}mysqldump-*.gz | head -n 1 | xargs rm -f BACKUPS=`expr $BACKUPS - 1` done # # create backups securely #umask 006 # dump all the databases in a gzip file FILENAME=${DIR}mysqldump-${DATESTAMP}.gz /usr/syno/mysql/bin/mysqldump --user=$DB_USER --password=$DB_PASS --opt --all-databases --flush-logs | gzip >> $FILENAME [/shell]
NB: Since DSM 6.0, “/usr/syno/mysql/bin/mysqldump” has been moved to “/bin/mysqldump” !!!
Possibly, type the following command in a Console (telnet/SSH) to set the user ‘root’ as owner of the script:
[shell]chown root /volume1/backup/backupMySql.sh[/shell]
IMPORTANT notice: I used to have “-u DB_USER” (with a blank space in between) in my script above and -“p$DB_USER” (without a blank) instead of –user and –password. But a reader (Fredy) reported that the script was not running fine for him unless removing the blank. As per documentation of mysqldump’s user parameter and password parameter, there should be a blank after -u but not after -p. However, samples on the web usually illustrate the use of mysqldump with “-uroot”. So, I decided to use the more explicit notation “–user=” and “–password=”. I did test this notation with a wrong username or password and the resulting dump is indeed empty. With the correct username and password, it works fine.
Since DSM 4.2, the Task Scheduler can be used to run the script on a daily basis.
- Go to the Start Menu
- Open the Control Panel
- In the “Application Settings”, open the Task Scheduler
- Select “Create a User-Defined Script”
- Type a name for that “Task:”
- Keep the “User:” root
- In the “Run Command” pane, type:
sh /volume1/backup/backupMySql.sh
Don’t forget the “sh” in front of your command, otherwise, it does not work (although the sample provided by Synology is missing it ?!)
If you don’t use the Task Scheduler, you can add a Cron Job to execute the script, e.g. every day at 0:01. Open a Console (Telnet/SSH) and type:
[shell]echo "1 0 * * * root sh /volume1/backup/backupMySql.sh" >> /etc/crontab[/shell]
FYI, a cron line syntax is “mm hh dd MMM DDD user task” where:
- mm is the minute (0..59)
- hh is the hour (0..23)
- dd is the day in the month (1..31)
- MMM is the month (jan, feb, … or 1..12)
- DDD is the day in the week (sun, mon, … or 0..7 where 0 and 7 beeing both sunday)
The following values can also be used:
- * : a every unit (0, 1, 2, 3, 4…)
- 5,8 : at unit 5 et 8
- 2-5 : units 2 to 5 (2, 3, 4, 5)
- */3 : every 3 units (0, 3, 6, 9…)
- 10-20/3 : every 3 units, from 10th to 20th (10, 13, 16, 19)
So, the script will start every day at 0:01h
Finally, you must restart the cron daemon to activate the new job. To find the process id and send a SIGHUP signal to this one, type the following command line in a Console (Telnet/SSH):
[shell]ps | grep crond | grep -v grep | awk '{print$1}' | xargs -t kill -HUP[/shell]
It should display “Kill -HUP xxxx” where xxx is the pid of the cron daemon.
Added on 03/01/2013
Attention: if you copy paste the command to restart the cron daemon from this page into the telnet console, some symbols will be wiped out: the ‘{ and ‘ around print$1… you have to type them manually…
Attention: if you upgrade the version of DSM, the changes made in the cron script are lost (at least, it’s what I have noticed after updating last month…). Reason why I recommend to use the new “Task Scheduler” feature available since DSM 4.2.
Tip to edit the cron script: you can use the package “Config File Editor” available here. Once installed, you can access it via the Main Menu. Then, you have to edit the file named “Config File Editor” and add the following line:
/etc/crontab,crontab
Once this line is added and saved, … I don’t remember how to force the change to be taken into account :/. But restarting the DiskStation is enough 😀
Finally, back into the Config File Editor, select the ‘crontab’ entry and modify this script, save your changes and restart the cron daemon.
Tip to restart the cron daemon: you can use the package “MODS Web Console” available here. To install it, go to the Package Center and add the following url via the Settings > Package Sources : https://www.beatificabytes.be/sspks. Once this url added, go to the tab “Community” and click Install on the “MODS Web Console” icon.
Run the Web Console via the Main Menu, log in with “admin”/”admin” (The defaults values if not changed) and type:
synoservice --restart crond
Added on 01/12/2013
If you want a lazy solution to notify the Administrator about the success or failure of the backup, you can use the ‘synonotify’ command (See details here). A more advanced solution would be to configure the “Mail Server” and use its ‘sendmail’ command: /volume1/@appstore/MailServer/sbin/sendmail…
Added on 08/01/2017
mysqldump has moved
- DSM 6.0 : /usr/syno/mysql/bin/mysqldump
- MariaDB 5: /volume1/@appstore/MariaDB/usr/bin/mysqldump
MariaDB 10: /volume1/@appstore/MariaDB10/usr/local/mariadb10/bin/mysqldump
How to Restore
To do a full restore, simply:
- Copy the archive (.gz) you plan to restore on your local PC
- Go to “phpyMyAdmin” > “Import” > “Chose File” (next to Browse your computer).
- Select the .gz file to be restored
- Press “Go” at the bottom of the page (no need to change any settings)
Pay attention: this is a complete restore, I.e.: all databases are dropped and recreated from scratch.
If you want to be able to restore a subset of your databases you better have to change the script to backup specific databases instead of –all-database.
If you want to restore only one table ‘mytable’ from the backup ‘mysqldump-datetime.gz’:
- Unzip first the archive: gunzip mysqldump-datetime.gz
- Then extract the part that restore the desired table with this command: sed -n -e ‘/DROP TABLE.*mytable/,/UNLOCK TABLES;/p’ mysqldump-datetime > mytabledump
Download the script here.
Leave a Reply