package server import ( "fmt" "github.com/siddontang/go/log" "github.com/siddontang/ledisdb/config" "io" "io/ioutil" "os" "path" "sort" "sync" "time" ) const ( snapshotTimeFormat = "2006-01-02T15:04:05.999999999" ) type snapshotStore struct { sync.Mutex cfg *config.Config names []string quit chan struct{} } func snapshotName(t time.Time) string { return fmt.Sprintf("dmp-%s", t.Format(snapshotTimeFormat)) } func parseSnapshotName(name string) (time.Time, error) { var timeString string if _, err := fmt.Sscanf(name, "dmp-%s", &timeString); err != nil { println(err.Error()) return time.Time{}, err } when, err := time.Parse(snapshotTimeFormat, timeString) if err != nil { return time.Time{}, err } return when, nil } func newSnapshotStore(cfg *config.Config) (*snapshotStore, error) { if len(cfg.Snapshot.Path) == 0 { cfg.Snapshot.Path = path.Join(cfg.DataDir, "snapshot") } if err := os.MkdirAll(cfg.Snapshot.Path, 0755); err != nil { return nil, err } s := new(snapshotStore) s.cfg = cfg s.names = make([]string, 0, s.cfg.Snapshot.MaxNum) s.quit = make(chan struct{}) if err := s.checkSnapshots(); err != nil { return nil, err } go s.run() return s, nil } func (s *snapshotStore) Close() { close(s.quit) } func (s *snapshotStore) checkSnapshots() error { cfg := s.cfg snapshots, err := ioutil.ReadDir(cfg.Snapshot.Path) if err != nil { log.Errorf("read %s error: %s", cfg.Snapshot.Path, err.Error()) return err } names := []string{} for _, info := range snapshots { if path.Ext(info.Name()) == ".tmp" { log.Errorf("temp snapshot file name %s, try remove", info.Name()) os.Remove(path.Join(cfg.Snapshot.Path, info.Name())) continue } if _, err := parseSnapshotName(info.Name()); err != nil { log.Errorf("invalid snapshot file name %s, err: %s", info.Name(), err.Error()) continue } names = append(names, info.Name()) } //from old to new sort.Strings(names) s.names = names s.purge(false) return nil } func (s *snapshotStore) run() { t := time.NewTicker(60 * time.Minute) defer t.Stop() for { select { case <-t.C: s.Lock() if err := s.checkSnapshots(); err != nil { log.Errorf("check snapshots error %s", err.Error()) } s.Unlock() case <-s.quit: return } } } func (s *snapshotStore) purge(create bool) { var names []string maxNum := s.cfg.Snapshot.MaxNum num := len(s.names) - maxNum if create { num++ if num > len(s.names) { num = len(s.names) } } if num > 0 { names = append([]string{}, s.names[0:num]...) n := copy(s.names, s.names[num:]) s.names = s.names[0:n] } for _, name := range names { if err := os.Remove(s.snapshotPath(name)); err != nil { log.Errorf("purge snapshot %s error %s", name, err.Error()) } } } func (s *snapshotStore) snapshotPath(name string) string { return path.Join(s.cfg.Snapshot.Path, name) } type snapshotDumper interface { Dump(w io.Writer) error } type snapshot struct { io.ReadCloser f *os.File } func (st *snapshot) Read(b []byte) (int, error) { return st.f.Read(b) } func (st *snapshot) Close() error { return st.f.Close() } func (st *snapshot) Size() int64 { s, _ := st.f.Stat() return s.Size() } func (s *snapshotStore) Create(d snapshotDumper) (*snapshot, time.Time, error) { s.Lock() defer s.Unlock() s.purge(true) now := time.Now() name := snapshotName(now) tmpName := name + ".tmp" if len(s.names) > 0 { lastTime, _ := parseSnapshotName(s.names[len(s.names)-1]) if now.Nanosecond() <= lastTime.Nanosecond() { return nil, time.Time{}, fmt.Errorf("create snapshot file time %s is behind %s ", now.Format(snapshotTimeFormat), lastTime.Format(snapshotTimeFormat)) } } f, err := os.OpenFile(s.snapshotPath(tmpName), os.O_RDWR|os.O_CREATE, 0644) if err != nil { return nil, time.Time{}, err } if err := d.Dump(f); err != nil { f.Close() os.Remove(s.snapshotPath(tmpName)) return nil, time.Time{}, err } f.Close() if err := os.Rename(s.snapshotPath(tmpName), s.snapshotPath(name)); err != nil { return nil, time.Time{}, err } if f, err = os.Open(s.snapshotPath(name)); err != nil { return nil, time.Time{}, err } s.names = append(s.names, name) return &snapshot{f: f}, now, nil } func (s *snapshotStore) OpenLatest() (*snapshot, time.Time, error) { s.Lock() defer s.Unlock() if len(s.names) == 0 { return nil, time.Time{}, nil } name := s.names[len(s.names)-1] t, _ := parseSnapshotName(name) f, err := os.Open(s.snapshotPath(name)) if err != nil { return nil, time.Time{}, err } return &snapshot{f: f}, t, err }