Avez-vous déjà été confronté à des situations où vous déployez une nouvelle mise à jour en production avec des schémas de base de données mis à jour, mais avez ensuite rencontré des bugs et devez annuler les choses... c'est à ce moment-là que la migration entre en place.
La migration de bases de données répond à plusieurs objectifs clés :
Pour créer une configuration complète de niveau production pour un service Golang utilisant GORM avec MySQL qui permet des migrations, des mises à jour et des restaurations faciles, vous devez inclure des outils de migration, gérer le regroupement de connexions de base de données et garantir des définitions de structure appropriées. Voici un exemple complet pour vous guider tout au long du processus :
/golang-service |-- main.go |-- database | |-- migration.go |-- models | |-- user.go |-- config | |-- config.go |-- migrations | |-- ... |-- go.mod
package config import ( "fmt" "log" "os" "time" "github.com/joho/godotenv" "gorm.io/driver/mysql" "gorm.io/gorm" ) var DB *gorm.DB func ConnectDB() { err := godotenv.Load() if err != nil { log.Fatal("Error loading .env file") } // charset=utf8mb4: Sets the character set to utf8mb4, which supports all Unicode characters, including emojis. // parseTime=True: Tells the driver to automatically parse DATE and DATETIME values into Go's time.Time type. // loc=Local: Uses the local timezone of the server for time-related queries and storage. dsn := fmt.Sprintf( "%s:%s@tcp(%s:%s)/%s?charset=utf8mb4&parseTime=True&loc=Local", os.Getenv("DB_USER"), os.Getenv("DB_PASS"), os.Getenv("DB_HOST"), os.Getenv("DB_PORT"), os.Getenv("DB_NAME"), ) db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{}) if err != nil { panic("failed to connect database") } sqlDB, err := db.DB() if err != nil { panic("failed to configure database connection") } // Set connection pool settings sqlDB.SetMaxIdleConns(10) sqlDB.SetMaxOpenConns(100) sqlDB.SetConnMaxLifetime(time.Hour) // 1.sqlDB.SetMaxIdleConns(10) // Sets the maximum number of idle (unused but open) connections in the connection pool. // A value of 10 means up to 10 connections can remain idle, ready to be reused. // 2. sqlDB.SetMaxOpenConns(100): // Sets the maximum number of open (active or idle) connections that can be created to the database. // A value of 100 limits the total number of connections, helping to prevent overloading the database. // 3. sqlDB.SetConnMaxLifetime(time.Hour): // Sets the maximum amount of time a connection can be reused before it’s closed. // A value of time.Hour means that each connection will be kept for up to 1 hour, after which it will be discarded and a new connection will be created if needed. DB = db }
package database import ( "golang-service/models" "golang-service/migrations" "gorm.io/gorm" ) func Migrate(db *gorm.DB) { db.AutoMigrate(&models.User{}) // Apply additional custom migrations if needed }
package models import "gorm.io/gorm" type User struct { gorm.Model Name string `json:"name"` }
DB_USER=root DB_PASS=yourpassword DB_HOST=127.0.0.1 DB_PORT=3306 DB_NAME=yourdb
package main import ( "golang-service/config" "golang-service/database" "golang-service/models" "github.com/gin-gonic/gin" "gorm.io/gorm" ) func main() { config.ConnectDB() database.Migrate(config.DB) r := gin.Default() r.POST("/users", createUser) r.GET("/users/:id", getUser) r.Run(":8080") } func createUser(c *gin.Context) { var user models.User if err := c.ShouldBindJSON(&user); err != nil { c.JSON(400, gin.H{"error": err.Error()}) return } if err := config.DB.Create(&user).Error; err != nil { c.JSON(500, gin.H{"error": err.Error()}) return } c.JSON(201, user) } func getUser(c *gin.Context) { id := c.Param("id") var user models.User if err := config.DB.First(&user, id).Error; err != nil { c.JSON(404, gin.H{"error": "User not found"}) return } c.JSON(200, user) }
Pour les environnements de production, nous pourrions utiliser une bibliothèque de migration comme golang-migrate pour appliquer, annuler ou refaire des migrations.
Installer golang-migrate :
go install -tags 'mysql' github.com/golang-migrate/migrate/v4/cmd/migrate@latest
Générer les fichiers de migration pour le tableau des utilisateurs
migrate create -ext=sql -dir=./migrations -seq create_users_table
Après avoir exécuté la commande, nous obtiendrons une paire de .up.sql (pour mettre à jour le schéma) et down.sql (pour une éventuelle restauration ultérieure). Le nombre 000001 est l'index de migration généré automatiquement.
/golang-service |-- migrations | |-- 000001_create_users_table.down.sql | |-- 000001_create_users_table.up.sql
Ajoutez une commande SQL pertinente au fichier .up et au fichier .down.
000001_create_users_table.up.sql
CREATE TABLE users ( id BIGINT AUTO_INCREMENT PRIMARY KEY, name VARCHAR(255) NOT NULL, created_at DATETIME, updated_at DATETIME, deleted_at DATETIME);
000001_create_users_table.down.sql
DROP TABLE IF EXISTS users;
Exécutez la migration vers le haut et appliquez les modifications à la base de données avec la commande suivante (indicateur -verbose pour voir plus de détails du journal) :
migrate -path ./migrations -database "mysql://user:password@tcp(localhost:3306)/dbname" -verbose up
Si nous rencontrons un problème avec la migration, nous pouvons utiliser la commande suivante pour voir la version actuelle de la migration et son statut :
migrate -path ./migrations -database "mysql://user:password@tcp(localhost:3306)/dbname" version
Si nous avons une migration interrompue pour certaines raisons, nous pouvons envisager d'utiliser la commande force (à utiliser avec précaution) avec le numéro de version de la migration sale. Si la version est 1 (on pourrait la vérifier dans la table migrations ou schema_migrations), nous exécuterions :
migrate -path ./migrations -database "mysql://user:password@tcp(localhost:3306)/dbname" force 1
À un moment donné, nous aimerions peut-être ajouter de nouvelles fonctionnalités et certaines d'entre elles pourraient nécessiter une modification des schémas de données, par exemple, nous aimerions ajouter un champ de courrier électronique à la table des utilisateurs. Nous le ferions comme suit.
Effectuer une nouvelle migration pour ajouter la colonne email au tableau des utilisateurs
migrate create -ext=sql -dir=./migrations -seq add_email_to_users
Nous avons maintenant une nouvelle paire de .up.sql et .down.sql
/golang-service |-- migrations | |-- 000001_create_users_table.down.sql | |-- 000001_create_users_table.up.sql | |-- 000002_add_email_to_users.down.sql | |-- 000002_add_email_to_users.up.sql
Ajout du contenu suivant aux *_add_email_to_users.*.sql fichiers
000002_add_email_to_users.up.sql
/golang-service |-- main.go |-- database | |-- migration.go |-- models | |-- user.go |-- config | |-- config.go |-- migrations | |-- ... |-- go.mod
000002_add_email_to_users.down.sql
package config import ( "fmt" "log" "os" "time" "github.com/joho/godotenv" "gorm.io/driver/mysql" "gorm.io/gorm" ) var DB *gorm.DB func ConnectDB() { err := godotenv.Load() if err != nil { log.Fatal("Error loading .env file") } // charset=utf8mb4: Sets the character set to utf8mb4, which supports all Unicode characters, including emojis. // parseTime=True: Tells the driver to automatically parse DATE and DATETIME values into Go's time.Time type. // loc=Local: Uses the local timezone of the server for time-related queries and storage. dsn := fmt.Sprintf( "%s:%s@tcp(%s:%s)/%s?charset=utf8mb4&parseTime=True&loc=Local", os.Getenv("DB_USER"), os.Getenv("DB_PASS"), os.Getenv("DB_HOST"), os.Getenv("DB_PORT"), os.Getenv("DB_NAME"), ) db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{}) if err != nil { panic("failed to connect database") } sqlDB, err := db.DB() if err != nil { panic("failed to configure database connection") } // Set connection pool settings sqlDB.SetMaxIdleConns(10) sqlDB.SetMaxOpenConns(100) sqlDB.SetConnMaxLifetime(time.Hour) // 1.sqlDB.SetMaxIdleConns(10) // Sets the maximum number of idle (unused but open) connections in the connection pool. // A value of 10 means up to 10 connections can remain idle, ready to be reused. // 2. sqlDB.SetMaxOpenConns(100): // Sets the maximum number of open (active or idle) connections that can be created to the database. // A value of 100 limits the total number of connections, helping to prevent overloading the database. // 3. sqlDB.SetConnMaxLifetime(time.Hour): // Sets the maximum amount of time a connection can be reused before it’s closed. // A value of time.Hour means that each connection will be kept for up to 1 hour, after which it will be discarded and a new connection will be created if needed. DB = db }
Exécutez à nouveau la commande up migration pour mettre à jour les schémas de données
package database import ( "golang-service/models" "golang-service/migrations" "gorm.io/gorm" ) func Migrate(db *gorm.DB) { db.AutoMigrate(&models.User{}) // Apply additional custom migrations if needed }
Nous devrons également mettre à jour la structure des utilisateurs de Golang (en ajoutant le champ Email) pour la maintenir synchronisée avec les nouveaux schémas..
package models import "gorm.io/gorm" type User struct { gorm.Model Name string `json:"name"` }
Si, pour une raison quelconque, nous rencontrons des bugs avec les nouveaux schémas mis à jour et que nous devons annuler, dans ce cas, nous utiliserons la commande down :
DB_USER=root DB_PASS=yourpassword DB_HOST=127.0.0.1 DB_PORT=3306 DB_NAME=yourdb
Le numéro 1 indique que nous souhaitons annuler 1 migration.
Ici, nous devons également mettre à jour manuellement la structure des utilisateurs de Golang (supprimer le champ E-mail) pour refléter les modifications du schéma de données.
package main import ( "golang-service/config" "golang-service/database" "golang-service/models" "github.com/gin-gonic/gin" "gorm.io/gorm" ) func main() { config.ConnectDB() database.Migrate(config.DB) r := gin.Default() r.POST("/users", createUser) r.GET("/users/:id", getUser) r.Run(":8080") } func createUser(c *gin.Context) { var user models.User if err := c.ShouldBindJSON(&user); err != nil { c.JSON(400, gin.H{"error": err.Error()}) return } if err := config.DB.Create(&user).Error; err != nil { c.JSON(500, gin.H{"error": err.Error()}) return } c.JSON(201, user) } func getUser(c *gin.Context) { id := c.Param("id") var user models.User if err := config.DB.First(&user, id).Error; err != nil { c.JSON(404, gin.H{"error": "User not found"}) return } c.JSON(200, user) }
Pour simplifier le processus de migration et de restauration, nous pouvons ajouter un Makefile .
go install -tags 'mysql' github.com/golang-migrate/migrate/v4/cmd/migrate@latest
Le contenu de Makefile comme suit.
migrate create -ext=sql -dir=./migrations -seq create_users_table
Maintenant, nous pouvons simplement exécuter make migrate_up ou make migrate_down sur CLI pour effectuer la migration et la restauration.
Avant d'annuler une migration ou d'apporter des modifications susceptibles d'affecter votre base de données, voici quelques points clés à considérer.
Il est donc crucial de sauvegarder vos données. Voici un bref guide :
Dump de base de données :
Utilisez des outils spécifiques à la base de données pour créer une sauvegarde complète de votre base de données. Pour MySQL, vous pouvez utiliser :
/golang-service |-- main.go |-- database | |-- migration.go |-- models | |-- user.go |-- config | |-- config.go |-- migrations | |-- ... |-- go.mod
Cela crée un fichier (backup_before_rollback.sql) qui contient toutes les données et le schéma de la base de données dbname.
Exporter des tableaux spécifiques :
Si vous n'avez besoin de sauvegarder que certaines tables, spécifiez-les dans la commande mysqldump :
package config import ( "fmt" "log" "os" "time" "github.com/joho/godotenv" "gorm.io/driver/mysql" "gorm.io/gorm" ) var DB *gorm.DB func ConnectDB() { err := godotenv.Load() if err != nil { log.Fatal("Error loading .env file") } // charset=utf8mb4: Sets the character set to utf8mb4, which supports all Unicode characters, including emojis. // parseTime=True: Tells the driver to automatically parse DATE and DATETIME values into Go's time.Time type. // loc=Local: Uses the local timezone of the server for time-related queries and storage. dsn := fmt.Sprintf( "%s:%s@tcp(%s:%s)/%s?charset=utf8mb4&parseTime=True&loc=Local", os.Getenv("DB_USER"), os.Getenv("DB_PASS"), os.Getenv("DB_HOST"), os.Getenv("DB_PORT"), os.Getenv("DB_NAME"), ) db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{}) if err != nil { panic("failed to connect database") } sqlDB, err := db.DB() if err != nil { panic("failed to configure database connection") } // Set connection pool settings sqlDB.SetMaxIdleConns(10) sqlDB.SetMaxOpenConns(100) sqlDB.SetConnMaxLifetime(time.Hour) // 1.sqlDB.SetMaxIdleConns(10) // Sets the maximum number of idle (unused but open) connections in the connection pool. // A value of 10 means up to 10 connections can remain idle, ready to be reused. // 2. sqlDB.SetMaxOpenConns(100): // Sets the maximum number of open (active or idle) connections that can be created to the database. // A value of 100 limits the total number of connections, helping to prevent overloading the database. // 3. sqlDB.SetConnMaxLifetime(time.Hour): // Sets the maximum amount of time a connection can be reused before it’s closed. // A value of time.Hour means that each connection will be kept for up to 1 hour, after which it will be discarded and a new connection will be created if needed. DB = db }
Vérifier la sauvegarde :
Assurez-vous que le fichier de sauvegarde a été créé et vérifiez sa taille ou ouvrez-le pour vous assurer qu'il contient les données nécessaires.
Stockez les sauvegardes en toute sécurité :
Conservez une copie de la sauvegarde dans un emplacement sécurisé, tel qu'un stockage cloud ou un serveur distinct, pour éviter la perte de données pendant le processus de restauration.
Pour sauvegarder vos données MySQL lorsque vous utilisez Golang et que vous les déployez sur AWS EKS, vous pouvez suivre ces étapes :
Utilisez mysqldump pour la sauvegarde de la base de données :
Créez un mysqldump de votre base de données MySQL à l'aide d'une tâche cron Kubernetes.
package database import ( "golang-service/models" "golang-service/migrations" "gorm.io/gorm" ) func Migrate(db *gorm.DB) { db.AutoMigrate(&models.User{}) // Apply additional custom migrations if needed }
Stockez-le dans un volume persistant ou un compartiment S3.
Automatiser avec Kubernetes CronJob :
Utilisez un CronJob Kubernetes pour automatiser le processus mysqldump.
Exemple de configuration YAML :yaml
package models import "gorm.io/gorm" type User struct { gorm.Model Name string `json:"name"` }
`
Utilisation des sauvegardes automatisées AWS RDS (si vous utilisez RDS) :
Si votre base de données MySQL est sur AWS RDS, vous pouvez tirer parti des sauvegardes et des instantanés automatisés RDS.
Définissez une période de conservation des sauvegardes et prenez des instantanés manuellement ou automatisez les instantanés à l'aide des fonctions Lambda.
Sauvegarder les volumes persistants (PV) avec Velero :
Utilisez Velero, un outil de sauvegarde pour Kubernetes, pour sauvegarder le volume persistant contenant les données MySQL.
Installez Velero sur votre cluster EKS et configurez-le pour sauvegarder sur S3.
En utilisant ces méthodes, vous pouvez vous assurer que vos données MySQL sont régulièrement sauvegardées et stockées en toute sécurité.
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!