MariaDB
Siehe auch
https://www.mariadbtutorial.com/ und https://www.mysqltutorial.org
- Ansible-Rolle MariaDB-Server:
- Ansible-Rolle MariaDB-Client:
Bemerkung
Ein Server mit MariaDB 10.6+ (egal ob dediziert oder zusammen mit einer Applikation) sollte heute mit mindestens 4 GB RAM betrieben werden.
MariaDB ist ein multi-user, multi-threaded, binärkompatibler Fork der relationalen Datenbank MySQL. Da Oracle inzwischen die Markenrechte an MySQL hält, musste ein neuer Name gefunden werden - „MariaDB“ und die zugrundeliegende Storage-Engine namens „Aria“ gehen auf den Namen der jüngeren Tochter des ehemaligen MySQL-Hauptentwicklers zurück, der auch den Fork initiiert hat.
In Fedora ist MySQL Ende 2012 durch MariaDB ersetzt worden, in Folge dadurch auch in und seit RHEL 7. Das Wikipedia-Projekt hat sich unter der Haube im April 2013 von MySQL verabschiedet.
Bis MariaDB 5.5 heisst der Service noch mysql
, danach mariadb
. Wird ein MariaDB 5 auf MariaDB 10 aktualisiert, bleibt es aber bis einschliesslich 10.3 beim Service-Namen mysql
.
Seit MariaDB 10.5 werden die Begriffe master und slave durch primary und replica ersetzt.
- Abkürzungen
AI = Auto-Increment
BIN = Binary
G = Generated Always As ()
NN = NOT NULL
PK = Primary Key
UN = Unsigned
UQ = Unique
ZF = Zerofill
Es gibt jede Menge Detail-Unterschiede zwischen MySQL 5 und MariaDB 10. Eine winzige Auswahl aus dem Betrieb:
Die Unterschiede in den Default-Einstellungen (
my.cnf
,SHOW VARIABLES
) sind gigantisch.MariaDB 10 ist restriktiver.
String-Längenbeschränkungen: in v5 wurden längere Strings beim INSERT in die Tabelle einfach abgeschnitten, in v10 ergibt das einen Fehler.
Integer: in v5 war es möglich, leere Strings in ein Integer-Feld zu schreiben; in v10 führt das zu einem Fehler.
Installation & Konfiguration
Wir empfehlen die Installation direkt aus dem Hersteller- und nicht aus den RHEL-Repos. Unbedingt auf die Gross-/Kleinschreibung bei yum install MariaDB-server
achten, damit der Server aus dem Hersteller-Repo geholt wird.
Repository Configuration Tool: https://downloads.mariadb.org/mariadb/repositories
VER=10.11
wget https://downloads.mariadb.com/MariaDB/mariadb_repo_setup
chmod +x mariadb_repo_setup
./mariadb_repo_setup --mariadb-server-version="mariadb-$VER"
yum -y install MariaDB-server
systemctl enable --now mariadb
# optional
yum -y install MariaDB-client MariaDB-backup
yum -y install maxscale
Egal welche $VER
für das Repo herangezogen wird: jede baseurl
liefert neben $VER
immer auch die aktuellste Version, die damit standardmässig installiert wird.
Tipp
Wer eine spezifische Version wie z.B. v 5.5 einsetzen muss, lässt sich mit yum list MariaDB-server --show-duplicates
zunächst anzeigen, welche Minor-Versionen verfügbar sind. Die Installation erfolgt dann per yum -y install MariaDB-server-5.5.66
.
WICHTIG: da hier eine spezifische, tiefere Version installiert wird, würde diese beim nächsten yum update
aktualisiert werden. Um das zu verhindern, muss die installierte Version per yum -y install yum-plugin-versionlock; yum versionlock mariadb*
gelockt werden.
Anschliessend kann man dem MariaDB-Standardbenutzer „root“ ein komplexes Passwort zuteilen sowie den Gast-Zugang und die zusätzlich installierte Datenbank „test“ löschen:
mariadb-secure-installation
Es erhöht die Sicherheit, die root-Benutzer nach der Installation umzubenennen - es sind bis zu vier, je nach dem, ob die Maschine über einen voll-qualifizierten Hostnamen verfügt oder nicht:
einen für den Zugang von ::1 (root@‘::1‘)
einen für den Zugang von 127.0.0.1 (root@‘127.0.0.1‘)
einen für den Zugang von hostname (root@‘hostname‘)
einen für den Zugang von hostname.fqdn (root@‘hostname.fqdn‘)
mysql --user=root --password
Zugriff dann per
mysql --user=root
root-Benutzer umbenennen - bis 10.3:
update mysql.user set user = 'mariadb-admin' where user = 'root';
flush privileges;
quit
Ab 10.4 ist mysql.user
neu eine View, und das Verfahren hat sich geändert. Hier gilt:
rename user 'root'@'localhost' to 'mariadb-admin'@'localhost';
Wer möchte, dass man sich von jedem Rechner aus (Adresse: „%“, dem Wildcard in SQL) mit MariaDB und root-Rechten verbinden darf, fügt einen Benutzer (hier per GRANT-Statement) hinzu:
grant all privileges
on *.*
to 'mariadb-admin'@'%'
identified by 'password'
with grant option;
flush privileges;
quit
Wer selektiver vorgeht und beispielsweise nur Zugriffe aus dem 10.26.6er-Netz zulassen möchte, verwendet create user 'mariadb-admin'@'10.26.6.%' ...
.
Bemerkung
Wer auf einer Maschine den Kommandozeilen-Client mysql
benötigt,
installiert ihn mit yum -y install mariadb
(damit wird nicht der
MariaDB-Server installiert). Der Client landet immer auf der Maschine, sobald man den
MariaDB-Server installiert.
Konfigurationsdateien unterhalb von /etc/my.cnf.d
werden erwartungsgemäss alphabetisch ausgewertet. Einstellungen in der /etc/my.cnf.d/mysettings.cnf
werden also durch eine /etc/my.cnf.d/z_mysettings.cnf
bei Bedarf überschrieben.
Passwort ändern - man beachte, dass „flush“ zuerst ausgeführt werden muss:
flush privileges;
alter user 'root'@'localhost' identified by 'myn3wp4ssw0rd';
Migration MariaDB > MariaDB
Umzug einer MariaDB-Installation von VMalt nach VMneu, ohne langwierigen SQL-Ex- und Import der Datenbanken, sondern durch Kopie der Datenverzeichnisse. Die Ziel-Maschine kann dabei auch eine neuere MariaDB-Version fahren (ein bereits installiertes MariaDB ist Voraussetzung); ein Downgrade dagegen funktioniert nicht.
Wo |
Was |
---|---|
VMalt |
|
VMalt |
|
VMneu |
|
VMneu |
|
VMneu |
|
VMneu |
|
VMneu |
|
VMneu |
|
VMneu |
Meldungen wie |
VMneu |
|
Tipp
Wenn die importierte Datenbank keinen mariadb-admin
-User hat, und das root
-Passwort unbekannt ist, lässt sich der MariaDB-Server ohne User-Schicht wie folgt starten: sudo -u mysql mariadbd --skip-grant-tables&
.
Migration MySQL > MariaDB
MariaDB 10.11 lässt sich ein originales /var/lib/mysql
von MySQL 8 nicht unterschieben, MariaDB rennt dann in den Fehler InnoDB: Plugin initialization aborted with error Generic error
. Hier kommt man also um einen logischen Dump z.B. per mydumper auf MySQL und anschliessendem Import in MariaDB nicht herum.
Wichtig: MySQL 8+ verwendet als Default-Collation utf8mb4_unicode_520_ci
, die MariaDB unbekannt ist (Fehler: Connection 21 - ERROR 1273: Unknown collation: 'utf8mb4_0900_ai_ci'
). Für MariaDB 10.10+ kann diese Collation vor dem Import durch uca1400_as_ci
ersetzt werden:
sed -i "s/utf8mb4_0900_ai_ci/uca1400_as_ci/g" /backup/mysql-dump/*.sql
Upgrading
Ein Upgrade von beispielsweise MariaDB 10.2 direkt auf MariaDB 10.6 ist möglich. Das Upgrade-Prozedere wird hier anhand der aus den MariaDB-Repos installierten Datenbank beschrieben.
Backups aller Datenbanken anlegen.
Evtl. Konfiguration in
/etc/my.cnf.d
an die zukünftige Version anpassen.Passwort des DBA bereithalten.
Auf die DB-bezogene Cronjobs deaktivieren.
MariaDB stoppen:
systemctl stop mariadb
MariaDB deinstallieren:
dnf remove MariaDB-server
. Das Verzeichnis/var/lib/mysql
bleibt dabei erhalten.Die bisher verwendeten
/etc/yum.repos.d/mariadb.repo
-Dateien deaktivieren.Repository-Datei unter
/etc/yum.repos.d
auf die neue Version anpassen.dnf clean all
Aktuellere Version installieren:
dnf install MariaDB-server
/etc/my.cnf
bzw./etc/my.cnf.d/*
anpassen. Beispielsweise sollten alle Einstellungen entfernt werden, die nicht mehr unterstützt werden. Siehe dazu auch die Tabelle am Ende dieses Dokuments.MariaDB starten:
systemctl enable mariadb; systemctl start mariadb
Upgrade-Skript ausführen (sonst erhält man Fehler der Art
Couldn't execute '...': Column count of mysql... is wrong. Expected 21, found 20. The table is probably corrupted (1805)
):mysql_upgrade --user=mariadb-admin --password
- es passt System-Tabellen so an, dass sie mit der neuen Version kompatibel sind, prüft darüber hinaus alle Tabellen auf Konsistenz und markiert sie als mit der neuen Version kompatibel.MariaDB restarten:
systemctl restart mariadb
Clients
Tipp
Verbinden über einen SSH-Tunnel: ssh -L 13306:localhost:3306 mariadb-host
, Zugriff dann auf localhost:13306.
- MySQL Workbench
Für die Administration von MySQL-Datenbanken eignet sich MySQL Workbench auf einer Admin-Workstation. Für MariaDB eignet sich das Tool Stand 2021-04 noch, jedoch nicht mehr in allen Bereichen, beispielsweise in der Benutzer-Administration. Die RPM-Pakete für die Yum-Repositories des Herstellers finden sich auf http://dev.mysql.com/downloads/repo/yum/.
Am Beispiel von Fedora Workstation:
VER=8.0.36-1.fc38 dnf -y install https://dev.mysql.com/get/Downloads/MySQLGUITools/mysql-workbench-community-$VER.x86_64.rpm
Konfiguration liegt unter ~/.mysql/workbench.
Welche Berechtigungen stecken hinter den „Administrative Roles“?
MaintenanceAdmin: Event, Reload, Show_db, Shutdown, Super
ProcessAdmin: Reload, Super
UserAdmin: Create_user, Reload
SecurityAdmin: Grant, Create_user, Reload, Show_db, User_manage_attr
MonitorAdmin: Process
DBManager: Create, Drop, Grant, Event, Alter, Delete, Index, Insert, Select, Update, Create_tmp_table, Lock_tables, Trigger, Create_view, Show_view, Create_routine, Alter_routine, Show_db
DBDesigner: Create, Alter, Index, Trigger, Create_view, Show_view, Create_routine, Alter_routine, Show_db, Routine_manage_attr
ReplicationAdmin: Repl_client, Repl_slave, Super
BackupAdmin: Event, Select, Lock_tables, Show_db
Backup und Restore
- Physische Backups
Im Prinzip Sicherung des Datenverzeichnisses
/var/lib/mysql
. In der Regel schnell, aber nur auf der gleichen MariaDB-Version verwendbar.- Logische Backups
Es werden universelle SQL-Dateien (ASCII) geschrieben, die auch auf anderer Hardware, anderen MariaDB-Versionen oder (unter Umständen) auf anderen DB-Systemen zum Import verwendet werden können. Der Nachteil: das dauert, auch auf moderner Hardware.
mysql-Client
Der Dump - realistisch ungefähr 1 GB pro Minute. Wichtig zu wissen: die innodb_buffer_pool
-Caches werden durch die beim Backup stattfindenden Full Table Scans zugemüllt, was folgende Statements verhindern:
# save the innodb_buffer_pool
$MYSQL --host=$HOST --user=$USER --password=$PASSWORD --execute='SET GLOBAL innodb_buffer_pool_dump_now = ON;'
# do the backup
# for example with mydumper or mysqldump
# restore the innodb_buffer_pool
$MYSQL --host=$HOST --user=$USER --password=$PASSWORD --execute='SET GLOBAL innodb_buffer_pool_load_now = ON;'
Tipp
Angaben im SQL-Dump der Form /*!40014 SET ... */;
sind Conditional Comments. Im Beispiel wird SET ...
nur auf einem MySQL 4.0.14 oder später ausgeführt.
Für globale Locks ist mindestens das RELOAD-Privileg nötig.
Der Restore mit dem Kommandozeilen-Client - realistisch ungefähr 200 MB pro Minute:
mysql --user=root --password mydb < mydb.sql
Fortschrittsanzeige mit Hilfe des „pipe viewers“:
dnf -y install pv
pv mydb.sql | mysql --user=root --password mydb
Import dauert ewig? Ein schnellerer Restore, der einige Checks abschaltet und damit um Faktoren beschleunigt abläuft, geschieht mittels:
SET autocommit=0;
SET unique_checks=0;
SET foreign_key_checks=0;
here comes the backup_file.sql
COMMIT;
Um SQL-Dumps im GB-Bereich zu bearbeiten, empfiehlt sich hexedit
: mit Tab auf die rechte ASCII-Seite wechseln, mit Space den Anfang der Datei bis zu den ersten SQL-Anweisungen überschreiben, und anschliessend obige Statements einfügen. F2 speichert den Dump.
mydumper und myloader
Siehe https://github.com/mydumper/mydumper.
Geht durch Threading etwas schneller zu Werke als mysqldump. Von uns empfohlen.
mydumper kann die User-Credentials aus der ~/.my.cnf
, einer cnf
-Datei aus /etc/my.cnf.d
, Sektion [client]
, oder einer beliebigen Datei, die per --defaults-files=/path/to/file
referenziert wird, verwenden:
[client]
user=mydumper_user
password=mydumper_s3cr3t_pass
- mydumper
Installation
VER=$(curl -Ls -o /dev/null -w %{url_effective} https://github.com/mydumper/mydumper/releases/latest | cut -d'/' -f8) # RHEL 9: dnf -y install https://github.com/mydumper/mydumper/releases/download/v$VER/mydumper-$VER.el9.x86_64.rpm # RHEL 8: dnf -y install compat-openssl10 dnf -y install https://github.com/mydumper/mydumper/releases/download/v$VER/mydumper-$VER.el8.x86_64.rpm
Full-Backup / Full-Dump:
HOST=localhost USERNAME=root PASSWORD=linuxfabrik BACKUP_DIR=/backup THREADS=0 # most used variant, all databases: mydumper --host=$HOST --user=$USER --password=$PASSWORD \ --compress \ --outputdir=$BACKUP_DIR # with more options: mydumper --host=$HOST --user=$USER --password=$PASSWORD \ --build-empty-files \ --events \ --logfile=$BACKUP_DIR/mydumper.log \ --outputdir=$BACKUP_DIR \ --routines \ --threads=$THREADS \ --triggers \ --verbose=3
Wird
--database
weggelassen, werden alle Datenbanken gedumpt.--logfile $BACKUP_DIR/mydumper.log
wird bei jedem Run überschrieben.--outputdir $BACKUP_DIR
verhindert das Erzeugen von Verzeichnissen der Artexport-20201027-145722
.--database
wählt bei Bedarf die zu exportierende Datenbank aus.--threads=0
wählt die Anzahl der Threads automatisch (erzeugt pro Core einen Thread, mindestens aber zwei). „0“ wird erst in neueren mydumper-Versionen unterstützt, und erzeugt sonst eine „floating point exception“. Default ist „4“.
Tipp
Ein Dump-Benutzer benötigt die Rechte
event, execute, lock_tables, process, reload, select, show databases, show view, super, trigger
. Werden beispielsweise Views oder Trigger nicht exportiert, sollten die Berechtigungen des Dump-Users überprüft werden.execute
braucht es für gewisse Tabellen in dersys
-DB.- myloader
Restore: Full-Backup wiederherstellen (bereits existierende und nicht im Backup enthaltene Datenbanken bleiben dabei natürlich unangetastet):
HOST=localhost USERNAME=root BACKUP_DIR=/backup THREADS=0 myloader --host=$HOST --user=$USER --ask-password \ --directory=$BACKUP_DIR \ --overwrite-tables \ --threads=$THREADS \ --verbose=3
Einzelne Datenbank(en) wiederherstellen (und dabei neu benennen):
HOST=localhost USERNAME=root BACKUP_DIR=/backup NEW_NAME_OF_DB=restored-nextcloud DB_TO_RESTORE=nextcloud THREADS=0 myloader --host=$HOST --user=$USER --ask-password \ --directory=$BACKUP_DIR \ --database=$NEW_NAME_OF_DB \ --overwrite-tables \ --queries-per-transaction=1000 \ --source-db=$DB_TO_RESTORE \ --threads=$THREADS \ --verbose=3
--enable-binlog
steht standardmässig auf False.--overwrite-tables
= DROP TABLE IF EXISTS, im Standard abgeschaltet. Backup-User benötigt sonst DROP-Rechte.
Der Restore muss mit einem Benutzer durchgeführt werden, der ausreichend Rechte zum Anlegen von Tabellen etc. besitzt. Falls in eine neue Datenbank wiederhergestellt werden soll, muss diese bereits angelegt sein. Falls der Restore Meldungen der Form
** (myloader:4191): CRITICAL **: 11:42:06.213: Error restoring db.table from file db.table-schema-view.sql.gz: Table 'db.other_table' doesn't exist
bringt, einfach den Import erneut laufen lassen (da dann die vermissten Tabellen mit ziemlicher Sicherheit vorhanden sind).Der Restore einzelner Tabellen gestaltet sich simpel, da diese als SQL-Datei vorliegen.
mysqldump, mariadb-dump
mysqldump
erzeugt logische Dumps.
Full-Backup / Full-Dump:
mysqldump --host=$HOST --port=3306 --user=$USER --password=$PASSWORD \
--create-options \
--default-character-set=utf8 \
--events \
--routines \
--single-transaction=TRUE \
--skip-dump-date \
--skip-extended-insert \
--triggers \
--all-databases > /path/to/backup.sql
Um einen Dump von Server A auf Server B zu bekommen, dieses auf Server B ausführen:
mysqldump --host=server_a --user=root --password --single-transaction --flush-logs --hex-blob --source-data=2 --routines --all-databases | mysql
# --source-data causes the binary log position and filename to be appended to the output.
# If equal to 1, will print it as a CHANGE MASTER command.
# If equal to 2, that command will be prefixed with a comment symbol.
Ein DB-für-DB Backup-Script auf Basis von mysqldump:
echo Starting the logical backup...
if command -v zstd &>/dev/null
then
PACKER=zstd
EXTENSION=zst
else
PACKER='gzip --fast'
EXTENSION=gz
fi
# We do NOT lock the database by "BACKUP STAGE START; BACKUP STAGE BLOCK_COMMIT; DO SLEEP(very_long);"
# or similar, because it is too dangerous to run into a deadlock situation
# (client writes to a locked DB, and both client and the backup are blocked until the end of the sleep,
# and the backup is still not consistent after the end of the sleep).
# Instead, we use --single-transaction again. So, use the logical backup with care and don't expect
# it to be consistent (foreign keys etc.).
\rm -rf $BACKUP_DIR_LOGICAL
mkdir -p $BACKUP_DIR_LOGICAL
# go through all databases
echo Dumping...
for DB in $DBS
do
if [ "$DB" == 'performance_schema' ]
then
continue
fi
CHARSET=$($MYSQL --host=$HOST --user=$USER --password=$PASSWORD \
$DB --execute='SELECT @@character_set_database;' | \
$AWK '{ print $1}' | $GREP -v '^@@character_set_database')
# go through all tables in database
TABLES=$($MYSQL --host=$HOST --user=$USER --password=$PASSWORD \
--execute='show tables;' $DB | \
$AWK '{ print $1}' | $GREP -v '^Tables' )
for TABLE in $TABLES
do
if [ "$DB" == 'mysql' ]
then
if [ $TABLE == 'general_log' ]
then
continue
fi
if [ $TABLE == 'slow_log' ]
then
continue
fi
if [ $TABLE == 'transaction_registry' ]
then
continue
fi
fi
$MYSQLDUMP --host=$HOST --user=$USER --password=$PASSWORD \
--create-options \
--default-character-set=$CHARSET \
--routines \
--single-transaction \
--skip-dump-date \
--skip-extended-insert \
--triggers \
$DB $TABLE | $PACKER > $BACKUP_DIR_LOGICAL/$DB.$TABLE.sql.$EXTENSION
if [ $? -ne 0 ]
then
echo "Error dumping $DB.$TABLE" >&2
else
echo "$DB.$TABLE dumped"
fi
done
done
echo Done.
Mariabackup
Ab Version 10.1.23 kann das ursprünglich von Percona XtraBackup 2.3.8 geforkte Mariabackup verwendet werden, um InnoDB und XtraDB zu sichern. Unter InnoDB sind „hot online“-Backups möglich.
Wem es auf Performance ankommt: Unserer Erfahrung nach ist das Tool für ein erstmaliges Full Backup (und damit auch für die Einrichtung einer Replikation) nicht schneller als mysqldump
oder mariadbdump
, im Gegenteil - es ist nochmal langsamer, da es sich unter anderem ausgiebig mit den InnoDB redo logs beschäftigt, siehe Output weiter unten.
- Full Backup erzeugen:
Siehe https://mariadb.com/kb/en/full-backup-and-restore-with-mariabackup/
rm -rf /backup/mariadb-dump/* mariabackup --host=$HOST --user=$USER --password=$PASSWORD \ --backup \ --target-dir=/backup/mariadb-dump/full
Der Output:
[00] 2020-10-19 15:20:37 Connecting to MySQL server host: localhost, user: mariadb-admin, password: set, port: not set, socket: not set [00] 2020-10-19 15:20:37 Using server version 10.5.6-MariaDB-log mariabackup based on MariaDB server 10.5.6-MariaDB Linux (x86_64) [00] 2020-10-19 15:20:37 uses posix_fadvise(). [00] 2020-10-19 15:20:37 cd to /var/lib/mysql/ [00] 2020-10-19 15:20:37 open files limit requested 0, set to 1024 [00] 2020-10-19 15:20:37 mariabackup: using the following InnoDB configuration: [00] 2020-10-19 15:20:37 innodb_data_home_dir = [00] 2020-10-19 15:20:37 innodb_data_file_path = ibdata1:12M:autoextend [00] 2020-10-19 15:20:37 innodb_log_group_home_dir = ./ [00] 2020-10-19 15:20:37 InnoDB: Using Linux native AIO 2020-10-19 15:20:37 0 [Note] InnoDB: Number of pools: 1 [00] 2020-10-19 15:20:37 mariabackup: Generating a list of tablespaces 2020-10-19 15:20:37 0 [Warning] InnoDB: Allocated tablespace ID 2727 for mysql/innodb_table_stats, old maximum was 0 [00] 2020-10-19 15:20:40 >> log scanned up to (31993329429) [01] 2020-10-19 15:20:40 Copying ibdata1 to /backup/mariadb-dump/ibdata1 [00] 2020-10-19 15:20:41 >> log scanned up to (31993329705) [00] 2020-10-19 15:20:42 >> log scanned up to (31993329722) ... [00] 2020-10-19 15:20:50 >> log scanned up to (31993330151) [00] 2020-10-19 15:20:51 >> log scanned up to (31993330151) 2020-10-19 15:20:52 0 [Note] InnoDB: Read redo log up to LSN=31993330688 [00] 2020-10-19 15:20:52 >> log scanned up to (31993330444) [00] 2020-10-19 15:20:53 >> log scanned up to (31993330444) ... [00] 2020-10-19 15:20:59 >> log scanned up to (31993330865) [00] 2020-10-19 15:21:00 >> log scanned up to (31993330929) [01] 2020-10-19 15:21:01 ...done [01] 2020-10-19 15:21:01 Copying ./mysql/innodb_table_stats.ibd to /backup/mariadb-dump/mysql/innodb_table_stats.ibd [01] 2020-10-19 15:21:01 ...done ...
Full Backup wiederherstellen:
mkdir /backup/mariadb-dump/temp cd /backup/mariadb-dump/temp tar --use-compress-program zstd -xvf ../full/server.tar.zst # make the full backup to be point-in-time consistent: mariabackup --prepare --target-dir=/backup/mariadb-dump/temp systemctl stop mariadb rm -rf /var/lib/mysql/* mariabackup --copy-back --target-dir=/backup/mariadb-dump/full chown -R mysql:mysql /var/lib/mysql/ restorecon -Fvr /var/lib/mysql/ systemctl start mariadb
Einzelne DB aus einem Full Backup wiederherstellen:
Siehe https://mariadb.com/resources/blog/how-to-restore-a-single-database-from-mariadb-backup/
mkdir /backup/mariadb-dump/temp cd /backup/mariadb-dump/temp tar --use-compress-program zstd -xvf ../full/server.tar.zst mariabackup --prepare --export --target-dir=/backup/mariadb-dump/temp --databases $DB
Optional Datenbank manuell erzeugen:
CREATE DATABASE ...
Tabelle identisch zum Original manuell erzeugen, jedoch OHNE Constraints:
CREATE TABLE ...
Tablespace löschen:
ALTER TABLE ... DISCARD TABLESPACE;
Backup zurückkopieren:
\cp /backup/mariadb-dump/temp/$DB/*.{ibd,cfg,frm} /var/lib/mysql/$DB/ chown -R mysql:mysql /var/lib/mysql/ restorecon -Fvr /var/lib/mysql/Tablespace importieren:
ALTER TABLE ... IMPORT TABLESPACE;
Constraints wiederherstellen:
ALTER TABLE .... ADD CONSTRAINT ....
Partielle Backups (einzelne Datenbank sichern):
mariabackup --host=$HOST --user=$USER --password=$PASSWORD \ --backup \ --target-dir=$BACKUP_DIR_PARTIAL/datadir \ --databases="$DB"
Eine einzelne Datenbank aus dem Full Backup wiederherstellen ist möglich, aber viel zu kompliziert. Wir setzen hier auf logische Backups.
Inkrementelle Backups:
columnstoreBackup
Aus dem MariaDB ColumnStore Tools Paket. Wird verwendet, um ColumnStore-Tabellen zu sichern.
Siehe https://mariadb.com/kb/en/backup-and-restore-for-mariadb-columnstore-110-onwards/
Replikation
Das Standard-Replikationsszenario besteht aus einem Primary- und (mindestens) einem Replica-Server. Der MariaDB Galera Cluster ist ein Multi-Primary-Cluster.
Primary/Replica
Begriffe:
bis 10.4: Master und Slave
10.5+: Primary und Replica
In diesem Standardszenario wird nur auf dem Primary geschrieben; Änderungen sind beinahe instant auf dem oder den Replica-Servern verfügbar. Die Replica können daher auch für reine Lese-Anforderungen eingesetzt werden. Die Replicas kommen (anders als früher) gut damit zurecht, wenn sie runtergefahren werden, auch wenn sich währenddessen auf dem Primary sowohl Schemata als auch Daten ändern.
Primary und Replica kommunizieren wenn nicht anders konfiguriert über Port 3306, also Firewall-Rules auf den beteiligten Hosts anpassen.
Achtung
Ein Replica übernimmt neben den Datenbanken auch die Benutzer-Accounts des Primary - also auch das Passwort für root, Backup-Benutzer und andere.
Siehe auch:
- Auf dem Primary:
Binary Logging einschalten, das optimale Binary Logging Format (optional) und eine im Netz eindeutige Server-ID auf einen Wert zwischen 1 und 4294967295 setzen:
binlog_format = MIXED log_bin = ON server_id = 1
Jetzt einen Replikations-Benutzer mit Passwort und REPLICATION SLAVE-Berechtigung anlegen:
GRANT REPLICATION SLAVE ON *.* TO 'mariadb-replica'@'replica-host' IDENTIFIED BY 'password'; FLUSH PRIVILEGES;
MariaDB-Server neu starten.
- Auf dem Replica:
Im Netz eindeutige Server-ID vergeben:
server_id = 2
Replica-Server neu starten.
Tipp
Unsere Ansible-Rollen für den MariaDB-Server leisten die Installation bis zu obigem Punkt. Anschliessend ist Handarbeit angesagt.
- Auf dem Primary:
Backup der Datenbanken beispielsweise mit
mariadb-dump
oder Mariabackup erzeugen und auf den Replica kopieren.Um falsch ermittelten Binlog-Positionen und inkonsitenten Backups vorzubeugen, den Parameter
--master-data
verwenden. Dieser …schaltet
--lock-tables
ab und--lock-all-tables
an (solange--single-transaction
nicht angegeben wird)setzt also ein Read-Lock (
FLUSH TABLES WITH READ LOCK
)ermittelt die aktuelle Position des Binary-Logs (sonst per
SHOW MASTER STATUS;
ermittelt) und schreibt diese in den Dumpgibt die DB nach dem Dump wieder frei (
UNLOCK TABLES;
)
# This sets a database read lock (due to --master-data)! mysqldump --verbose --user=$DBUSER --password --master-data --opt --comments --hex-blob --dump-date --no-autocommit --flush-privileges --all-databases | gzip -1 > all-databases.sql.gz scp all-databases.sql.gz ... rm -f all-databases.sql.gz
Achtung
Wir hatten schon den Fall, dass mysqldump
bei grossen Datenbanken (>= 100 GB) selbst mit --master-data
keinen konsistenten/verwendbaren Dump erzeugen konnte, der als Import auf einem Replika funktioniert hätte. Stattdessen hat sich mariabackup --backup --user $USER --password $PASSWORD --target-dir /backup/mariadb-dump/physical/
bewährt.
- Auf dem Replica:
Falls vorher schon mal importiert wurde, nicht vergessen, die Datenbanken auf dem Replika zu entfernen. Sonst einfach die Datenbanken importieren, zum Beispiel per
nice --adjustment 19 zcat all-databases.sql.gz | mysql --user=$DBUSER --password
CHANGE MASTER TO MASTER_HOST='primary.domain.com', MASTER_USER='mariadb-replica', MASTER_PASSWORD='password', MASTER_PORT=3306, MASTER_CONNECT_RETRY=10, MASTER_USE_GTID = slave_pos;
Replica starten:
START SLAVE;
Replikation prüfen:
# prior to 10.5.1: SHOW SLAVE STATUS; SHOW REPLICA STATUS;
Der Replica bzw. die Replikation funktioniert nur, wenn diese Werte zurückkommen:
Slave_IO_Running: Yes Slave_SQL_Running: Yes
Ein „Slave_SQL_Running=No“ heisst, dass die Replikation kaputt ist. Während
Queueing master event to the relay log
wird der Sync für die Replikation vorbereitet. Ein'Waiting for master to send event'
in der Spalte Slave_IO_State bedeutet lediglich, dass der Replica auf neue Inputs vom Primary wartet.
Auf dem Primary: tail -f /var/log/messages /var/lib/mysql/*.log
Auf dem Replica: tail -f /var/log/messages /var/lib/mysql/*.log
Falls auf dem Replica etwas falsch konfiguriert wurde, den Replica mit STOP SLAVE;
stoppen (evtl. hilft noch ein RESET SLAVE;
), mit CHANGE MASTER TO
umkonfiguren und per START SLAVE;
wieder starten.
Umgang mit MariaDB
/var/lib/mysql initialisieren
systemctl stop mariadb
rm -rf /var/lib/mysql
mkdir -p /var/lib/mysql/mysql
chown -R mysql:mysql /var/lib/mysql
restorecon -r /var/lib/mysql
mysql_install_db --user=mysql \
--basedir=/usr \
--datadir=/var/lib/mysql
systemctl start mariadb
mysql_secure_installation
Binary-Logs aufräumen
Im Replika-Set werden auf dem Primary laufend die Binary-Logdateien in der Form /var/lib/mysql/ON.000001
usw. geschrieben. So räumt man diese auf, ohne die Replikation zu zerstören:
Auf dem Primary:
SHOW BINARY LOGS;
Auf allen Replikas:
SHOW REPLICA STATUS;
(Inhalt der Spalte „Master_Log_File“ merken)Das älteste von einem Replika verwendete Log-File muss erhalten bleiben, alle älteren davor können gelöscht werden.
alle Log-Dateien vor ‚ON.000063‘ löschen:
PURGE BINARY LOGS TO 'ON.000063';
Benutzer
Der initiale DBA im System heisst root bzw. genauer root@localhost. Einen root@127.0.0.1 gibt es nicht.
Benutzer auflisten / list users / show users:
# MariaDB 10.5+
select * from mysql.global_priv;
# earlier versions:
select user, host, grant_priv from mysql.user;
Rechte eines Benutzers ermitteln:
show grants for 'user';
Benutzer erstellen und keinerlei brauchbare Rechte geben:
grant
usage
on *.*
to 'user'@'host'
identified by 'password';
flush privileges;
Benutzer mit SELECT-Rechten auf sämtliche Tabellen einer „accounting“-Datenbank anlegen:
grant
select
on accounting.*
to 'user'@'host'
identified by 'password';
flush privileges;
Benutzer mit minimalen Backup-Rechten auf eine DB anlegen, die auch Events, Triggers, Views etc. verwendet:
grant
select, event, lock tables, process, reload, show view, super, trigger
on *.*
to 'user'@'host'
identified by 'password';
flush privileges;
Benutzer mit minimalen Backup- UND Restore-Rechten anlegen (der Restore sollte nach Möglichkeit jedoch mit einem anderen, administrativen Account durchgeführt werden):
grant
select, create view, drop, event, insert, lock tables, process, reload, show view, super, trigger
on *.*
to 'user'@'host'
identified by 'password';
flush privileges;
Bemerkung
Werden zwei oder mehr Privilegien auf Datenbank und Tabellen angegeben, wird immer die spezifischere Regel angewendet. Beispiel:
database.table:ALTER
database.*:DELETE,SELECT
Damit darf der Benutzer DELETE und SELECT auf alle Tabellen in database
ausführen, auf die Tabelle table
allerdings nur ein ALTER. Soll der Benutzer auf table
zusätzlich ALTER ausführen dürfen, müssen die Regeln wie folgt aussehen:
database.table:ALTER,DELETE,SELECT
database.*:DELETE,SELECT
Benutzer alle Berechtigungen entziehen:
revoke all privileges, grant option from 'user'@'host';
Passwort eines Benutzers ändern/zurücksetzen - man beachte, dass flush
zuerst ausgeführt werden muss:
flush privileges;
set password for `user`@`host` = password('new-password');
Passwort des root-Benutzers zurücksetzen
Ab 10.4:
systemctl stop mariadb
cat > /etc/my.cnf.d/zzz-password-recovery.cnf << 'EOF'
[server]
skip_grant_tables = ON
skip_networking = 1
EOF
systemctl start mariadb
mysql --user root
FLUSH PRIVILEGES;
SET PASSWORD FOR `mariadb-admin`@`%` = PASSWORD('new-password');
rm /etc/my.cnf.d/zzz-password-recovery.cnf
systemctl restart mariadb
Bash und MariaDB-Sessions
Einige SQL-Befehle verlangen, dass die Session für nachfolgende Befehle offen bleibt, z.B. FLUSH TABLES WITH READ LOCK
- doch wie macht man das in einem Bash-Script? Der Trick ist, das Kommando auszuführen, genügend lange in SQL zu warten, das Konstrukt in den Hintergrund zu schicken, anschliessend seine Arbeit zu erledigen und am Ende den Hintergrundprozess zu killen:
echo Locking the database for max. 10 minutes by sleeping in background task...
$MYSQL --host=$HOST --user=$USER --password=$PASSWORD \
--execute "BACKUP STAGE START; BACKUP STAGE BLOCK_COMMIT; DO SLEEP(600);" &
sleep 1
# do whatever you want to do
echo Finished, killing the previous background task to unlock.
kill $! 2>/dev/null
wait $! 2>/dev/null
Storage Engines
ColumnStore kann im Gegensatz zu früher ab Version 10.5.4 direkt als Storage Engine in MariaDB laufen, muss aber nach wie vor separat installiert werden.
MariaDB 10.6 mit ColumnStore
# minimize swapping
vm.swappiness = 1
# Increase the TCP max buffer size
net.core.rmem_max = 16777216
net.core.wmem_max = 16777216
# Increase the TCP buffer limits
# min, default, and max number of bytes to use
net.ipv4.tcp_rmem = 4096 87380 16777216
net.ipv4.tcp_wmem = 4096 65536 16777216
# don't cache ssthresh from previous connection
net.ipv4.tcp_no_metrics_save = 1
# for 1 GigE, increase this to 2500
# for 10 GigE, increase this to 30000
net.core.netdev_max_backlog = 2500
sysctl --load=/etc/sysctl.d/90-mariadb-columnstore.conf
Während der Installation SELinux auf Permissive setzen:
setenforce 0
Character Encoding:
dnf -y install glibc-locale-source glibc-langpack-en
localedef -i en_US -f UTF-8 en_US.UTF-8
MariaDB inklusive ColumnStore-Engine installieren:
dnf -y install wget
wget https://downloads.mariadb.com/MariaDB/mariadb_repo_setup
echo "367a80b01083c34899958cdd62525104a3de6069161d309039e84048d89ee98b mariadb_repo_setup" | sha256sum -c -
chmod +x mariadb_repo_setup
./mariadb_repo_setup --mariadb-server-version="mariadb-10.6"
dnf -y install epel-release
dnf -y install jemalloc
dnf -y install MariaDB-server MariaDB-backup MariaDB-shared MariaDB-client MariaDB-columnstore-engine
SELinux konfigurieren und aktivieren (dieses Vorgehen kann auch immer mal wieder im laufenden Betrieb nötig sein, bis die Policy steht):
dnf -y install policycoreutils python3-policycoreutils policycoreutils-python-utils
grep mysqld /var/log/audit/audit.log | audit2allow -M mariadb_local
sudo semodule -i mariadb_local.pp
setenforce 1
Cross Engine Join-Benutzer in ColumnStore konfigurieren:
mcsSetConfig CrossEngineSupport Host 127.0.0.1
mcsSetConfig CrossEngineSupport Port 3306
mcsSetConfig CrossEngineSupport User cross_engine
mcsSetConfig CrossEngineSupport Password linuxfabrik
Start:
systemctl enable --now mariadb
systemctl enable --now mariadb-columnstore
Aufräumen:
mariadb-secure-installation
Benutzer anlegen auf ColumnStore im Single-Node-Betrieb (funktioniert bis zum Anlegen des „mariadb-admin“ anfangs nur lokal unter dem „root“-Benutzer, nicht per MySQL Workbench o.ä.):
mariadb
# new mariadb-admin
grant all privileges on *.* to 'mariadb-admin'@'localhost' identified by 'linuxfabrik' with grant option;
# cross-engine-join-user from above
grant select, process on *.* to `cross_engine`@`127.0.0.1` identified by 'linuxfabrik';
grant select, process on *.* to `cross_engine`@`localhost` identified by 'linuxfabrik';
flush privileges;
Test:
SHOW STATUS LIKE '%columnstore%';
CREATE DATABASE inventory;
CREATE TABLE inventory.products (
product_name varchar(11) NOT NULL DEFAULT '',
supplier varchar(128) NOT NULL DEFAULT '',
quantity varchar(128) NOT NULL DEFAULT '',
unit_cost varchar(128) NOT NULL DEFAULT ''
) ENGINE=Columnstore DEFAULT CHARSET=utf8;
DROP DATABASE inventory;
Settings:
<!-- better config -->
<VersionBuffer>
<!-- VersionBufferFileSize must be a multiple of 8192.
One version buffer file will be put on each DB root. -->
<VersionBufferFileSize>4GB</VersionBufferFileSize>
</VersionBuffer>
<DBBC>
<NumBlocksPct>25</NumBlocksPct>
</DBBC>
<HashJoin>
<TotalUmMemory>50%</TotalPmUmMemory>
<AllowDiskBasedJoin>Y</AllowDiskBasedJoin>
</HashJoin>
Restart:
systemctl restart mariadb-columnstore.service
Siehe auch https://mariadb.com/docs/ent/deploy/topologies/single-node/community-columnstore-cs10-6
ColumnStore bis 1.2.5 (bis RHEL 7)
# minimize swapping
vm.swappiness = 1
# Increase the TCP max buffer size
net.core.rmem_max = 16777216
net.core.wmem_max = 16777216
# Increase the TCP buffer limits
# min, default, and max number of bytes to use
net.ipv4.tcp_rmem = 4096 87380 16777216
net.ipv4.tcp_wmem = 4096 65536 16777216
# don't cache ssthresh from previous connection
net.ipv4.tcp_no_metrics_save = 1
# for 1 GigE, increase this to 2500
# for 10 GigE, increase this to 30000
net.core.netdev_max_backlog = 2500
sysctl --load=/etc/sysctl.d/90-mariadb-columnstore.conf
Während der Installation SELinux auf Permissive setzen:
setenforce 0
Character Encoding:
localedef -i en_US -f UTF-8 en_US.UTF-8
ColumnStore installieren:
yum -y install boost
yum -y install expect perl perl-DBI openssl zlib file sudo libaio rsync snappy net-tools numactl-libs nmap
wget https://downloads.mariadb.com/ColumnStore/1.2.5/centos/x86_64/7/mariadb-columnstore-1.2.5-1-centos7.x86_64.rpm.tar.gz
tar xvzf mariadb-columnstore-*-centos7.x86_64.rpm.tar.gz
yum -y install mariadb-columnstore-*.rpm
/usr/local/mariadb/columnstore/bin/quick_installer_single_server.sh
source /etc/profile.d/columnstoreAlias.sh
#
# MariaDB ColumnStore Log Rotate file that gets installed in /etc/logrotate.d
# as part of the package installation
#
/var/log/mariadb/columnstore/*.log {
missingok
rotate 7
daily
dateext
copytruncate
olddir /var/log/mariadb/columnstore/archive
su root root
}
#/usr/local/mariadb/columnstore/etc/Columnstore.xml {
# daily
# dateext
# copy
# olddir /usr/local/mariadb/columnstore/etc/
#}
/usr/local/mariadb/columnstore/mysql/db/*.err {
missingok
rotate 7
daily
dateext
copytruncate
olddir /usr/local/mariadb/columnstore/mysql/db
su root root
}
Settings finden sich in zwei Dateien (empfohlene Einstellungen):
<!-- better config -->
<VersionBuffer>
<!-- VersionBufferFileSize must be a multiple of 8192.
One version buffer file will be put on each DB root. -->
<VersionBufferFileSize>4GB</VersionBufferFileSize>
</VersionBuffer>
<DBBC>
<NumBlocksPct>25</NumBlocksPct>
</DBBC>
<HashJoin>
<TotalUmMemory>50%</TotalPmUmMemory>
<AllowDiskBasedJoin>Y</AllowDiskBasedJoin>
</HashJoin>
[client]
...
default-character-set=utf8
[mysqld]
...
character-set-server = utf8
collation-server = uca1400_as_ci
init-connect = 'SET NAMES utf8'
max_allowed_packet = 4M
max_length_for_sort_data = 4M
/usr/local/mariadb/columnstore/bin/setConfig SystemConfig SystemLang en_US.utf8
mcsadmin restartSystem y
SELinux konfigurieren und aktivieren:
yum -y install policycoreutils-python
grep mysqld /var/log/audit/audit.log | audit2allow -M mariadb_local
semodule -i mariadb_local.pp
setenforce 1
Start:
systemctl enable --now columnstore
Aufräumen:
/usr/local/mariadb/columnstore/mysql/bin/mysql_secure_installation
Benutzer anlegen auf ColumnStore im Single-Node-Betrieb, z.B. in MySQL Workbench:
/usr/local/mariadb/columnstore/mysql/bin/mysql --user root --password
# new mariadb-admin
grant all privileges on *.* to 'mariadb-admin'@'localhost' identified by 'linuxfabrik' with grant option;
Test:
CREATE DATABASE inventory;
CREATE TABLE inventory.products (
product_name varchar(11) NOT NULL DEFAULT '',
supplier varchar(128) NOT NULL DEFAULT '',
quantity varchar(128) NOT NULL DEFAULT '',
unit_cost varchar(128) NOT NULL DEFAULT ''
) ENGINE=Columnstore DEFAULT CHARSET=utf8;
DROP DATABASE inventory;
Siehe auch https://mariadb.com/docs/deploy/community-single-columnstore/.
Benutzer in ColumnStore anlegen:
mcsmysql -u root
# or in MySQL Workbench, for example
grant select on newuser.* to `newuser`@`localhost` identified by 'password';
grant create temporary tables on infinidb_vtable.* to `newuser`@`localhost`;
flush privileges;
SQL
Typen von SQL-Statements:
Data Manipulation Language (DML): SELECT, INSERT, UPDATE, DELETE, MERGE
Data Definition Language (DDL): CREATE, ALTER, DROP, RENAME, TRUNCATE, COMMENT
Data Control Language (DCL): GRANT, REVOKE
Transaction Control: COMMIT, ROLLBACK, SAVEPOINT
Test-Datenbanken
Die hier genannten Beispiel-Datenbanken werden teilweise auch für MySQL-Zertifizierungen und Schulungen verwendet.
- Datenbank „employees“ (167 MB, inkl. gewünschter Inkonsistenzen):
DB-Name: employees
git clone https://github.com/datacharmer/test_db.git cd test_db/ mysql --user=$DBUSER --password < employees.sql
Datenbank-Tabelle „employees“ für Tests 4mal pro Sekunde laufend mit Daten füllen - ideal, um eine Ausgangsbasis für Backup- und Replikationstests zu haben:
for i in {1000000..2000000}; do echo $i mysql --user=$DBUSER --password=$DBPASS \ --execute="INSERT INTO \`employees\`.\`employees\` (\`emp_no\`, \`birth_date\`, \`first_name\`, \`last_name\`, \`gender\`, \`hire_date\`) VALUES ('$i', '2020-12-31', '$i', '$i', 'M', '2020-12-31');" sleep 0.25 done
- Datenbank „world_x“
DB-Name: world_x
wget https://downloads.mysql.com/docs/world_x-db.tar.gz tar xvzf world_x-db.tar.gz mysql --user=$DBUSER --password < world_x-db/world_x.sql
- Datenbank „sakila“
designed to represent a DVD rental store
DB-Name: sakila
wget https://downloads.mysql.com/docs/sakila-db.tar.gz tar xvzf sakila-db.tar.gz mysql --user=$DBUSER --password < sakila-db/sakila-schema.sql mysql --user=$DBUSER --password < sakila-db/sakila-data.sql
- Datenbank „menagerie“
Hilfe: nicht verfügbar
DB-Name: menagerie
wget https://downloads.mysql.com/docs/menagerie-db.tar.gz tar xvzf menagerie-db.tar.gz mysql --user=$DBUSER --password --execute='create database menagerie;' mysql --user=$DBUSER --password menagerie < menagerie-db/cr_pet_tbl.sql mysqlimport --user=$DBUSER --password --local menagerie menagerie-db/pet.txt mysql --user=$DBUSER --password menagerie < menagerie-db/ins_puff_rec.sql mysql --user=$DBUSER --password menagerie < menagerie-db/cr_event_tbl.sql mysqlimport --user=$DBUSER --password --local menagerie menagerie-db/event.txt
Tabellen optimieren und reparieren
Achtung: das Kommando lockt die Tabellen, daher nur bei Bedarf und zu Randzeiten ausführen.
mysqlcheck --user=root --password=password --optimize --all-databases
mysqlcheck --user=root --password=password --repair --all-databases
JSON in SQL
JSON-Text in Tabelle data
beispielsweise {"email":{"value":"info@example.com"}}
.
Passende SQL-Query:
select json_extract(data, '$.email.value') from ...
Cluster
Master/Master Active/Passive-Setup einrichten
Auf Master 1:
show master status;
Auf Master 2:
show master status\G;
STOP SLAVE;
CHANGE MASTER TO
MASTER_HOST='master1.example.com',
MASTER_USER='mariadb-replica',
MASTER_PASSWORD='password',
MASTER_PORT=3306,
MASTER_LOG_FILE='mariadb-bin.000002', # from Master 1
MASTER_LOG_POS=664; # from Master 1
START SLAVE;
show slave status\G
Zurück auf Master 1, hier aber die Informationen von Master 2 eintragen:
STOP SLAVE;
CHANGE MASTER TO
MASTER_HOST='master2.example.com',
MASTER_USER='mariadb-replica',
MASTER_PASSWORD='password',
MASTER_PORT=3306,
MASTER_LOG_FILE='mariadb-bin.000002', # from Master 2
MASTER_LOG_POS=664; # from Master 2
START SLAVE;
Security und Auditing
MariaDB Audit-Plugin
Grundsätzlich besteht der Zweck des MariaDB Audit Plugin darin, die Aktivitäten des Servers zu protokollieren. Für jede Client-Sitzung wird aufgezeichnet, wer sich mit dem Server verbunden hat (d.h. Benutzername und Host), welche Abfragen ausgeführt wurden, auf welche Tabellen zugegriffen wurde und welche Servervariablen geändert wurden. Diese Informationen werden in einer rotierenden Log-Datei gespeichert oder können an den lokalen syslogd gesendet werden.
Installation:
[mysqld]
plugin_load_add = server_audit
# Prohibiting Uninstallation:
server_audit = FORCE_PLUS_PERMANENT
server_audit_events = "CONNECT,QUERY,TABLE,QUERY_DDL,QUERY_DML,QUERY_DCL,QUERY_DML_NO_SELECT"
server_audit_excl_users =
server_audit_file_path = server_audit.log
server_audit_file_rotate_now = off
server_audit_file_rotate_size = 10M
server_audit_file_rotations = 9
server_audit_incl_users =
server_audit_logging = on
server_audit_output_type = file
server_audit_query_log_limit = 1024
server_audit_syslog_facility = LOG_USER
server_audit_syslog_ident = mysql-server_auditing
server_audit_syslog_info =
server_audit_syslog_priority = LOG_INFO
Anschliessend:
SHOW GLOBAL VARIABLES LIKE 'server_audit%';
Log-Datei findet sich mit obiger Konfiguration auf einem RHEL-basierten System unter /var/lib/mysql/server_audit.log
. Beispiel für den Inhalt:
timestamp ,server,username,host,ConnID,queryid,operation, database, object, retcode
20211115 09:04:01,server,username,host,15, 35, QUERY, database, 'SET NAMES latin1', 0
Mehr Info hier: https://mariadb.com/kb/en/mariadb-audit-plugin/.
Tuning und Troubleshooting
Slow Query Log untersuchen
# summarize the output file and sort the output by rows affected or average rows affected
mariadb-dumpslow -s t log_file
mysqltuner
dnf -y install mysqltuner
wget https://raw.githubusercontent.com/major/MySQLTuner-perl/master/basic_passwords.txt --output-document=basic_passwords.txt
wget https://raw.githubusercontent.com/major/MySQLTuner-perl/master/vulnerabilities.csv --output-document=vulnerabilities.csv
mysqltuner --user mariadb-admin --pass mypass --passwordfile basic_passwords.txt --cvefile vulnerabilities.csv
MySQL sollte da schon 24h oder länger gelaufen sein.
Die Ausgaben bedeuten:
[--]
: „Info Print“[OK]
: „Good Print“[!!]
: „Bad Print“
Beispielausgabe:
>> MySQLTuner 1.7.21 - Major Hayden <major@mhtx.net>
>> Bug reports, feature requests, and downloads at http://mysqltuner.pl/
>> Run with '--help' for additional options and output filtering
[--] Skipped version check for MySQLTuner script
[OK] Logged in using credentials passed on the command line
[OK] Currently running supported MySQL version 10.3.30-MariaDB
[OK] Operating on 64-bit architecture
-------- Log file Recommendations ------------------------------------------------------------------
[!!] Log file ./mariadb-error.log doesn't exist
-------- Storage Engine Statistics -----------------------------------------------------------------
[--] Status: +Aria +CSV +InnoDB +MEMORY +MRG_MyISAM +MyISAM +PERFORMANCE_SCHEMA +SEQUENCE
[--] Data in MyISAM tables: 10.5M (Tables: 4)
[--] Data in InnoDB tables: 1.5G (Tables: 906)
[!!] Total fragmented tables: 2
-------- Analysis Performance Metrics --------------------------------------------------------------
[--] innodb_stats_on_metadata: OFF
[OK] No stat updates during querying INFORMATION_SCHEMA.
-------- Security Recommendations ------------------------------------------------------------------
[OK] There are no anonymous accounts for any database users
[OK] All database users have passwords assigned
[!!] User ''root'@%' does not specify hostname restrictions.
[!!] There is no basic password file list!
-------- CVE Security Recommendations --------------------------------------------------------------
[--] Skipped due to --cvefile option undefined
-------- Performance Metrics -----------------------------------------------------------------------
[--] Up for: 1d 14h 48m 46s (48M q [343.594 qps], 192K conn, TX: 501G, RX: 7G)
[--] Reads / Writes: 95% / 5%
[--] Binary logging is disabled
[--] Physical Memory : 7.6G
[--] Max MySQL memory : 4.3G
[--] Other process memory: 0B
[--] Total buffers: 1.3G global + 19.7M per thread (151 max threads)
[--] P_S Max memory usage: 104M
[--] Galera GCache Max memory usage: 0B
[OK] Maximum reached memory usage: 1.7G (21.62% of installed RAM)
[OK] Maximum possible memory usage: 4.3G (56.06% of installed RAM)
[OK] Overall possible memory usage with other process is compatible with memory available
[OK] Slow queries: 0% (0/48M)
[OK] Highest usage of available connections: 9% (14/151)
[OK] Aborted connections: 0.00% (1/192942)
[OK] Query cache is disabled by default due to mutex contention on multiprocessor machines.
[OK] Sorts requiring temporary tables: 0% (44 temp sorts / 10M sorts)
[!!] Joins performed without indexes: 2912
[!!] Temporary tables created on disk: 34% (272K on disk / 783K total)
[OK] Thread cache hit rate: 99% (14 created / 192K connections)
[OK] Table cache hit rate: 99% (1K open / 1K opened)
[!!] table_definition_cache(400) is lower than number of tables(1585)
[OK] Open file limit used: 0% (67/32K)
[OK] Table locks acquired immediately: 100% (5K immediate / 5K locks)
-------- Performance schema ------------------------------------------------------------------------
[--] Memory used by P_S: 104.0M
[--] Sys schema isn't installed.
-------- ThreadPool Metrics ------------------------------------------------------------------------
[--] ThreadPool stat is enabled.
[--] Thread Pool Size: 4 thread(s).
[--] Using default value is good enough for your version (10.3.30-MariaDB)
-------- MyISAM Metrics ----------------------------------------------------------------------------
[!!] Key buffer used: 18.3% (24M used / 134M cache)
[OK] Key buffer size / total MyISAM indexes: 128.0M/3.3M
[OK] Read Key buffer hit rate: 98.8% (1K cached / 21 reads)
-------- InnoDB Metrics ----------------------------------------------------------------------------
[--] InnoDB is enabled.
[--] InnoDB Thread Concurrency: 0
[OK] InnoDB File per table is activated
[!!] InnoDB buffer pool / data size: 1.0G/1.5G
[OK] Ratio InnoDB log file size / InnoDB Buffer pool size: 128.0M * 2/1.0G should be equal to 25%
[OK] InnoDB buffer pool instances: 1
[--] Number of InnoDB Buffer Pool Chunk : 8 for 1 Buffer Pool Instance(s)
[OK] Innodb_buffer_pool_size aligned with Innodb_buffer_pool_chunk_size & Innodb_buffer_pool_instances
[OK] InnoDB Read buffer efficiency: 100.00% (2181003656 hits/ 2181088118 total)
[!!] InnoDB Write Log efficiency: 75.7% (1591920 hits/ 2102867 total)
[OK] InnoDB log waits: 0.00% (0 waits / 510947 writes)
-------- Aria Metrics ------------------------------------------------------------------------------
[--] Aria Storage Engine is enabled.
[OK] Aria pagecache size / total Aria indexes: 128.0M/0B
[OK] Aria pagecache hit rate: 98.6% (18M cached / 264K reads)
-------- TokuDB Metrics ----------------------------------------------------------------------------
[--] TokuDB is disabled.
-------- XtraDB Metrics ----------------------------------------------------------------------------
[--] XtraDB is disabled.
-------- Galera Metrics ----------------------------------------------------------------------------
[--] Galera is disabled.
-------- Replication Metrics -----------------------------------------------------------------------
[--] Galera Synchronous replication: NO
[--] No replication slave(s) for this server.
[--] Binlog format: ROW
[--] XA support enabled: ON
[--] Semi synchronous replication Master: OFF
[--] Semi synchronous replication Slave: OFF
[--] This is a standalone server
-------- Recommendations ---------------------------------------------------------------------------
General recommendations:
Run OPTIMIZE TABLE to defragment tables for better performance
OPTIMIZE TABLE `mydb`.`http_error_log`; -- can free 100 MB
OPTIMIZE TABLE `mydb`.`versions`; -- can free 5382 MB
Total freed space after theses OPTIMIZE TABLE : 5482 Mb
Restrict Host for 'root'@% to 'root'@LimitedIPRangeOrLocalhost
RENAME USER 'root'@'%' TO 'root'@LimitedIPRangeOrLocalhost;
We will suggest raising the 'join_buffer_size' until JOINs not using indexes are found.
See https://dev.mysql.com/doc/internals/en/join-buffer-size.html
(specially the conclusions at the bottom of the page).
When making adjustments, make tmp_table_size/max_heap_table_size equal
Reduce your SELECT DISTINCT queries which have no LIMIT clause
Consider installing Sys schema from https://github.com/mysql/mysql-sys for MySQL
Consider installing Sys schema from https://github.com/FromDual/mariadb-sys for MariaDB
Variables to adjust:
join_buffer_size (> 1.0M, or always use indexes with JOINs)
tmp_table_size (> 16M)
max_heap_table_size (> 16M)
table_definition_cache(400) > 1585 or -1 (autosizing if supported)
innodb_buffer_pool_size (>= 1.5G) if possible.
Zu prüfende Einstellungen auf einem dedizierten MariaDB-System:
binlog_format = MIXED
innodb_buffer_pool_instances = 2
innodb_buffer_pool_size = 80% vom verfügbaren ("available") RAM
innodb_file_format = barracuda
innodb_file_per_table = on
innodb_io_capacity = 4000
innodb_large_prefix = on
innodb_log_file_size = 25% von innodb_buffer_pool_size / 2
innodb_stats_on_metadata = off
join_buffer_size = 1M
log_error = hostname.err
max_heap_table_size = 256M
performance_schema = on
query_cache_limit = 2M
query_cache_size = 0
query_cache_type = off
skip_name_resolve = on
thread_cache_size = 256
tmp_table_size = 256M
Built on 2024-09-03