@@ -31,10 +31,20 @@ type Loader interface {
31
31
32
32
// Migration represents migration step that will be runned on DB.
33
33
type Migration struct {
34
- ID int // ID of the migration, unique, positive, starts from 1.
35
- Name string // Name of the migration
36
- Apply string // Apply query
37
- Rollback string // Rollback query
34
+ ID int // ID of the migration, unique, positive, starts from 1.
35
+ Name string // Name of the migration
36
+ Apply string // Apply query
37
+ Rollback string // Rollback query
38
+ ApplyFn MigrationFn // Apply func
39
+ RollbackFn MigrationFn // Rollback func
40
+
41
+ isQuery bool // shortcut for the type of migration (query or func)
42
+ }
43
+
44
+ type MigrationFn func (db DB ) error
45
+
46
+ type DB interface {
47
+ Exec (ctx context.Context , query string , args ... interface {}) error
38
48
}
39
49
40
50
// Run the Migrator with migration queries provided by the Loader.
@@ -62,7 +72,10 @@ func loadMigrations(ms []*Migration, err error) ([]*Migration, error) {
62
72
case m .ID > want :
63
73
return nil , fmt .Errorf ("missing migration number: %d (have %d)" , want , m .ID )
64
74
default :
65
- // pass
75
+ if (m .Apply != "" || m .Rollback != "" ) && (m .ApplyFn != nil || m .RollbackFn != nil ) {
76
+ return nil , fmt .Errorf ("mixing queries and functions is not allowed (migration %d)" , m .ID )
77
+ }
78
+ m .isQuery = m .Apply != ""
66
79
}
67
80
}
68
81
return ms , nil
@@ -115,15 +128,20 @@ func runMigrationExclusive(ctx context.Context, m Migrator, ms []*Migration) err
115
128
for currentVersion != targetVersion {
116
129
current := ms [currentVersion ]
117
130
sequence := current .ID
118
- query := current .Apply
131
+ query , queryFn := current .Apply , current . ApplyFn
119
132
120
133
if direction == - 1 {
121
134
current = ms [currentVersion - 1 ]
122
135
sequence = current .ID - 1
123
- query = current .Rollback
136
+ query , queryFn = current .Rollback , current . RollbackFn
124
137
}
125
138
126
- if err := m .Exec (ctx , query ); err != nil {
139
+ if current .isQuery {
140
+ err = m .Exec (ctx , query )
141
+ } else {
142
+ err = queryFn (m )
143
+ }
144
+ if err != nil {
127
145
return fmt .Errorf ("exec: %w" , err )
128
146
}
129
147
0 commit comments