forked from zph/moresql
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathfull_sync.go
153 lines (140 loc) · 3.74 KB
/
full_sync.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
143
144
145
146
147
148
149
150
151
152
153
package moresql
import (
"expvar"
"fmt"
"time"
log "github.com/Sirupsen/logrus"
"github.com/jmoiron/sqlx"
"github.com/orcaman/concurrent-map"
"github.com/paulbellamy/ratecounter"
"github.com/rwynn/gtm"
mgo "gopkg.in/mgo.v2"
)
type Syncer interface {
Read() func()
Write() func()
BuildOpFromMgo() func(o Statement, e DBResult, coll Collection) gtm.Op
}
type FullSyncer struct {
Config Config
Output *sqlx.DB
Mongo *mgo.Session
C chan DBResult
done chan bool
insertCounter *ratecounter.RateCounter
readCounter *ratecounter.RateCounter
}
func (z *FullSyncer) Read() {
for dbName, v := range z.Config {
db := z.Mongo.DB(dbName)
for name := range v.Collections {
coll := db.C(name)
iter := coll.Find(nil).Iter()
var result map[string]interface{}
for iter.Next(&result) {
z.readCounter.Incr(1)
z.C <- DBResult{dbName, name, result}
// Clear out result data for next round
result = make(map[string]interface{})
}
if err := iter.Close(); err != nil {
log.Error("Unable to close iterator: %s", err)
}
}
}
close(z.C)
wg.Done()
}
func (z *FullSyncer) Write() {
var workers [workerCountOverflow]int
tables := z.buildTables()
for _ = range workers {
wg.Add(1)
go z.writer(&tables)
}
wg.Done()
}
func BuildOpFromMgo(mongoFields []string, e DBResult, coll Collection) *gtm.Op {
var op gtm.Op
op.Data = e.Data
opRef := EnsureOpHasAllFields(&op, mongoFields)
opRef.Id = e.Data["_id"]
// Set to I so we are consistent about these beings inserts
// This avoids our guardclause in sanitize
opRef.Operation = "i"
data := SanitizeData(coll.Fields, opRef)
opRef.Data = data
return opRef
}
func (z *FullSyncer) writer(tables *cmap.ConcurrentMap) {
ForStatement:
for {
select {
case e, more := <-z.C:
if !more {
break ForStatement
}
key := createFanKey(e.MongoDB, e.Collection)
v, ok := tables.Get(key)
if ok && !v.(bool) {
// Table doesn't exist, skip
break
}
o, coll := z.statementFromDbCollection(e.MongoDB, e.Collection)
op := BuildOpFromMgo(o.mongoFields(), e, coll)
s := o.BuildUpsert()
log.WithFields(log.Fields{
"collection": e.Collection,
"id": op.Id,
}).Info("Syncing record")
log.Debug("SQL Command ", s)
log.Debug("Data ", op.Data)
log.Debug("Executing statement: ", s)
_, err := z.Output.NamedExec(s, op.Data)
log.Debug("Statement executed successfully")
z.insertCounter.Incr(1)
if err != nil {
log.WithFields(log.Fields{
"description": err,
}).Error("Error")
if err.Error() == fmt.Sprintf(`pq: relation "%s" does not exist`, e.Collection) {
tables.Set(key, false)
}
}
}
}
wg.Done()
}
func (z *FullSyncer) statementFromDbCollection(db string, collectionName string) (Statement, Collection) {
c := z.Config[db].Collections[collectionName]
return Statement{c}, c
}
func (z *FullSyncer) buildTables() (tables cmap.ConcurrentMap) {
tables = cmap.New()
for dbName, db := range z.Config {
for collectionName := range db.Collections {
// Assume all tables are present
tables.Set(createFanKey(dbName, collectionName), true)
}
}
return
}
func NewSynchronizer(config Config, pg *sqlx.DB, mongo *mgo.Session) FullSyncer {
c := make(chan DBResult)
insertCounter := ratecounter.NewRateCounter(1 * time.Second)
readCounter := ratecounter.NewRateCounter(1 * time.Second)
expvar.Publish("insert/sec", insertCounter)
expvar.Publish("read/sec", readCounter)
done := make(chan bool, 2)
sync := FullSyncer{config, pg, mongo, c, done, insertCounter, readCounter}
return sync
}
func FullSync(config Config, pg *sqlx.DB, mongo *mgo.Session) {
sync := NewSynchronizer(config, pg, mongo)
wg.Add(2)
log.Debug("Starting writer")
go sync.Write()
log.Debug("Starting reader")
go sync.Read()
wg.Wait()
}