snapshotter.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. /*
  2. Copyright 2014 CoreOS, Inc.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package snap
  14. import (
  15. "errors"
  16. "fmt"
  17. "hash/crc32"
  18. "io/ioutil"
  19. "log"
  20. "os"
  21. "path"
  22. "sort"
  23. "strings"
  24. "github.com/coreos/etcd/pkg/pbutil"
  25. "github.com/coreos/etcd/raft"
  26. "github.com/coreos/etcd/raft/raftpb"
  27. "github.com/coreos/etcd/snap/snappb"
  28. )
  29. const (
  30. snapSuffix = ".snap"
  31. )
  32. var (
  33. ErrNoSnapshot = errors.New("snap: no available snapshot")
  34. ErrEmptySnapshot = errors.New("snap: empty snapshot")
  35. ErrCRCMismatch = errors.New("snap: crc mismatch")
  36. crcTable = crc32.MakeTable(crc32.Castagnoli)
  37. )
  38. type Snapshotter struct {
  39. dir string
  40. }
  41. func New(dir string) *Snapshotter {
  42. return &Snapshotter{
  43. dir: dir,
  44. }
  45. }
  46. func (s *Snapshotter) SaveSnap(snapshot raftpb.Snapshot) error {
  47. if raft.IsEmptySnap(snapshot) {
  48. return nil
  49. }
  50. return s.save(&snapshot)
  51. }
  52. func (s *Snapshotter) save(snapshot *raftpb.Snapshot) error {
  53. fname := fmt.Sprintf("%016x-%016x%s", snapshot.Metadata.Term, snapshot.Metadata.Index, snapSuffix)
  54. b := pbutil.MustMarshal(snapshot)
  55. crc := crc32.Update(0, crcTable, b)
  56. snap := snappb.Snapshot{Crc: crc, Data: b}
  57. d, err := snap.Marshal()
  58. if err != nil {
  59. return err
  60. }
  61. return ioutil.WriteFile(path.Join(s.dir, fname), d, 0666)
  62. }
  63. func (s *Snapshotter) Load() (*raftpb.Snapshot, error) {
  64. names, err := s.snapNames()
  65. if err != nil {
  66. return nil, err
  67. }
  68. var snap *raftpb.Snapshot
  69. for _, name := range names {
  70. if snap, err = loadSnap(s.dir, name); err == nil {
  71. break
  72. }
  73. }
  74. return snap, err
  75. }
  76. func loadSnap(dir, name string) (*raftpb.Snapshot, error) {
  77. var err error
  78. var b []byte
  79. fpath := path.Join(dir, name)
  80. defer func() {
  81. if err != nil {
  82. renameBroken(fpath)
  83. }
  84. }()
  85. b, err = ioutil.ReadFile(fpath)
  86. if err != nil {
  87. log.Printf("snap: snapshotter cannot read file %v: %v", name, err)
  88. return nil, err
  89. }
  90. var serializedSnap snappb.Snapshot
  91. if err = serializedSnap.Unmarshal(b); err != nil {
  92. log.Printf("snap: corrupted snapshot file %v: %v", name, err)
  93. return nil, err
  94. }
  95. if len(serializedSnap.Data) == 0 || serializedSnap.Crc == 0 {
  96. log.Printf("snap: unexpected empty snapshot")
  97. return nil, ErrEmptySnapshot
  98. }
  99. crc := crc32.Update(0, crcTable, serializedSnap.Data)
  100. if crc != serializedSnap.Crc {
  101. log.Printf("snap: corrupted snapshot file %v: crc mismatch", name)
  102. return nil, ErrCRCMismatch
  103. }
  104. var snap raftpb.Snapshot
  105. if err = snap.Unmarshal(serializedSnap.Data); err != nil {
  106. log.Printf("snap: corrupted snapshot file %v: %v", name, err)
  107. return nil, err
  108. }
  109. return &snap, nil
  110. }
  111. // snapNames returns the filename of the snapshots in logical time order (from newest to oldest).
  112. // If there is no available snapshots, an ErrNoSnapshot will be returned.
  113. func (s *Snapshotter) snapNames() ([]string, error) {
  114. dir, err := os.Open(s.dir)
  115. if err != nil {
  116. return nil, err
  117. }
  118. defer dir.Close()
  119. names, err := dir.Readdirnames(-1)
  120. if err != nil {
  121. return nil, err
  122. }
  123. snaps := checkSuffix(names)
  124. if len(snaps) == 0 {
  125. return nil, ErrNoSnapshot
  126. }
  127. sort.Sort(sort.Reverse(sort.StringSlice(snaps)))
  128. return snaps, nil
  129. }
  130. func checkSuffix(names []string) []string {
  131. snaps := []string{}
  132. for i := range names {
  133. if strings.HasSuffix(names[i], snapSuffix) {
  134. snaps = append(snaps, names[i])
  135. } else {
  136. log.Printf("snap: unexpected non-snap file %v", names[i])
  137. }
  138. }
  139. return snaps
  140. }
  141. func renameBroken(path string) {
  142. brokenPath := path + ".broken"
  143. if err := os.Rename(path, brokenPath); err != nil {
  144. log.Printf("snap: cannot rename broken snapshot file %v to %v: %v", path, brokenPath, err)
  145. }
  146. }