instance.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596
  1. // Copyright (C) 2014 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at https://mozilla.org/MPL/2.0/.
  6. package db
  7. import (
  8. "bytes"
  9. "fmt"
  10. "github.com/syncthing/syncthing/lib/protocol"
  11. "github.com/syndtr/goleveldb/leveldb"
  12. "github.com/syndtr/goleveldb/leveldb/iterator"
  13. "github.com/syndtr/goleveldb/leveldb/util"
  14. )
  15. type deletionHandler func(t readWriteTransaction, folder, device, name []byte, dbi iterator.Iterator)
  16. type instance struct {
  17. *Lowlevel
  18. keyer keyer
  19. }
  20. func newInstance(ll *Lowlevel) *instance {
  21. return &instance{
  22. Lowlevel: ll,
  23. keyer: newDefaultKeyer(ll.folderIdx, ll.deviceIdx),
  24. }
  25. }
  26. func (db *instance) updateFiles(folder, device []byte, fs []protocol.FileInfo, meta *metadataTracker) {
  27. t := db.newReadWriteTransaction()
  28. defer t.close()
  29. var fk []byte
  30. var gk []byte
  31. for _, f := range fs {
  32. name := []byte(f.Name)
  33. fk = db.keyer.GenerateDeviceFileKey(fk, folder, device, name)
  34. // Get and unmarshal the file entry. If it doesn't exist or can't be
  35. // unmarshalled we'll add it as a new entry.
  36. bs, err := t.Get(fk, nil)
  37. var ef FileInfoTruncated
  38. if err == nil {
  39. err = ef.Unmarshal(bs)
  40. }
  41. // Local flags or the invalid bit might change without the version
  42. // being bumped. The IsInvalid() method handles both.
  43. if err == nil && ef.Version.Equal(f.Version) && ef.IsInvalid() == f.IsInvalid() {
  44. continue
  45. }
  46. devID := protocol.DeviceIDFromBytes(device)
  47. if err == nil {
  48. meta.removeFile(devID, ef)
  49. }
  50. meta.addFile(devID, f)
  51. t.insertFile(fk, folder, device, f)
  52. gk = db.keyer.GenerateGlobalVersionKey(gk, folder, name)
  53. t.updateGlobal(gk, folder, device, f, meta)
  54. // Write out and reuse the batch every few records, to avoid the batch
  55. // growing too large and thus allocating unnecessarily much memory.
  56. t.checkFlush()
  57. }
  58. }
  59. func (db *instance) addSequences(folder []byte, fs []protocol.FileInfo) {
  60. t := db.newReadWriteTransaction()
  61. defer t.close()
  62. var sk []byte
  63. var dk []byte
  64. for _, f := range fs {
  65. sk = db.keyer.GenerateSequenceKey(sk, folder, f.Sequence)
  66. dk = db.keyer.GenerateDeviceFileKey(dk, folder, protocol.LocalDeviceID[:], []byte(f.Name))
  67. t.Put(sk, dk)
  68. l.Debugf("adding sequence; folder=%q sequence=%v %v", folder, f.Sequence, f.Name)
  69. t.checkFlush()
  70. }
  71. }
  72. func (db *instance) removeSequences(folder []byte, fs []protocol.FileInfo) {
  73. t := db.newReadWriteTransaction()
  74. defer t.close()
  75. var sk []byte
  76. for _, f := range fs {
  77. t.Delete(db.keyer.GenerateSequenceKey(sk, folder, f.Sequence))
  78. l.Debugf("removing sequence; folder=%q sequence=%v %v", folder, f.Sequence, f.Name)
  79. t.checkFlush()
  80. }
  81. }
  82. func (db *instance) withHave(folder, device, prefix []byte, truncate bool, fn Iterator) {
  83. if len(prefix) > 0 {
  84. unslashedPrefix := prefix
  85. if bytes.HasSuffix(prefix, []byte{'/'}) {
  86. unslashedPrefix = unslashedPrefix[:len(unslashedPrefix)-1]
  87. } else {
  88. prefix = append(prefix, '/')
  89. }
  90. if f, ok := db.getFileTrunc(db.keyer.GenerateDeviceFileKey(nil, folder, device, unslashedPrefix), true); ok && !fn(f) {
  91. return
  92. }
  93. }
  94. t := db.newReadOnlyTransaction()
  95. defer t.close()
  96. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateDeviceFileKey(nil, folder, device, prefix)), nil)
  97. defer dbi.Release()
  98. for dbi.Next() {
  99. name := db.keyer.NameFromDeviceFileKey(dbi.Key())
  100. if len(prefix) > 0 && !bytes.HasPrefix(name, prefix) {
  101. return
  102. }
  103. // The iterator function may keep a reference to the unmarshalled
  104. // struct, which in turn references the buffer it was unmarshalled
  105. // from. dbi.Value() just returns an internal slice that it reuses, so
  106. // we need to copy it.
  107. f, err := unmarshalTrunc(append([]byte{}, dbi.Value()...), truncate)
  108. if err != nil {
  109. l.Debugln("unmarshal error:", err)
  110. continue
  111. }
  112. if !fn(f) {
  113. return
  114. }
  115. }
  116. }
  117. func (db *instance) withHaveSequence(folder []byte, startSeq int64, fn Iterator) {
  118. t := db.newReadOnlyTransaction()
  119. defer t.close()
  120. dbi := t.NewIterator(&util.Range{Start: db.keyer.GenerateSequenceKey(nil, folder, startSeq), Limit: db.keyer.GenerateSequenceKey(nil, folder, maxInt64)}, nil)
  121. defer dbi.Release()
  122. for dbi.Next() {
  123. f, ok := db.getFile(dbi.Value())
  124. if !ok {
  125. l.Debugln("missing file for sequence number", db.keyer.SequenceFromSequenceKey(dbi.Key()))
  126. continue
  127. }
  128. if shouldDebug() {
  129. if seq := db.keyer.SequenceFromSequenceKey(dbi.Key()); f.Sequence != seq {
  130. panic(fmt.Sprintf("sequence index corruption (folder %v, file %v): sequence %d != expected %d", string(folder), f.Name, f.Sequence, seq))
  131. }
  132. }
  133. if !fn(f) {
  134. return
  135. }
  136. }
  137. }
  138. func (db *instance) withAllFolderTruncated(folder []byte, fn func(device []byte, f FileInfoTruncated) bool) {
  139. t := db.newReadWriteTransaction()
  140. defer t.close()
  141. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateDeviceFileKey(nil, folder, nil, nil).WithoutNameAndDevice()), nil)
  142. defer dbi.Release()
  143. var gk []byte
  144. for dbi.Next() {
  145. device, ok := db.keyer.DeviceFromDeviceFileKey(dbi.Key())
  146. if !ok {
  147. // Not having the device in the index is bad. Clear it.
  148. t.Delete(dbi.Key())
  149. t.checkFlush()
  150. continue
  151. }
  152. var f FileInfoTruncated
  153. // The iterator function may keep a reference to the unmarshalled
  154. // struct, which in turn references the buffer it was unmarshalled
  155. // from. dbi.Value() just returns an internal slice that it reuses, so
  156. // we need to copy it.
  157. err := f.Unmarshal(append([]byte{}, dbi.Value()...))
  158. if err != nil {
  159. l.Debugln("unmarshal error:", err)
  160. continue
  161. }
  162. switch f.Name {
  163. case "", ".", "..", "/": // A few obviously invalid filenames
  164. l.Infof("Dropping invalid filename %q from database", f.Name)
  165. name := []byte(f.Name)
  166. gk = db.keyer.GenerateGlobalVersionKey(gk, folder, name)
  167. t.removeFromGlobal(gk, folder, device, name, nil)
  168. t.Delete(dbi.Key())
  169. t.checkFlush()
  170. continue
  171. }
  172. if !fn(device, f) {
  173. return
  174. }
  175. }
  176. }
  177. func (db *instance) getFile(key []byte) (protocol.FileInfo, bool) {
  178. if f, ok := db.getFileTrunc(key, false); ok {
  179. return f.(protocol.FileInfo), true
  180. }
  181. return protocol.FileInfo{}, false
  182. }
  183. func (db *instance) getFileTrunc(key []byte, trunc bool) (FileIntf, bool) {
  184. bs, err := db.Get(key, nil)
  185. if err == leveldb.ErrNotFound {
  186. return nil, false
  187. }
  188. if err != nil {
  189. l.Debugln("surprise error:", err)
  190. return nil, false
  191. }
  192. f, err := unmarshalTrunc(bs, trunc)
  193. if err != nil {
  194. l.Debugln("unmarshal error:", err)
  195. return nil, false
  196. }
  197. return f, true
  198. }
  199. func (db *instance) getGlobal(folder, file []byte, truncate bool) (FileIntf, bool) {
  200. t := db.newReadOnlyTransaction()
  201. defer t.close()
  202. _, _, f, ok := db.getGlobalInto(t, nil, nil, folder, file, truncate)
  203. return f, ok
  204. }
  205. func (db *instance) getGlobalInto(t readOnlyTransaction, gk, dk, folder, file []byte, truncate bool) ([]byte, []byte, FileIntf, bool) {
  206. gk = db.keyer.GenerateGlobalVersionKey(gk, folder, file)
  207. bs, err := t.Get(gk, nil)
  208. if err != nil {
  209. return gk, dk, nil, false
  210. }
  211. vl, ok := unmarshalVersionList(bs)
  212. if !ok {
  213. return gk, dk, nil, false
  214. }
  215. dk = db.keyer.GenerateDeviceFileKey(dk, folder, vl.Versions[0].Device, file)
  216. if fi, ok := db.getFileTrunc(dk, truncate); ok {
  217. return gk, dk, fi, true
  218. }
  219. return gk, dk, nil, false
  220. }
  221. func (db *instance) withGlobal(folder, prefix []byte, truncate bool, fn Iterator) {
  222. if len(prefix) > 0 {
  223. unslashedPrefix := prefix
  224. if bytes.HasSuffix(prefix, []byte{'/'}) {
  225. unslashedPrefix = unslashedPrefix[:len(unslashedPrefix)-1]
  226. } else {
  227. prefix = append(prefix, '/')
  228. }
  229. if f, ok := db.getGlobal(folder, unslashedPrefix, truncate); ok && !fn(f) {
  230. return
  231. }
  232. }
  233. t := db.newReadOnlyTransaction()
  234. defer t.close()
  235. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateGlobalVersionKey(nil, folder, prefix)), nil)
  236. defer dbi.Release()
  237. var fk []byte
  238. for dbi.Next() {
  239. name := db.keyer.NameFromGlobalVersionKey(dbi.Key())
  240. if len(prefix) > 0 && !bytes.HasPrefix(name, prefix) {
  241. return
  242. }
  243. vl, ok := unmarshalVersionList(dbi.Value())
  244. if !ok {
  245. continue
  246. }
  247. fk = db.keyer.GenerateDeviceFileKey(fk, folder, vl.Versions[0].Device, name)
  248. f, ok := db.getFileTrunc(fk, truncate)
  249. if !ok {
  250. continue
  251. }
  252. if !fn(f) {
  253. return
  254. }
  255. }
  256. }
  257. func (db *instance) availability(folder, file []byte) []protocol.DeviceID {
  258. k := db.keyer.GenerateGlobalVersionKey(nil, folder, file)
  259. bs, err := db.Get(k, nil)
  260. if err == leveldb.ErrNotFound {
  261. return nil
  262. }
  263. if err != nil {
  264. l.Debugln("surprise error:", err)
  265. return nil
  266. }
  267. vl, ok := unmarshalVersionList(bs)
  268. if !ok {
  269. return nil
  270. }
  271. var devices []protocol.DeviceID
  272. for _, v := range vl.Versions {
  273. if !v.Version.Equal(vl.Versions[0].Version) {
  274. break
  275. }
  276. if v.Invalid {
  277. continue
  278. }
  279. n := protocol.DeviceIDFromBytes(v.Device)
  280. devices = append(devices, n)
  281. }
  282. return devices
  283. }
  284. func (db *instance) withNeed(folder, device []byte, truncate bool, fn Iterator) {
  285. if bytes.Equal(device, protocol.LocalDeviceID[:]) {
  286. db.withNeedLocal(folder, truncate, fn)
  287. return
  288. }
  289. t := db.newReadOnlyTransaction()
  290. defer t.close()
  291. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateGlobalVersionKey(nil, folder, nil).WithoutName()), nil)
  292. defer dbi.Release()
  293. var fk []byte
  294. for dbi.Next() {
  295. vl, ok := unmarshalVersionList(dbi.Value())
  296. if !ok {
  297. continue
  298. }
  299. haveFV, have := vl.Get(device)
  300. // XXX: This marks Concurrent (i.e. conflicting) changes as
  301. // needs. Maybe we should do that, but it needs special
  302. // handling in the puller.
  303. if have && haveFV.Version.GreaterEqual(vl.Versions[0].Version) {
  304. continue
  305. }
  306. name := db.keyer.NameFromGlobalVersionKey(dbi.Key())
  307. needVersion := vl.Versions[0].Version
  308. needDevice := protocol.DeviceIDFromBytes(vl.Versions[0].Device)
  309. for i := range vl.Versions {
  310. if !vl.Versions[i].Version.Equal(needVersion) {
  311. // We haven't found a valid copy of the file with the needed version.
  312. break
  313. }
  314. if vl.Versions[i].Invalid {
  315. // The file is marked invalid, don't use it.
  316. continue
  317. }
  318. fk = db.keyer.GenerateDeviceFileKey(fk, folder, vl.Versions[i].Device, name)
  319. bs, err := t.Get(fk, nil)
  320. if err != nil {
  321. l.Debugln("surprise error:", err)
  322. continue
  323. }
  324. gf, err := unmarshalTrunc(bs, truncate)
  325. if err != nil {
  326. l.Debugln("unmarshal error:", err)
  327. continue
  328. }
  329. if gf.IsDeleted() && !have {
  330. // We don't need deleted files that we don't have
  331. break
  332. }
  333. l.Debugf("need folder=%q device=%v name=%q have=%v invalid=%v haveV=%v globalV=%v globalDev=%v", folder, protocol.DeviceIDFromBytes(device), name, have, haveFV.Invalid, haveFV.Version, needVersion, needDevice)
  334. if !fn(gf) {
  335. return
  336. }
  337. // This file is handled, no need to look further in the version list
  338. break
  339. }
  340. }
  341. }
  342. func (db *instance) withNeedLocal(folder []byte, truncate bool, fn Iterator) {
  343. t := db.newReadOnlyTransaction()
  344. defer t.close()
  345. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateNeedFileKey(nil, folder, nil).WithoutName()), nil)
  346. defer dbi.Release()
  347. var dk []byte
  348. var gk []byte
  349. var f FileIntf
  350. var ok bool
  351. for dbi.Next() {
  352. gk, dk, f, ok = db.getGlobalInto(t, gk, dk, folder, db.keyer.NameFromGlobalVersionKey(dbi.Key()), truncate)
  353. if !ok {
  354. continue
  355. }
  356. if !fn(f) {
  357. return
  358. }
  359. }
  360. }
  361. func (db *instance) dropFolder(folder []byte) {
  362. t := db.newReadWriteTransaction()
  363. defer t.close()
  364. for _, key := range [][]byte{
  365. // Remove all items related to the given folder from the device->file bucket
  366. db.keyer.GenerateDeviceFileKey(nil, folder, nil, nil).WithoutNameAndDevice(),
  367. // Remove all sequences related to the folder
  368. db.keyer.GenerateSequenceKey(nil, []byte(folder), 0).WithoutSequence(),
  369. // Remove all items related to the given folder from the global bucket
  370. db.keyer.GenerateGlobalVersionKey(nil, folder, nil).WithoutName(),
  371. // Remove all needs related to the folder
  372. db.keyer.GenerateNeedFileKey(nil, folder, nil).WithoutName(),
  373. } {
  374. t.deleteKeyPrefix(key)
  375. }
  376. }
  377. func (db *instance) dropDeviceFolder(device, folder []byte, meta *metadataTracker) {
  378. t := db.newReadWriteTransaction()
  379. defer t.close()
  380. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateDeviceFileKey(nil, folder, device, nil)), nil)
  381. defer dbi.Release()
  382. var gk []byte
  383. for dbi.Next() {
  384. key := dbi.Key()
  385. name := db.keyer.NameFromDeviceFileKey(key)
  386. gk = db.keyer.GenerateGlobalVersionKey(gk, folder, name)
  387. t.removeFromGlobal(gk, folder, device, name, meta)
  388. t.Delete(key)
  389. t.checkFlush()
  390. }
  391. }
  392. func (db *instance) checkGlobals(folder []byte, meta *metadataTracker) {
  393. t := db.newReadWriteTransaction()
  394. defer t.close()
  395. dbi := t.NewIterator(util.BytesPrefix(db.keyer.GenerateGlobalVersionKey(nil, folder, nil).WithoutName()), nil)
  396. defer dbi.Release()
  397. var fk []byte
  398. for dbi.Next() {
  399. vl, ok := unmarshalVersionList(dbi.Value())
  400. if !ok {
  401. continue
  402. }
  403. // Check the global version list for consistency. An issue in previous
  404. // versions of goleveldb could result in reordered writes so that
  405. // there are global entries pointing to no longer existing files. Here
  406. // we find those and clear them out.
  407. name := db.keyer.NameFromGlobalVersionKey(dbi.Key())
  408. var newVL VersionList
  409. for i, version := range vl.Versions {
  410. fk = db.keyer.GenerateDeviceFileKey(fk, folder, version.Device, name)
  411. _, err := t.Get(fk, nil)
  412. if err == leveldb.ErrNotFound {
  413. continue
  414. }
  415. if err != nil {
  416. l.Debugln("surprise error:", err)
  417. return
  418. }
  419. newVL.Versions = append(newVL.Versions, version)
  420. if i == 0 {
  421. if fi, ok := db.getFile(fk); ok {
  422. meta.addFile(protocol.GlobalDeviceID, fi)
  423. }
  424. }
  425. }
  426. if len(newVL.Versions) != len(vl.Versions) {
  427. t.Put(dbi.Key(), mustMarshal(&newVL))
  428. t.checkFlush()
  429. }
  430. }
  431. l.Debugf("db check completed for %q", folder)
  432. }
  433. func (db *instance) getIndexID(device, folder []byte) protocol.IndexID {
  434. key := db.keyer.GenerateIndexIDKey(nil, device, folder)
  435. cur, err := db.Get(key, nil)
  436. if err != nil {
  437. return 0
  438. }
  439. var id protocol.IndexID
  440. if err := id.Unmarshal(cur); err != nil {
  441. return 0
  442. }
  443. return id
  444. }
  445. func (db *instance) setIndexID(device, folder []byte, id protocol.IndexID) {
  446. key := db.keyer.GenerateIndexIDKey(nil, device, folder)
  447. bs, _ := id.Marshal() // marshalling can't fail
  448. if err := db.Put(key, bs, nil); err != nil {
  449. panic("storing index ID: " + err.Error())
  450. }
  451. }
  452. func (db *instance) dropMtimes(folder []byte) {
  453. db.dropPrefix(db.keyer.GenerateMtimesKey(nil, folder))
  454. }
  455. func (db *instance) dropFolderMeta(folder []byte) {
  456. db.dropPrefix(db.keyer.GenerateFolderMetaKey(nil, folder))
  457. }
  458. func (db *instance) dropPrefix(prefix []byte) {
  459. t := db.newReadWriteTransaction()
  460. defer t.close()
  461. dbi := t.NewIterator(util.BytesPrefix(prefix), nil)
  462. defer dbi.Release()
  463. for dbi.Next() {
  464. t.Delete(dbi.Key())
  465. }
  466. }
  467. func unmarshalTrunc(bs []byte, truncate bool) (FileIntf, error) {
  468. if truncate {
  469. var tf FileInfoTruncated
  470. err := tf.Unmarshal(bs)
  471. return tf, err
  472. }
  473. var tf protocol.FileInfo
  474. err := tf.Unmarshal(bs)
  475. return tf, err
  476. }
  477. func unmarshalVersionList(data []byte) (VersionList, bool) {
  478. var vl VersionList
  479. if err := vl.Unmarshal(data); err != nil {
  480. l.Debugln("unmarshal error:", err)
  481. return VersionList{}, false
  482. }
  483. if len(vl.Versions) == 0 {
  484. l.Debugln("empty version list")
  485. return VersionList{}, false
  486. }
  487. return vl, true
  488. }
  489. type errorSuggestion struct {
  490. inner error
  491. suggestion string
  492. }
  493. func (e errorSuggestion) Error() string {
  494. return fmt.Sprintf("%s (%s)", e.inner.Error(), e.suggestion)
  495. }