-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexec_migrate_sqlstatements.go
96 lines (79 loc) · 2.77 KB
/
exec_migrate_sqlstatements.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
package adapt
import (
"fmt"
"time"
)
func (e *exec) migrateWithSqlStatements(parsed *ParsedMigration, beforeFinishCallback func(target DBTarget) error) error {
if !e.driverIsDatabaseDriver {
e.log.Error("underlying driver isn't a DatabaseDriver! No way to apply a SqlStatementsSource")
return fmt.Errorf("SqlStatementsSource usage violation")
}
e.log.Debug("parsed migration has n statements", "n", len(parsed.Stmts))
if e.driverIsDatabaseDriverCustomMigration {
e.log.Debug("driver is a DatabaseDriverCustomMigration. Using the provided Migrate callback")
err := e.driverAsDatabaseDriverCustomMigration.Migrate(parsed, beforeFinishCallback)
if err != nil {
e.log.Error("failed to migrate using the custom migrate callback provided", "error", err)
return err
}
return nil
}
exec := func(target DBTarget) error {
for _, s := range parsed.Stmts {
e.log.Debug("executing statement", "statement", s)
started := time.Now()
if _, err := target.Exec(s); err != nil {
e.log.Error("failed executing statement", "statement", s, "error", err)
return err
}
end := time.Now()
e.log.Debug("executing statement took", "duration", end.Sub(started))
}
if beforeFinishCallback != nil {
e.log.Debug("beforeFinishCallback is provided. calling so cleanup can be performed within the (eventually running) same transaction")
err := beforeFinishCallback(target)
if err != nil {
e.log.Error("beforeFinishCallback failed", "error", err)
return err
} else {
e.log.Debug("beforeFinishCallback successful")
}
}
return nil
}
if !e.driverAsDatabaseDriver.SupportsTx() {
e.log.Debug("executing statements without transaction, because driver doesn't support transactions")
return exec(e.driverAsDatabaseDriver.DB())
}
if !parsed.UseTx {
e.log.Debug("executing statements without transaction, because transactions are disabled for this migration specifically")
return exec(e.driverAsDatabaseDriver.DB())
}
ctx, opts := e.driverAsDatabaseDriver.TxBeginOpts()
e.log.Debug("starting tx")
tx, err := e.driverAsDatabaseDriver.DB().BeginTx(ctx, opts)
if err != nil {
e.log.Error("failed to begin tx", "error", err)
return err
}
defer func() {
if err != nil {
e.log.Warn("exec failed. trying to rollback tx", "error", err)
if errRb := tx.Rollback(); errRb != nil {
e.log.Error("rollback failed too", "error", errRb)
} else {
e.log.Info("rollback successful")
}
err = fmt.Errorf("exec failed (%q) but rollback succeeded. Integrity should be protected, but manual cleanup is probably necessary", err)
return
}
e.log.Debug("committing tx")
err = tx.Commit()
if err != nil {
e.log.Error("commit failed", "error", err)
}
}()
e.log.Debug("executing statements in transaction")
err = exec(tx)
return err
}