leveldb.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595
  1. package files
  2. import (
  3. "bytes"
  4. "sort"
  5. "sync"
  6. "github.com/calmh/syncthing/lamport"
  7. "github.com/calmh/syncthing/protocol"
  8. "github.com/syndtr/goleveldb/leveldb"
  9. "github.com/syndtr/goleveldb/leveldb/iterator"
  10. "github.com/syndtr/goleveldb/leveldb/opt"
  11. "github.com/syndtr/goleveldb/leveldb/util"
  12. )
  13. var (
  14. clockTick uint64
  15. clockMut sync.Mutex
  16. )
  17. func clock(v uint64) uint64 {
  18. clockMut.Lock()
  19. defer clockMut.Unlock()
  20. if v > clockTick {
  21. clockTick = v + 1
  22. } else {
  23. clockTick++
  24. }
  25. return clockTick
  26. }
  27. const (
  28. keyTypeNode = iota
  29. keyTypeGlobal
  30. )
  31. type fileVersion struct {
  32. version uint64
  33. node []byte
  34. }
  35. type versionList struct {
  36. versions []fileVersion
  37. }
  38. type fileList []protocol.FileInfo
  39. func (l fileList) Len() int {
  40. return len(l)
  41. }
  42. func (l fileList) Swap(a, b int) {
  43. l[a], l[b] = l[b], l[a]
  44. }
  45. func (l fileList) Less(a, b int) bool {
  46. return l[a].Name < l[b].Name
  47. }
  48. type dbReader interface {
  49. Get([]byte, *opt.ReadOptions) ([]byte, error)
  50. }
  51. type dbWriter interface {
  52. Put([]byte, []byte)
  53. Delete([]byte)
  54. }
  55. /*
  56. keyTypeNode (1 byte)
  57. repository (64 bytes)
  58. node (32 bytes)
  59. name (variable size)
  60. |
  61. scanner.File
  62. keyTypeGlobal (1 byte)
  63. repository (64 bytes)
  64. name (variable size)
  65. |
  66. []fileVersion (sorted)
  67. */
  68. func nodeKey(repo, node, file []byte) []byte {
  69. k := make([]byte, 1+64+32+len(file))
  70. k[0] = keyTypeNode
  71. copy(k[1:], []byte(repo))
  72. copy(k[1+64:], node[:])
  73. copy(k[1+64+32:], []byte(file))
  74. return k
  75. }
  76. func globalKey(repo, file []byte) []byte {
  77. k := make([]byte, 1+64+len(file))
  78. k[0] = keyTypeGlobal
  79. copy(k[1:], []byte(repo))
  80. copy(k[1+64:], []byte(file))
  81. return k
  82. }
  83. func nodeKeyName(key []byte) []byte {
  84. return key[1+64+32:]
  85. }
  86. func globalKeyName(key []byte) []byte {
  87. return key[1+64:]
  88. }
  89. type deletionHandler func(db dbReader, batch dbWriter, repo, node, name []byte, dbi iterator.Iterator) uint64
  90. type fileIterator func(f protocol.FileInfo) bool
  91. func ldbGenericReplace(db *leveldb.DB, repo, node []byte, fs []protocol.FileInfo, deleteFn deletionHandler) uint64 {
  92. sort.Sort(fileList(fs)) // sort list on name, same as on disk
  93. start := nodeKey(repo, node, nil) // before all repo/node files
  94. limit := nodeKey(repo, node, []byte{0xff, 0xff, 0xff, 0xff}) // after all repo/node files
  95. batch := new(leveldb.Batch)
  96. snap, err := db.GetSnapshot()
  97. if err != nil {
  98. panic(err)
  99. }
  100. defer snap.Release()
  101. dbi := snap.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
  102. defer dbi.Release()
  103. moreDb := dbi.Next()
  104. fsi := 0
  105. var maxLocalVer uint64
  106. for {
  107. var newName, oldName []byte
  108. moreFs := fsi < len(fs)
  109. if !moreDb && !moreFs {
  110. break
  111. }
  112. if !moreFs && deleteFn == nil {
  113. // We don't have any more updated files to process and deletion
  114. // has not been requested, so we can exit early
  115. break
  116. }
  117. if moreFs {
  118. newName = []byte(fs[fsi].Name)
  119. }
  120. if moreDb {
  121. oldName = nodeKeyName(dbi.Key())
  122. }
  123. cmp := bytes.Compare(newName, oldName)
  124. if debug {
  125. l.Debugf("generic replace; repo=%q node=%x moreFs=%v moreDb=%v cmp=%d newName=%q oldName=%q", repo, node, moreFs, moreDb, cmp, newName, oldName)
  126. }
  127. switch {
  128. case moreFs && (!moreDb || cmp == -1):
  129. // Disk is missing this file. Insert it.
  130. if lv := ldbInsert(batch, repo, node, newName, fs[fsi]); lv > maxLocalVer {
  131. maxLocalVer = lv
  132. }
  133. ldbUpdateGlobal(snap, batch, repo, node, newName, fs[fsi].Version)
  134. fsi++
  135. case moreFs && moreDb && cmp == 0:
  136. // File exists on both sides - compare versions.
  137. var ef protocol.FileInfo
  138. ef.UnmarshalXDR(dbi.Value())
  139. if fs[fsi].Version > ef.Version {
  140. if lv := ldbInsert(batch, repo, node, newName, fs[fsi]); lv > maxLocalVer {
  141. maxLocalVer = lv
  142. }
  143. ldbUpdateGlobal(snap, batch, repo, node, newName, fs[fsi].Version)
  144. }
  145. // Iterate both sides.
  146. fsi++
  147. moreDb = dbi.Next()
  148. case moreDb && (!moreFs || cmp == 1):
  149. if deleteFn != nil {
  150. if lv := deleteFn(snap, batch, repo, node, oldName, dbi); lv > maxLocalVer {
  151. maxLocalVer = lv
  152. }
  153. }
  154. moreDb = dbi.Next()
  155. }
  156. }
  157. err = db.Write(batch, nil)
  158. if err != nil {
  159. panic(err)
  160. }
  161. return maxLocalVer
  162. }
  163. func ldbReplace(db *leveldb.DB, repo, node []byte, fs []protocol.FileInfo) uint64 {
  164. // TODO: Return the remaining maxLocalVer?
  165. return ldbGenericReplace(db, repo, node, fs, func(db dbReader, batch dbWriter, repo, node, name []byte, dbi iterator.Iterator) uint64 {
  166. // Disk has files that we are missing. Remove it.
  167. if debug {
  168. l.Debugf("delete; repo=%q node=%x name=%q", repo, node, name)
  169. }
  170. ldbRemoveFromGlobal(db, batch, repo, node, name)
  171. batch.Delete(dbi.Key())
  172. return 0
  173. })
  174. }
  175. func ldbReplaceWithDelete(db *leveldb.DB, repo, node []byte, fs []protocol.FileInfo) uint64 {
  176. return ldbGenericReplace(db, repo, node, fs, func(db dbReader, batch dbWriter, repo, node, name []byte, dbi iterator.Iterator) uint64 {
  177. var f protocol.FileInfo
  178. err := f.UnmarshalXDR(dbi.Value())
  179. if err != nil {
  180. panic(err)
  181. }
  182. if !protocol.IsDeleted(f.Flags) {
  183. if debug {
  184. l.Debugf("mark deleted; repo=%q node=%x name=%q", repo, node, name)
  185. }
  186. ts := clock(f.LocalVersion)
  187. f.Blocks = nil
  188. f.Version = lamport.Default.Tick(f.Version)
  189. f.Flags |= protocol.FlagDeleted
  190. f.LocalVersion = ts
  191. batch.Put(dbi.Key(), f.MarshalXDR())
  192. ldbUpdateGlobal(db, batch, repo, node, nodeKeyName(dbi.Key()), f.Version)
  193. return ts
  194. }
  195. return 0
  196. })
  197. }
  198. func ldbUpdate(db *leveldb.DB, repo, node []byte, fs []protocol.FileInfo) uint64 {
  199. batch := new(leveldb.Batch)
  200. snap, err := db.GetSnapshot()
  201. if err != nil {
  202. panic(err)
  203. }
  204. defer snap.Release()
  205. var maxLocalVer uint64
  206. for _, f := range fs {
  207. name := []byte(f.Name)
  208. fk := nodeKey(repo, node, name)
  209. bs, err := snap.Get(fk, nil)
  210. if err == leveldb.ErrNotFound {
  211. if lv := ldbInsert(batch, repo, node, name, f); lv > maxLocalVer {
  212. maxLocalVer = lv
  213. }
  214. ldbUpdateGlobal(snap, batch, repo, node, name, f.Version)
  215. continue
  216. }
  217. var ef protocol.FileInfo
  218. err = ef.UnmarshalXDR(bs)
  219. if err != nil {
  220. panic(err)
  221. }
  222. if ef.Version != f.Version {
  223. if lv := ldbInsert(batch, repo, node, name, f); lv > maxLocalVer {
  224. maxLocalVer = lv
  225. }
  226. ldbUpdateGlobal(snap, batch, repo, node, name, f.Version)
  227. }
  228. }
  229. err = db.Write(batch, nil)
  230. if err != nil {
  231. panic(err)
  232. }
  233. return maxLocalVer
  234. }
  235. func ldbInsert(batch dbWriter, repo, node, name []byte, file protocol.FileInfo) uint64 {
  236. if debug {
  237. l.Debugf("insert; repo=%q node=%x %v", repo, node, file)
  238. }
  239. if file.LocalVersion == 0 {
  240. file.LocalVersion = clock(0)
  241. }
  242. nk := nodeKey(repo, node, name)
  243. batch.Put(nk, file.MarshalXDR())
  244. return file.LocalVersion
  245. }
  246. // ldbUpdateGlobal adds this node+version to the version list for the given
  247. // file. If the node is already present in the list, the version is updated.
  248. // If the file does not have an entry in the global list, it is created.
  249. func ldbUpdateGlobal(db dbReader, batch dbWriter, repo, node, file []byte, version uint64) bool {
  250. if debug {
  251. l.Debugf("update global; repo=%q node=%x file=%q version=%d", repo, node, file, version)
  252. }
  253. gk := globalKey(repo, file)
  254. svl, err := db.Get(gk, nil)
  255. if err != nil && err != leveldb.ErrNotFound {
  256. panic(err)
  257. }
  258. var fl versionList
  259. nv := fileVersion{
  260. node: node,
  261. version: version,
  262. }
  263. if svl != nil {
  264. err = fl.UnmarshalXDR(svl)
  265. if err != nil {
  266. panic(err)
  267. }
  268. for i := range fl.versions {
  269. if bytes.Compare(fl.versions[i].node, node) == 0 {
  270. if fl.versions[i].version == version {
  271. // No need to do anything
  272. return false
  273. }
  274. fl.versions = append(fl.versions[:i], fl.versions[i+1:]...)
  275. break
  276. }
  277. }
  278. }
  279. for i := range fl.versions {
  280. if fl.versions[i].version <= version {
  281. t := append(fl.versions, fileVersion{})
  282. copy(t[i+1:], t[i:])
  283. t[i] = nv
  284. fl.versions = t
  285. goto done
  286. }
  287. }
  288. fl.versions = append(fl.versions, nv)
  289. done:
  290. batch.Put(gk, fl.MarshalXDR())
  291. return true
  292. }
  293. // ldbRemoveFromGlobal removes the node from the global version list for the
  294. // given file. If the version list is empty after this, the file entry is
  295. // removed entirely.
  296. func ldbRemoveFromGlobal(db dbReader, batch dbWriter, repo, node, file []byte) {
  297. if debug {
  298. l.Debugf("remove from global; repo=%q node=%x file=%q", repo, node, file)
  299. }
  300. gk := globalKey(repo, file)
  301. svl, err := db.Get(gk, nil)
  302. if err != nil {
  303. panic(err)
  304. }
  305. var fl versionList
  306. err = fl.UnmarshalXDR(svl)
  307. if err != nil {
  308. panic(err)
  309. }
  310. for i := range fl.versions {
  311. if bytes.Compare(fl.versions[i].node, node) == 0 {
  312. fl.versions = append(fl.versions[:i], fl.versions[i+1:]...)
  313. break
  314. }
  315. }
  316. if len(fl.versions) == 0 {
  317. batch.Delete(gk)
  318. } else {
  319. batch.Put(gk, fl.MarshalXDR())
  320. }
  321. }
  322. func ldbWithHave(db *leveldb.DB, repo, node []byte, fn fileIterator) {
  323. start := nodeKey(repo, node, nil) // before all repo/node files
  324. limit := nodeKey(repo, node, []byte{0xff, 0xff, 0xff, 0xff}) // after all repo/node files
  325. snap, err := db.GetSnapshot()
  326. if err != nil {
  327. panic(err)
  328. }
  329. defer snap.Release()
  330. dbi := snap.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
  331. defer dbi.Release()
  332. for dbi.Next() {
  333. var f protocol.FileInfo
  334. err := f.UnmarshalXDR(dbi.Value())
  335. if err != nil {
  336. panic(err)
  337. }
  338. if cont := fn(f); !cont {
  339. return
  340. }
  341. }
  342. }
  343. func ldbGet(db *leveldb.DB, repo, node, file []byte) protocol.FileInfo {
  344. nk := nodeKey(repo, node, file)
  345. bs, err := db.Get(nk, nil)
  346. if err == leveldb.ErrNotFound {
  347. return protocol.FileInfo{}
  348. }
  349. if err != nil {
  350. panic(err)
  351. }
  352. var f protocol.FileInfo
  353. err = f.UnmarshalXDR(bs)
  354. if err != nil {
  355. panic(err)
  356. }
  357. return f
  358. }
  359. func ldbGetGlobal(db *leveldb.DB, repo, file []byte) protocol.FileInfo {
  360. k := globalKey(repo, file)
  361. snap, err := db.GetSnapshot()
  362. if err != nil {
  363. panic(err)
  364. }
  365. defer snap.Release()
  366. bs, err := snap.Get(k, nil)
  367. if err == leveldb.ErrNotFound {
  368. return protocol.FileInfo{}
  369. }
  370. if err != nil {
  371. panic(err)
  372. }
  373. var vl versionList
  374. err = vl.UnmarshalXDR(bs)
  375. if err != nil {
  376. panic(err)
  377. }
  378. if len(vl.versions) == 0 {
  379. l.Debugln(k)
  380. panic("no versions?")
  381. }
  382. k = nodeKey(repo, vl.versions[0].node, file)
  383. bs, err = snap.Get(k, nil)
  384. if err != nil {
  385. panic(err)
  386. }
  387. var f protocol.FileInfo
  388. err = f.UnmarshalXDR(bs)
  389. if err != nil {
  390. panic(err)
  391. }
  392. return f
  393. }
  394. func ldbWithGlobal(db *leveldb.DB, repo []byte, fn fileIterator) {
  395. start := globalKey(repo, nil)
  396. limit := globalKey(repo, []byte{0xff, 0xff, 0xff, 0xff})
  397. snap, err := db.GetSnapshot()
  398. if err != nil {
  399. panic(err)
  400. }
  401. defer snap.Release()
  402. dbi := snap.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
  403. defer dbi.Release()
  404. for dbi.Next() {
  405. var vl versionList
  406. err := vl.UnmarshalXDR(dbi.Value())
  407. if err != nil {
  408. panic(err)
  409. }
  410. if len(vl.versions) == 0 {
  411. l.Debugln(dbi.Key())
  412. panic("no versions?")
  413. }
  414. fk := nodeKey(repo, vl.versions[0].node, globalKeyName(dbi.Key()))
  415. bs, err := snap.Get(fk, nil)
  416. if err != nil {
  417. panic(err)
  418. }
  419. var f protocol.FileInfo
  420. err = f.UnmarshalXDR(bs)
  421. if err != nil {
  422. panic(err)
  423. }
  424. if cont := fn(f); !cont {
  425. return
  426. }
  427. }
  428. }
  429. func ldbAvailability(db *leveldb.DB, repo, file []byte) []protocol.NodeID {
  430. k := globalKey(repo, file)
  431. bs, err := db.Get(k, nil)
  432. if err == leveldb.ErrNotFound {
  433. return nil
  434. }
  435. if err != nil {
  436. panic(err)
  437. }
  438. var vl versionList
  439. err = vl.UnmarshalXDR(bs)
  440. if err != nil {
  441. panic(err)
  442. }
  443. var nodes []protocol.NodeID
  444. for _, v := range vl.versions {
  445. if v.version != vl.versions[0].version {
  446. break
  447. }
  448. var n protocol.NodeID
  449. copy(n[:], v.node)
  450. nodes = append(nodes, n)
  451. }
  452. return nodes
  453. }
  454. func ldbWithNeed(db *leveldb.DB, repo, node []byte, fn fileIterator) {
  455. start := globalKey(repo, nil)
  456. limit := globalKey(repo, []byte{0xff, 0xff, 0xff, 0xff})
  457. snap, err := db.GetSnapshot()
  458. if err != nil {
  459. panic(err)
  460. }
  461. defer snap.Release()
  462. dbi := snap.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
  463. defer dbi.Release()
  464. for dbi.Next() {
  465. var vl versionList
  466. err := vl.UnmarshalXDR(dbi.Value())
  467. if err != nil {
  468. panic(err)
  469. }
  470. if len(vl.versions) == 0 {
  471. l.Debugln(dbi.Key())
  472. panic("no versions?")
  473. }
  474. have := false // If we have the file, any version
  475. need := false // If we have a lower version of the file
  476. var haveVersion uint64
  477. for _, v := range vl.versions {
  478. if bytes.Compare(v.node, node) == 0 {
  479. have = true
  480. haveVersion = v.version
  481. need = v.version < vl.versions[0].version
  482. break
  483. }
  484. }
  485. if need || !have {
  486. name := globalKeyName(dbi.Key())
  487. if debug {
  488. l.Debugf("need repo=%q node=%x name=%q need=%v have=%v haveV=%d globalV=%d", repo, node, name, need, have, haveVersion, vl.versions[0].version)
  489. }
  490. fk := nodeKey(repo, vl.versions[0].node, name)
  491. bs, err := snap.Get(fk, nil)
  492. if err != nil {
  493. panic(err)
  494. }
  495. var gf protocol.FileInfo
  496. err = gf.UnmarshalXDR(bs)
  497. if err != nil {
  498. panic(err)
  499. }
  500. if protocol.IsDeleted(gf.Flags) && !have {
  501. // We don't need deleted files that we don't have
  502. continue
  503. }
  504. if cont := fn(gf); !cont {
  505. return
  506. }
  507. }
  508. }
  509. }