leveldb_dbinstance.go 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889
  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. "encoding/binary"
  10. "os"
  11. "sort"
  12. "strings"
  13. "sync/atomic"
  14. "github.com/syncthing/syncthing/lib/protocol"
  15. "github.com/syncthing/syncthing/lib/sync"
  16. "github.com/syndtr/goleveldb/leveldb"
  17. "github.com/syndtr/goleveldb/leveldb/errors"
  18. "github.com/syndtr/goleveldb/leveldb/iterator"
  19. "github.com/syndtr/goleveldb/leveldb/opt"
  20. "github.com/syndtr/goleveldb/leveldb/storage"
  21. "github.com/syndtr/goleveldb/leveldb/util"
  22. )
  23. type deletionHandler func(t readWriteTransaction, folder, device, name []byte, dbi iterator.Iterator)
  24. type Instance struct {
  25. committed int64 // this must be the first attribute in the struct to ensure 64 bit alignment on 32 bit plaforms
  26. *leveldb.DB
  27. location string
  28. folderIdx *smallIndex
  29. deviceIdx *smallIndex
  30. }
  31. const (
  32. keyPrefixLen = 1
  33. keyFolderLen = 4 // indexed
  34. keyDeviceLen = 4 // indexed
  35. keyHashLen = 32
  36. )
  37. func Open(file string) (*Instance, error) {
  38. opts := &opt.Options{
  39. OpenFilesCacheCapacity: 100,
  40. WriteBuffer: 4 << 20,
  41. }
  42. db, err := leveldb.OpenFile(file, opts)
  43. if leveldbIsCorrupted(err) {
  44. db, err = leveldb.RecoverFile(file, opts)
  45. }
  46. if leveldbIsCorrupted(err) {
  47. // The database is corrupted, and we've tried to recover it but it
  48. // didn't work. At this point there isn't much to do beyond dropping
  49. // the database and reindexing...
  50. l.Infoln("Database corruption detected, unable to recover. Reinitializing...")
  51. if err := os.RemoveAll(file); err != nil {
  52. return nil, err
  53. }
  54. db, err = leveldb.OpenFile(file, opts)
  55. }
  56. if err != nil {
  57. return nil, err
  58. }
  59. return newDBInstance(db, file), nil
  60. }
  61. func OpenMemory() *Instance {
  62. db, _ := leveldb.Open(storage.NewMemStorage(), nil)
  63. return newDBInstance(db, "<memory>")
  64. }
  65. func newDBInstance(db *leveldb.DB, location string) *Instance {
  66. i := &Instance{
  67. DB: db,
  68. location: location,
  69. }
  70. i.folderIdx = newSmallIndex(i, []byte{KeyTypeFolderIdx})
  71. i.deviceIdx = newSmallIndex(i, []byte{KeyTypeDeviceIdx})
  72. return i
  73. }
  74. // UpdateSchema does transitions to the current db version if necessary
  75. func (db *Instance) UpdateSchema() {
  76. miscDB := NewNamespacedKV(db, string(KeyTypeMiscData))
  77. prevVersion, _ := miscDB.Int64("dbVersion")
  78. if prevVersion == 0 {
  79. db.updateSchema0to1()
  80. }
  81. if prevVersion != dbVersion {
  82. miscDB.PutInt64("dbVersion", dbVersion)
  83. }
  84. }
  85. // Committed returns the number of items committed to the database since startup
  86. func (db *Instance) Committed() int64 {
  87. return atomic.LoadInt64(&db.committed)
  88. }
  89. // Location returns the filesystem path where the database is stored
  90. func (db *Instance) Location() string {
  91. return db.location
  92. }
  93. func (db *Instance) updateFiles(folder, device []byte, fs []protocol.FileInfo, meta *metadataTracker) {
  94. t := db.newReadWriteTransaction()
  95. defer t.close()
  96. var fk []byte
  97. for _, f := range fs {
  98. name := []byte(f.Name)
  99. fk = db.deviceKeyInto(fk[:cap(fk)], folder, device, name)
  100. // Get and unmarshal the file entry. If it doesn't exist or can't be
  101. // unmarshalled we'll add it as a new entry.
  102. bs, err := t.Get(fk, nil)
  103. var ef FileInfoTruncated
  104. if err == nil {
  105. err = ef.Unmarshal(bs)
  106. }
  107. // The Invalid flag might change without the version being bumped.
  108. if err == nil && ef.Version.Equal(f.Version) && ef.Invalid == f.Invalid {
  109. continue
  110. }
  111. devID := protocol.DeviceIDFromBytes(device)
  112. if err == nil {
  113. meta.removeFile(devID, ef)
  114. }
  115. meta.addFile(devID, f)
  116. t.insertFile(folder, device, f)
  117. t.updateGlobal(folder, device, f, meta)
  118. // Write out and reuse the batch every few records, to avoid the batch
  119. // growing too large and thus allocating unnecessarily much memory.
  120. t.checkFlush()
  121. }
  122. }
  123. func (db *Instance) withHave(folder, device, prefix []byte, truncate bool, fn Iterator) {
  124. t := db.newReadOnlyTransaction()
  125. defer t.close()
  126. dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, device, prefix)[:keyPrefixLen+keyFolderLen+keyDeviceLen+len(prefix)]), nil)
  127. defer dbi.Release()
  128. slashedPrefix := prefix
  129. if !bytes.HasSuffix(prefix, []byte{'/'}) {
  130. slashedPrefix = append(slashedPrefix, '/')
  131. }
  132. for dbi.Next() {
  133. name := db.deviceKeyName(dbi.Key())
  134. if len(prefix) > 0 && !bytes.Equal(name, prefix) && !bytes.HasPrefix(name, slashedPrefix) {
  135. return
  136. }
  137. // The iterator function may keep a reference to the unmarshalled
  138. // struct, which in turn references the buffer it was unmarshalled
  139. // from. dbi.Value() just returns an internal slice that it reuses, so
  140. // we need to copy it.
  141. f, err := unmarshalTrunc(append([]byte{}, dbi.Value()...), truncate)
  142. if err != nil {
  143. l.Debugln("unmarshal error:", err)
  144. continue
  145. }
  146. if cont := fn(f); !cont {
  147. return
  148. }
  149. }
  150. }
  151. func (db *Instance) withAllFolderTruncated(folder []byte, fn func(device []byte, f FileInfoTruncated) bool) {
  152. t := db.newReadWriteTransaction()
  153. defer t.close()
  154. dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, nil, nil)[:keyPrefixLen+keyFolderLen]), nil)
  155. defer dbi.Release()
  156. for dbi.Next() {
  157. device := db.deviceKeyDevice(dbi.Key())
  158. var f FileInfoTruncated
  159. // The iterator function may keep a reference to the unmarshalled
  160. // struct, which in turn references the buffer it was unmarshalled
  161. // from. dbi.Value() just returns an internal slice that it reuses, so
  162. // we need to copy it.
  163. err := f.Unmarshal(append([]byte{}, dbi.Value()...))
  164. if err != nil {
  165. l.Debugln("unmarshal error:", err)
  166. continue
  167. }
  168. switch f.Name {
  169. case "", ".", "..", "/": // A few obviously invalid filenames
  170. l.Infof("Dropping invalid filename %q from database", f.Name)
  171. t.removeFromGlobal(folder, device, nil, nil)
  172. t.Delete(dbi.Key())
  173. t.checkFlush()
  174. continue
  175. }
  176. if cont := fn(device, f); !cont {
  177. return
  178. }
  179. }
  180. }
  181. func (db *Instance) getFile(folder, device, file []byte) (protocol.FileInfo, bool) {
  182. return getFile(db, db.deviceKey(folder, device, file))
  183. }
  184. func (db *Instance) getGlobal(folder, file []byte, truncate bool) (FileIntf, bool) {
  185. k := db.globalKey(folder, file)
  186. t := db.newReadOnlyTransaction()
  187. defer t.close()
  188. bs, err := t.Get(k, nil)
  189. if err != nil {
  190. return nil, false
  191. }
  192. var vl VersionList
  193. err = vl.Unmarshal(bs)
  194. if err == leveldb.ErrNotFound {
  195. return nil, false
  196. }
  197. if err != nil {
  198. l.Debugln("unmarshal error:", k, err)
  199. return nil, false
  200. }
  201. if len(vl.Versions) == 0 {
  202. l.Debugln("no versions:", k)
  203. return nil, false
  204. }
  205. k = db.deviceKey(folder, vl.Versions[0].Device, file)
  206. bs, err = t.Get(k, nil)
  207. if err != nil {
  208. l.Debugln("surprise error:", k, err)
  209. return nil, false
  210. }
  211. fi, err := unmarshalTrunc(bs, truncate)
  212. if err != nil {
  213. l.Debugln("unmarshal error:", k, err)
  214. return nil, false
  215. }
  216. return fi, true
  217. }
  218. func (db *Instance) withGlobal(folder, prefix []byte, truncate bool, fn Iterator) {
  219. t := db.newReadOnlyTransaction()
  220. defer t.close()
  221. dbi := t.NewIterator(util.BytesPrefix(db.globalKey(folder, prefix)), nil)
  222. defer dbi.Release()
  223. slashedPrefix := prefix
  224. if !bytes.HasSuffix(prefix, []byte{'/'}) {
  225. slashedPrefix = append(slashedPrefix, '/')
  226. }
  227. var fk []byte
  228. for dbi.Next() {
  229. var vl VersionList
  230. err := vl.Unmarshal(dbi.Value())
  231. if err != nil {
  232. l.Debugln("unmarshal error:", err)
  233. continue
  234. }
  235. if len(vl.Versions) == 0 {
  236. l.Debugln("no versions:", dbi.Key())
  237. continue
  238. }
  239. name := db.globalKeyName(dbi.Key())
  240. if len(prefix) > 0 && !bytes.Equal(name, prefix) && !bytes.HasPrefix(name, slashedPrefix) {
  241. return
  242. }
  243. fk = db.deviceKeyInto(fk[:cap(fk)], folder, vl.Versions[0].Device, name)
  244. bs, err := t.Get(fk, nil)
  245. if err != nil {
  246. l.Debugln("surprise error:", err)
  247. continue
  248. }
  249. f, err := unmarshalTrunc(bs, truncate)
  250. if err != nil {
  251. l.Debugln("unmarshal error:", err)
  252. continue
  253. }
  254. if cont := fn(f); !cont {
  255. return
  256. }
  257. }
  258. }
  259. func (db *Instance) availability(folder, file []byte) []protocol.DeviceID {
  260. k := db.globalKey(folder, file)
  261. bs, err := db.Get(k, nil)
  262. if err == leveldb.ErrNotFound {
  263. return nil
  264. }
  265. if err != nil {
  266. l.Debugln("surprise error:", err)
  267. return nil
  268. }
  269. var vl VersionList
  270. err = vl.Unmarshal(bs)
  271. if err != nil {
  272. l.Debugln("unmarshal error:", err)
  273. return nil
  274. }
  275. var devices []protocol.DeviceID
  276. for _, v := range vl.Versions {
  277. if !v.Version.Equal(vl.Versions[0].Version) {
  278. break
  279. }
  280. if v.Invalid {
  281. continue
  282. }
  283. n := protocol.DeviceIDFromBytes(v.Device)
  284. devices = append(devices, n)
  285. }
  286. return devices
  287. }
  288. func (db *Instance) withNeed(folder, device []byte, truncate bool, fn Iterator) {
  289. t := db.newReadOnlyTransaction()
  290. defer t.close()
  291. dbi := t.NewIterator(util.BytesPrefix(db.globalKey(folder, nil)[:keyPrefixLen+keyFolderLen]), nil)
  292. defer dbi.Release()
  293. var fk []byte
  294. for dbi.Next() {
  295. var vl VersionList
  296. err := vl.Unmarshal(dbi.Value())
  297. if err != nil {
  298. l.Debugln("unmarshal error:", err)
  299. continue
  300. }
  301. if len(vl.Versions) == 0 {
  302. l.Debugln("no versions:", dbi.Key())
  303. continue
  304. }
  305. have := false // If we have the file, any version
  306. need := false // If we have a lower version of the file
  307. var haveFileVersion FileVersion
  308. for _, v := range vl.Versions {
  309. if bytes.Equal(v.Device, device) {
  310. have = true
  311. haveFileVersion = v
  312. // XXX: This marks Concurrent (i.e. conflicting) changes as
  313. // needs. Maybe we should do that, but it needs special
  314. // handling in the puller.
  315. need = !v.Version.GreaterEqual(vl.Versions[0].Version)
  316. break
  317. }
  318. }
  319. if have && !need {
  320. continue
  321. }
  322. name := db.globalKeyName(dbi.Key())
  323. needVersion := vl.Versions[0].Version
  324. needDevice := protocol.DeviceIDFromBytes(vl.Versions[0].Device)
  325. for i := range vl.Versions {
  326. if !vl.Versions[i].Version.Equal(needVersion) {
  327. // We haven't found a valid copy of the file with the needed version.
  328. break
  329. }
  330. if vl.Versions[i].Invalid {
  331. // The file is marked invalid, don't use it.
  332. continue
  333. }
  334. fk = db.deviceKeyInto(fk[:cap(fk)], folder, vl.Versions[i].Device, name)
  335. bs, err := t.Get(fk, nil)
  336. if err != nil {
  337. l.Debugln("surprise error:", err)
  338. continue
  339. }
  340. gf, err := unmarshalTrunc(bs, truncate)
  341. if err != nil {
  342. l.Debugln("unmarshal error:", err)
  343. continue
  344. }
  345. if gf.IsDeleted() && !have {
  346. // We don't need deleted files that we don't have
  347. break
  348. }
  349. l.Debugf("need folder=%q device=%v name=%q need=%v have=%v invalid=%v haveV=%v globalV=%v globalDev=%v", folder, protocol.DeviceIDFromBytes(device), name, need, have, haveFileVersion.Invalid, haveFileVersion.Version, needVersion, needDevice)
  350. if cont := fn(gf); !cont {
  351. return
  352. }
  353. // This file is handled, no need to look further in the version list
  354. break
  355. }
  356. }
  357. }
  358. func (db *Instance) ListFolders() []string {
  359. t := db.newReadOnlyTransaction()
  360. defer t.close()
  361. dbi := t.NewIterator(util.BytesPrefix([]byte{KeyTypeGlobal}), nil)
  362. defer dbi.Release()
  363. folderExists := make(map[string]bool)
  364. for dbi.Next() {
  365. folder, ok := db.globalKeyFolder(dbi.Key())
  366. if ok && !folderExists[string(folder)] {
  367. folderExists[string(folder)] = true
  368. }
  369. }
  370. folders := make([]string, 0, len(folderExists))
  371. for k := range folderExists {
  372. folders = append(folders, k)
  373. }
  374. sort.Strings(folders)
  375. return folders
  376. }
  377. func (db *Instance) dropFolder(folder []byte) {
  378. t := db.newReadOnlyTransaction()
  379. defer t.close()
  380. // Remove all items related to the given folder from the device->file bucket
  381. dbi := t.NewIterator(util.BytesPrefix([]byte{KeyTypeDevice}), nil)
  382. for dbi.Next() {
  383. itemFolder := db.deviceKeyFolder(dbi.Key())
  384. if bytes.Equal(folder, itemFolder) {
  385. db.Delete(dbi.Key(), nil)
  386. }
  387. }
  388. dbi.Release()
  389. // Remove all items related to the given folder from the global bucket
  390. dbi = t.NewIterator(util.BytesPrefix([]byte{KeyTypeGlobal}), nil)
  391. for dbi.Next() {
  392. itemFolder, ok := db.globalKeyFolder(dbi.Key())
  393. if ok && bytes.Equal(folder, itemFolder) {
  394. db.Delete(dbi.Key(), nil)
  395. }
  396. }
  397. dbi.Release()
  398. }
  399. func (db *Instance) dropDeviceFolder(device, folder []byte, meta *metadataTracker) {
  400. t := db.newReadWriteTransaction()
  401. defer t.close()
  402. dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, device, nil)), nil)
  403. defer dbi.Release()
  404. for dbi.Next() {
  405. key := dbi.Key()
  406. name := db.deviceKeyName(key)
  407. t.removeFromGlobal(folder, device, name, meta)
  408. t.Delete(key)
  409. t.checkFlush()
  410. }
  411. }
  412. func (db *Instance) checkGlobals(folder []byte, meta *metadataTracker) {
  413. t := db.newReadWriteTransaction()
  414. defer t.close()
  415. dbi := t.NewIterator(util.BytesPrefix(db.globalKey(folder, nil)[:keyPrefixLen+keyFolderLen]), nil)
  416. defer dbi.Release()
  417. var fk []byte
  418. for dbi.Next() {
  419. gk := dbi.Key()
  420. var vl VersionList
  421. err := vl.Unmarshal(dbi.Value())
  422. if err != nil {
  423. l.Debugln("unmarshal error:", err)
  424. continue
  425. }
  426. // Check the global version list for consistency. An issue in previous
  427. // versions of goleveldb could result in reordered writes so that
  428. // there are global entries pointing to no longer existing files. Here
  429. // we find those and clear them out.
  430. name := db.globalKeyName(gk)
  431. var newVL VersionList
  432. for i, version := range vl.Versions {
  433. fk = db.deviceKeyInto(fk[:cap(fk)], folder, version.Device, name)
  434. _, err := t.Get(fk, nil)
  435. if err == leveldb.ErrNotFound {
  436. continue
  437. }
  438. if err != nil {
  439. l.Debugln("surprise error:", err)
  440. return
  441. }
  442. newVL.Versions = append(newVL.Versions, version)
  443. if i == 0 {
  444. if fi, ok := t.getFile(folder, version.Device, name); ok {
  445. meta.addFile(globalDeviceID, fi)
  446. }
  447. }
  448. }
  449. if len(newVL.Versions) != len(vl.Versions) {
  450. t.Put(dbi.Key(), mustMarshal(&newVL))
  451. t.checkFlush()
  452. }
  453. }
  454. l.Debugf("db check completed for %q", folder)
  455. }
  456. func (db *Instance) updateSchema0to1() {
  457. t := db.newReadWriteTransaction()
  458. defer t.close()
  459. dbi := t.NewIterator(util.BytesPrefix([]byte{KeyTypeDevice}), nil)
  460. defer dbi.Release()
  461. symlinkConv := 0
  462. changedFolders := make(map[string]struct{})
  463. ignAdded := 0
  464. meta := newMetadataTracker() // dummy metadata tracker
  465. for dbi.Next() {
  466. folder := db.deviceKeyFolder(dbi.Key())
  467. device := db.deviceKeyDevice(dbi.Key())
  468. name := string(db.deviceKeyName(dbi.Key()))
  469. // Remove files with absolute path (see #4799)
  470. if strings.HasPrefix(name, "/") {
  471. if _, ok := changedFolders[string(folder)]; !ok {
  472. changedFolders[string(folder)] = struct{}{}
  473. }
  474. t.removeFromGlobal(folder, device, nil, nil)
  475. t.Delete(dbi.Key())
  476. t.checkFlush()
  477. continue
  478. }
  479. // Change SYMLINK_FILE and SYMLINK_DIRECTORY types to the current SYMLINK
  480. // type (previously SYMLINK_UNKNOWN). It does this for all devices, both
  481. // local and remote, and does not reset delta indexes. It shouldn't really
  482. // matter what the symlink type is, but this cleans it up for a possible
  483. // future when SYMLINK_FILE and SYMLINK_DIRECTORY are no longer understood.
  484. var f protocol.FileInfo
  485. if err := f.Unmarshal(dbi.Value()); err != nil {
  486. // probably can't happen
  487. continue
  488. }
  489. if f.Type == protocol.FileInfoTypeDeprecatedSymlinkDirectory || f.Type == protocol.FileInfoTypeDeprecatedSymlinkFile {
  490. f.Type = protocol.FileInfoTypeSymlink
  491. bs, err := f.Marshal()
  492. if err != nil {
  493. panic("can't happen: " + err.Error())
  494. }
  495. t.Put(dbi.Key(), bs)
  496. t.checkFlush()
  497. symlinkConv++
  498. }
  499. // Add invalid files to global list
  500. if f.Invalid {
  501. if t.updateGlobal(folder, device, f, meta) {
  502. if _, ok := changedFolders[string(folder)]; !ok {
  503. changedFolders[string(folder)] = struct{}{}
  504. }
  505. ignAdded++
  506. }
  507. }
  508. }
  509. for folder := range changedFolders {
  510. db.dropFolderMeta([]byte(folder))
  511. }
  512. l.Infof("Updated symlink type for %d index entries and added %d invalid files to global list", symlinkConv, ignAdded)
  513. }
  514. // deviceKey returns a byte slice encoding the following information:
  515. // keyTypeDevice (1 byte)
  516. // folder (4 bytes)
  517. // device (4 bytes)
  518. // name (variable size)
  519. func (db *Instance) deviceKey(folder, device, file []byte) []byte {
  520. return db.deviceKeyInto(nil, folder, device, file)
  521. }
  522. func (db *Instance) deviceKeyInto(k []byte, folder, device, file []byte) []byte {
  523. reqLen := keyPrefixLen + keyFolderLen + keyDeviceLen + len(file)
  524. if len(k) < reqLen {
  525. k = make([]byte, reqLen)
  526. }
  527. k[0] = KeyTypeDevice
  528. binary.BigEndian.PutUint32(k[keyPrefixLen:], db.folderIdx.ID(folder))
  529. binary.BigEndian.PutUint32(k[keyPrefixLen+keyFolderLen:], db.deviceIdx.ID(device))
  530. copy(k[keyPrefixLen+keyFolderLen+keyDeviceLen:], file)
  531. return k[:reqLen]
  532. }
  533. // deviceKeyName returns the device ID from the key
  534. func (db *Instance) deviceKeyName(key []byte) []byte {
  535. return key[keyPrefixLen+keyFolderLen+keyDeviceLen:]
  536. }
  537. // deviceKeyFolder returns the folder name from the key
  538. func (db *Instance) deviceKeyFolder(key []byte) []byte {
  539. folder, ok := db.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
  540. if !ok {
  541. panic("bug: lookup of nonexistent folder ID")
  542. }
  543. return folder
  544. }
  545. // deviceKeyDevice returns the device ID from the key
  546. func (db *Instance) deviceKeyDevice(key []byte) []byte {
  547. device, ok := db.deviceIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen+keyFolderLen:]))
  548. if !ok {
  549. panic("bug: lookup of nonexistent device ID")
  550. }
  551. return device
  552. }
  553. // globalKey returns a byte slice encoding the following information:
  554. // keyTypeGlobal (1 byte)
  555. // folder (4 bytes)
  556. // name (variable size)
  557. func (db *Instance) globalKey(folder, file []byte) []byte {
  558. k := make([]byte, keyPrefixLen+keyFolderLen+len(file))
  559. k[0] = KeyTypeGlobal
  560. binary.BigEndian.PutUint32(k[keyPrefixLen:], db.folderIdx.ID(folder))
  561. copy(k[keyPrefixLen+keyFolderLen:], file)
  562. return k
  563. }
  564. // globalKeyName returns the filename from the key
  565. func (db *Instance) globalKeyName(key []byte) []byte {
  566. return key[keyPrefixLen+keyFolderLen:]
  567. }
  568. // globalKeyFolder returns the folder name from the key
  569. func (db *Instance) globalKeyFolder(key []byte) ([]byte, bool) {
  570. return db.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
  571. }
  572. func (db *Instance) getIndexID(device, folder []byte) protocol.IndexID {
  573. key := db.indexIDKey(device, folder)
  574. cur, err := db.Get(key, nil)
  575. if err != nil {
  576. return 0
  577. }
  578. var id protocol.IndexID
  579. if err := id.Unmarshal(cur); err != nil {
  580. return 0
  581. }
  582. return id
  583. }
  584. func (db *Instance) setIndexID(device, folder []byte, id protocol.IndexID) {
  585. key := db.indexIDKey(device, folder)
  586. bs, _ := id.Marshal() // marshalling can't fail
  587. if err := db.Put(key, bs, nil); err != nil {
  588. panic("storing index ID: " + err.Error())
  589. }
  590. }
  591. func (db *Instance) indexIDKey(device, folder []byte) []byte {
  592. k := make([]byte, keyPrefixLen+keyDeviceLen+keyFolderLen)
  593. k[0] = KeyTypeIndexID
  594. binary.BigEndian.PutUint32(k[keyPrefixLen:], db.deviceIdx.ID(device))
  595. binary.BigEndian.PutUint32(k[keyPrefixLen+keyDeviceLen:], db.folderIdx.ID(folder))
  596. return k
  597. }
  598. func (db *Instance) indexIDDevice(key []byte) []byte {
  599. device, ok := db.deviceIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
  600. if !ok {
  601. // uuh ...
  602. return nil
  603. }
  604. return device
  605. }
  606. func (db *Instance) mtimesKey(folder []byte) []byte {
  607. prefix := make([]byte, 5) // key type + 4 bytes folder idx number
  608. prefix[0] = KeyTypeVirtualMtime
  609. binary.BigEndian.PutUint32(prefix[1:], db.folderIdx.ID(folder))
  610. return prefix
  611. }
  612. func (db *Instance) folderMetaKey(folder []byte) []byte {
  613. prefix := make([]byte, 5) // key type + 4 bytes folder idx number
  614. prefix[0] = KeyTypeFolderMeta
  615. binary.BigEndian.PutUint32(prefix[1:], db.folderIdx.ID(folder))
  616. return prefix
  617. }
  618. // DropLocalDeltaIndexIDs removes all index IDs for the local device ID from
  619. // the database. This will cause a full index transmission on the next
  620. // connection.
  621. func (db *Instance) DropLocalDeltaIndexIDs() {
  622. db.dropDeltaIndexIDs(true)
  623. }
  624. // DropRemoteDeltaIndexIDs removes all index IDs for the other devices than
  625. // the local one from the database. This will cause them to send us a full
  626. // index on the next connection.
  627. func (db *Instance) DropRemoteDeltaIndexIDs() {
  628. db.dropDeltaIndexIDs(false)
  629. }
  630. func (db *Instance) dropDeltaIndexIDs(local bool) {
  631. t := db.newReadWriteTransaction()
  632. defer t.close()
  633. dbi := t.NewIterator(util.BytesPrefix([]byte{KeyTypeIndexID}), nil)
  634. defer dbi.Release()
  635. for dbi.Next() {
  636. device := db.indexIDDevice(dbi.Key())
  637. if bytes.Equal(device, protocol.LocalDeviceID[:]) == local {
  638. t.Delete(dbi.Key())
  639. }
  640. }
  641. }
  642. func (db *Instance) dropMtimes(folder []byte) {
  643. db.dropPrefix(db.mtimesKey(folder))
  644. }
  645. func (db *Instance) dropFolderMeta(folder []byte) {
  646. db.dropPrefix(db.folderMetaKey(folder))
  647. }
  648. func (db *Instance) dropPrefix(prefix []byte) {
  649. t := db.newReadWriteTransaction()
  650. defer t.close()
  651. dbi := t.NewIterator(util.BytesPrefix(prefix), nil)
  652. defer dbi.Release()
  653. for dbi.Next() {
  654. t.Delete(dbi.Key())
  655. }
  656. }
  657. func unmarshalTrunc(bs []byte, truncate bool) (FileIntf, error) {
  658. if truncate {
  659. var tf FileInfoTruncated
  660. err := tf.Unmarshal(bs)
  661. return tf, err
  662. }
  663. var tf protocol.FileInfo
  664. err := tf.Unmarshal(bs)
  665. return tf, err
  666. }
  667. // A "better" version of leveldb's errors.IsCorrupted.
  668. func leveldbIsCorrupted(err error) bool {
  669. switch {
  670. case err == nil:
  671. return false
  672. case errors.IsCorrupted(err):
  673. return true
  674. case strings.Contains(err.Error(), "corrupted"):
  675. return true
  676. }
  677. return false
  678. }
  679. // A smallIndex is an in memory bidirectional []byte to uint32 map. It gives
  680. // fast lookups in both directions and persists to the database. Don't use for
  681. // storing more items than fit comfortably in RAM.
  682. type smallIndex struct {
  683. db *Instance
  684. prefix []byte
  685. id2val map[uint32]string
  686. val2id map[string]uint32
  687. nextID uint32
  688. mut sync.Mutex
  689. }
  690. func newSmallIndex(db *Instance, prefix []byte) *smallIndex {
  691. idx := &smallIndex{
  692. db: db,
  693. prefix: prefix,
  694. id2val: make(map[uint32]string),
  695. val2id: make(map[string]uint32),
  696. mut: sync.NewMutex(),
  697. }
  698. idx.load()
  699. return idx
  700. }
  701. // load iterates over the prefix space in the database and populates the in
  702. // memory maps.
  703. func (i *smallIndex) load() {
  704. tr := i.db.newReadOnlyTransaction()
  705. it := tr.NewIterator(util.BytesPrefix(i.prefix), nil)
  706. for it.Next() {
  707. val := string(it.Value())
  708. id := binary.BigEndian.Uint32(it.Key()[len(i.prefix):])
  709. i.id2val[id] = val
  710. i.val2id[val] = id
  711. if id >= i.nextID {
  712. i.nextID = id + 1
  713. }
  714. }
  715. it.Release()
  716. tr.close()
  717. }
  718. // ID returns the index number for the given byte slice, allocating a new one
  719. // and persisting this to the database if necessary.
  720. func (i *smallIndex) ID(val []byte) uint32 {
  721. i.mut.Lock()
  722. // intentionally avoiding defer here as we want this call to be as fast as
  723. // possible in the general case (folder ID already exists). The map lookup
  724. // with the conversion of []byte to string is compiler optimized to not
  725. // copy the []byte, which is why we don't assign it to a temp variable
  726. // here.
  727. if id, ok := i.val2id[string(val)]; ok {
  728. i.mut.Unlock()
  729. return id
  730. }
  731. id := i.nextID
  732. i.nextID++
  733. valStr := string(val)
  734. i.val2id[valStr] = id
  735. i.id2val[id] = valStr
  736. key := make([]byte, len(i.prefix)+8) // prefix plus uint32 id
  737. copy(key, i.prefix)
  738. binary.BigEndian.PutUint32(key[len(i.prefix):], id)
  739. i.db.Put(key, val, nil)
  740. i.mut.Unlock()
  741. return id
  742. }
  743. // Val returns the value for the given index number, or (nil, false) if there
  744. // is no such index number.
  745. func (i *smallIndex) Val(id uint32) ([]byte, bool) {
  746. i.mut.Lock()
  747. val, ok := i.id2val[id]
  748. i.mut.Unlock()
  749. if !ok {
  750. return nil, false
  751. }
  752. return []byte(val), true
  753. }