Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
128 lines
2.0 KiB
128 lines
2.0 KiB
package nodb |
|
|
|
import ( |
|
"fmt" |
|
"sync" |
|
"time" |
|
|
|
"gitea.com/lunny/nodb/config" |
|
"gitea.com/lunny/nodb/store" |
|
"gitea.com/lunny/log" |
|
) |
|
|
|
type Nodb struct { |
|
cfg *config.Config |
|
|
|
ldb *store.DB |
|
dbs [MaxDBNumber]*DB |
|
|
|
quit chan struct{} |
|
jobs *sync.WaitGroup |
|
|
|
binlog *BinLog |
|
|
|
wLock sync.RWMutex //allow one write at same time |
|
commitLock sync.Mutex //allow one write commit at same time |
|
} |
|
|
|
func Open(cfg *config.Config) (*Nodb, error) { |
|
if len(cfg.DataDir) == 0 { |
|
cfg.DataDir = config.DefaultDataDir |
|
} |
|
|
|
ldb, err := store.Open(cfg) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
l := new(Nodb) |
|
|
|
l.quit = make(chan struct{}) |
|
l.jobs = new(sync.WaitGroup) |
|
|
|
l.ldb = ldb |
|
|
|
if cfg.BinLog.MaxFileNum > 0 && cfg.BinLog.MaxFileSize > 0 { |
|
l.binlog, err = NewBinLog(cfg) |
|
if err != nil { |
|
return nil, err |
|
} |
|
} else { |
|
l.binlog = nil |
|
} |
|
|
|
for i := uint8(0); i < MaxDBNumber; i++ { |
|
l.dbs[i] = l.newDB(i) |
|
} |
|
|
|
l.activeExpireCycle() |
|
|
|
return l, nil |
|
} |
|
|
|
func (l *Nodb) Close() { |
|
close(l.quit) |
|
l.jobs.Wait() |
|
|
|
l.ldb.Close() |
|
|
|
if l.binlog != nil { |
|
l.binlog.Close() |
|
l.binlog = nil |
|
} |
|
} |
|
|
|
func (l *Nodb) Select(index int) (*DB, error) { |
|
if index < 0 || index >= int(MaxDBNumber) { |
|
return nil, fmt.Errorf("invalid db index %d", index) |
|
} |
|
|
|
return l.dbs[index], nil |
|
} |
|
|
|
func (l *Nodb) FlushAll() error { |
|
for index, db := range l.dbs { |
|
if _, err := db.FlushAll(); err != nil { |
|
log.Errorf("flush db %d error %s", index, err.Error()) |
|
} |
|
} |
|
|
|
return nil |
|
} |
|
|
|
// very dangerous to use |
|
func (l *Nodb) DataDB() *store.DB { |
|
return l.ldb |
|
} |
|
|
|
func (l *Nodb) activeExpireCycle() { |
|
var executors []*elimination = make([]*elimination, len(l.dbs)) |
|
for i, db := range l.dbs { |
|
executors[i] = db.newEliminator() |
|
} |
|
|
|
l.jobs.Add(1) |
|
go func() { |
|
tick := time.NewTicker(1 * time.Second) |
|
end := false |
|
done := make(chan struct{}) |
|
for !end { |
|
select { |
|
case <-tick.C: |
|
go func() { |
|
for _, eli := range executors { |
|
eli.active() |
|
} |
|
done <- struct{}{} |
|
}() |
|
<-done |
|
case <-l.quit: |
|
end = true |
|
break |
|
} |
|
} |
|
|
|
tick.Stop() |
|
l.jobs.Done() |
|
}() |
|
}
|
|
|