generated from mrz1836/go-template
-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmigrate.go
142 lines (120 loc) · 4.55 KB
/
migrate.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
package datastore
import (
"context"
"errors"
"fmt"
"github.com/newrelic/go-agent/v3/newrelic"
"go.mongodb.org/mongo-driver/mongo"
mongoOptions "go.mongodb.org/mongo-driver/mongo/options"
"gorm.io/gorm"
"gorm.io/gorm/logger"
)
// AutoMigrateDatabase will detect the database engine and perform the necessary migrations for the provided models.
// It supports MySQL, PostgreSQL, SQLite, and MongoDB. The function first checks if auto-migration is enabled in the
// client options. If not enabled, it logs a debug message and returns without performing any migration.
//
// For each model provided, it checks if the model has already been migrated to avoid duplicate migrations. If a model
// has already been migrated, it returns an error. Otherwise, it adds the model to the list of migrated models.
//
// Depending on the database engine, it performs the migration using the appropriate method:
// - For MongoDB, it calls `autoMigrateMongoDatabase` to create indexes as needed.
// - For SQL databases (MySQL, PostgreSQL, SQLite), it calls `autoMigrateSQLDatabase` using GORM to create or update
// the table schema.
//
// The function logs debug information about the migration process, including the database engine and the models being
// migrated.
//
// Parameters:
// - ctx: The context for the migration process, used for logging and tracing.
// - models: A variadic list of models to be migrated.
//
// Returns:
// - An error if the migration fails or if a model has already been migrated.
func (c *Client) AutoMigrateDatabase(ctx context.Context, models ...interface{}) error {
// Gracefully skip if not enabled
if !c.options.autoMigrate {
c.DebugLog(ctx, "auto migrate is disabled, skipping...")
return nil
}
// Make sure we have a supported engine
if c.Engine() != MySQL &&
c.Engine() != PostgreSQL &&
c.Engine() != SQLite &&
c.Engine() != MongoDB {
return ErrUnsupportedEngine
}
// Check the models against previously migrated models
for _, modelInterface := range models {
modelType := fmt.Sprintf("%T", modelInterface)
if c.HasMigratedModel(modelType) {
return errors.New("model " + modelType + " was already migrated")
}
c.options.migratedModels = append(c.options.migratedModels, modelType)
}
// Debug logs
c.DebugLog(ctx, fmt.Sprintf(
"database migration starting... engine: %s model_count: %d, models: %v",
c.Engine().String(),
len(models),
c.options.migratedModels,
))
// Migrate database for Mongo
if c.Engine() == MongoDB {
return autoMigrateMongoDatabase(ctx, c.Engine(), c.options, models...)
}
// Migrate database for SQL (using GORM)
return autoMigrateSQLDatabase(ctx, c.Engine(), c.options.db, c.IsDebug(), c.options.loggerDB, models...)
}
// IsAutoMigrate returns whether auto migration is on
func (c *Client) IsAutoMigrate() bool {
return c.options.autoMigrate
}
// autoMigrateMongoDatabase will start a new database for Mongo
func autoMigrateMongoDatabase(ctx context.Context, _ Engine, options *clientOptions,
_ ...interface{}) error {
var err error
if options.fields.customMongoIndexer != nil {
for collectionName, idx := range options.fields.customMongoIndexer() {
for _, index := range idx {
if err = createMongoIndex(
ctx, options, collectionName, false, index,
); err != nil {
return err
}
}
}
}
return nil
}
// createMongoIndex will create a mongo index
func createMongoIndex(ctx context.Context, options *clientOptions, modelName string, withPrefix bool,
index mongo.IndexModel) error {
collectionName := modelName
if !withPrefix {
collectionName = setPrefix(options.mongoDBConfig.TablePrefix, collectionName)
}
collection := options.mongoDB.Collection(collectionName)
_, err := collection.Indexes().CreateOne(
ctx, index, mongoOptions.CreateIndexes().SetMaxTime(defaultDatabaseCreateIndexTimeout),
)
return err
}
// autoMigrateSQLDatabase will attempt to create or update table schema
//
// See: https://gorm.io/docs/migration.html
func autoMigrateSQLDatabase(ctx context.Context, engine Engine, sqlWriteDB *gorm.DB,
debug bool, optionalLogger logger.Interface, models ...interface{}) error {
// Create a segment
txn := newrelic.FromContext(ctx)
if txn != nil {
defer txn.StartSegment("auto_migrate_sql_database").End()
}
// Create a session with config settings
sessionDb := sqlWriteDB.Session(getGormSessionConfig(sqlWriteDB.PrepareStmt, debug, optionalLogger))
// Run the auto migrate method
if engine == MySQL {
return sessionDb.Set("gorm:table_options", "ENGINE=InnoDB").AutoMigrate(models...)
}
// PostgreSQL and SQLite
return sessionDb.AutoMigrate(models...)
}