leveldb_dbinstance.go 24 KB

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