mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-03 07:12:27 +00:00
Change DB label format (index folders, devices)
This commit is contained in:
parent
18ae87962d
commit
ac190b2e39
@ -48,7 +48,7 @@ var locations = map[locationEnum]string{
|
|||||||
locKeyFile: "${config}/key.pem",
|
locKeyFile: "${config}/key.pem",
|
||||||
locHTTPSCertFile: "${config}/https-cert.pem",
|
locHTTPSCertFile: "${config}/https-cert.pem",
|
||||||
locHTTPSKeyFile: "${config}/https-key.pem",
|
locHTTPSKeyFile: "${config}/https-key.pem",
|
||||||
locDatabase: "${config}/index-v0.11.0.db",
|
locDatabase: "${config}/index-v0.13.0.db",
|
||||||
locLogFile: "${config}/syncthing.log", // -logfile on Windows
|
locLogFile: "${config}/syncthing.log", // -logfile on Windows
|
||||||
locCsrfTokens: "${config}/csrftokens.txt",
|
locCsrfTokens: "${config}/csrftokens.txt",
|
||||||
locPanicLog: "${config}/panic-${timestamp}.log",
|
locPanicLog: "${config}/panic-${timestamp}.log",
|
||||||
|
@ -633,6 +633,7 @@ func syncthingMain(runtimeOptions RuntimeOptions) {
|
|||||||
|
|
||||||
dbFile := locations[locDatabase]
|
dbFile := locations[locDatabase]
|
||||||
ldb, err := db.Open(dbFile)
|
ldb, err := db.Open(dbFile)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
l.Fatalln("Cannot open database:", err, "- Is another copy of Syncthing already running?")
|
l.Fatalln("Cannot open database:", err, "- Is another copy of Syncthing already running?")
|
||||||
}
|
}
|
||||||
@ -1156,6 +1157,7 @@ func cleanConfigDirectory() {
|
|||||||
"panic-*.log": 7 * 24 * time.Hour, // keep panic logs for a week
|
"panic-*.log": 7 * 24 * time.Hour, // keep panic logs for a week
|
||||||
"audit-*.log": 7 * 24 * time.Hour, // keep audit logs for a week
|
"audit-*.log": 7 * 24 * time.Hour, // keep audit logs for a week
|
||||||
"index": 14 * 24 * time.Hour, // keep old index format for two weeks
|
"index": 14 * 24 * time.Hour, // keep old index format for two weeks
|
||||||
|
"index*.converted": 14 * 24 * time.Hour, // keep old converted indexes for two weeks
|
||||||
"config.xml.v*": 30 * 24 * time.Hour, // old config versions for a month
|
"config.xml.v*": 30 * 24 * time.Hour, // old config versions for a month
|
||||||
"*.idx.gz": 30 * 24 * time.Hour, // these should for sure no longer exist
|
"*.idx.gz": 30 * 24 * time.Hour, // these should for sure no longer exist
|
||||||
"backup-of-v0.8": 30 * 24 * time.Hour, // these neither
|
"backup-of-v0.8": 30 * 24 * time.Hour, // these neither
|
||||||
|
1
lib/db/.gitignore
vendored
1
lib/db/.gitignore
vendored
@ -1 +1,2 @@
|
|||||||
|
!*.zip
|
||||||
testdata/*.db
|
testdata/*.db
|
||||||
|
@ -4,16 +4,9 @@
|
|||||||
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
||||||
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
|
||||||
// Package db provides a set type to track local/remote files with newness
|
|
||||||
// checks. We must do a certain amount of normalization in here. We will get
|
|
||||||
// fed paths with either native or wire-format separators and encodings
|
|
||||||
// depending on who calls us. We transform paths to wire-format (NFC and
|
|
||||||
// slashes) on the way to the database, and transform to native format
|
|
||||||
// (varying separator and encoding) on the way back out.
|
|
||||||
package db
|
package db
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
@ -30,10 +23,10 @@ const maxBatchSize = 256 << 10
|
|||||||
|
|
||||||
type BlockMap struct {
|
type BlockMap struct {
|
||||||
db *Instance
|
db *Instance
|
||||||
folder string
|
folder uint32
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewBlockMap(db *Instance, folder string) *BlockMap {
|
func NewBlockMap(db *Instance, folder uint32) *BlockMap {
|
||||||
return &BlockMap{
|
return &BlockMap{
|
||||||
db: db,
|
db: db,
|
||||||
folder: folder,
|
folder: folder,
|
||||||
@ -123,7 +116,7 @@ func (m *BlockMap) Discard(files []protocol.FileInfo) error {
|
|||||||
// Drop block map, removing all entries related to this block map from the db.
|
// Drop block map, removing all entries related to this block map from the db.
|
||||||
func (m *BlockMap) Drop() error {
|
func (m *BlockMap) Drop() error {
|
||||||
batch := new(leveldb.Batch)
|
batch := new(leveldb.Batch)
|
||||||
iter := m.db.NewIterator(util.BytesPrefix(m.blockKeyInto(nil, nil, "")[:1+64]), nil)
|
iter := m.db.NewIterator(util.BytesPrefix(m.blockKeyInto(nil, nil, "")[:keyPrefixLen+keyFolderLen]), nil)
|
||||||
defer iter.Release()
|
defer iter.Release()
|
||||||
for iter.Next() {
|
for iter.Next() {
|
||||||
if batch.Len() > maxBatchSize {
|
if batch.Len() > maxBatchSize {
|
||||||
@ -173,12 +166,13 @@ func (f *BlockFinder) String() string {
|
|||||||
func (f *BlockFinder) Iterate(folders []string, hash []byte, iterFn func(string, string, int32) bool) bool {
|
func (f *BlockFinder) Iterate(folders []string, hash []byte, iterFn func(string, string, int32) bool) bool {
|
||||||
var key []byte
|
var key []byte
|
||||||
for _, folder := range folders {
|
for _, folder := range folders {
|
||||||
key = blockKeyInto(key, hash, folder, "")
|
folderID := f.db.folderIdx.ID([]byte(folder))
|
||||||
|
key = blockKeyInto(key, hash, folderID, "")
|
||||||
iter := f.db.NewIterator(util.BytesPrefix(key), nil)
|
iter := f.db.NewIterator(util.BytesPrefix(key), nil)
|
||||||
defer iter.Release()
|
defer iter.Release()
|
||||||
|
|
||||||
for iter.Next() && iter.Error() == nil {
|
for iter.Next() && iter.Error() == nil {
|
||||||
folder, file := fromBlockKey(iter.Key())
|
file := blockKeyName(iter.Key())
|
||||||
index := int32(binary.BigEndian.Uint32(iter.Value()))
|
index := int32(binary.BigEndian.Uint32(iter.Value()))
|
||||||
if iterFn(folder, osutil.NativeFilename(file), index) {
|
if iterFn(folder, osutil.NativeFilename(file), index) {
|
||||||
return true
|
return true
|
||||||
@ -194,48 +188,41 @@ func (f *BlockFinder) Fix(folder, file string, index int32, oldHash, newHash []b
|
|||||||
buf := make([]byte, 4)
|
buf := make([]byte, 4)
|
||||||
binary.BigEndian.PutUint32(buf, uint32(index))
|
binary.BigEndian.PutUint32(buf, uint32(index))
|
||||||
|
|
||||||
|
folderID := f.db.folderIdx.ID([]byte(folder))
|
||||||
batch := new(leveldb.Batch)
|
batch := new(leveldb.Batch)
|
||||||
batch.Delete(blockKeyInto(nil, oldHash, folder, file))
|
batch.Delete(blockKeyInto(nil, oldHash, folderID, file))
|
||||||
batch.Put(blockKeyInto(nil, newHash, folder, file), buf)
|
batch.Put(blockKeyInto(nil, newHash, folderID, file), buf)
|
||||||
return f.db.Write(batch, nil)
|
return f.db.Write(batch, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// m.blockKey returns a byte slice encoding the following information:
|
// m.blockKey returns a byte slice encoding the following information:
|
||||||
// keyTypeBlock (1 byte)
|
// keyTypeBlock (1 byte)
|
||||||
// folder (64 bytes)
|
// folder (4 bytes)
|
||||||
// block hash (32 bytes)
|
// block hash (32 bytes)
|
||||||
// file name (variable size)
|
// file name (variable size)
|
||||||
func blockKeyInto(o, hash []byte, folder, file string) []byte {
|
func blockKeyInto(o, hash []byte, folder uint32, file string) []byte {
|
||||||
reqLen := 1 + 64 + 32 + len(file)
|
reqLen := keyPrefixLen + keyFolderLen + keyHashLen + len(file)
|
||||||
if cap(o) < reqLen {
|
if cap(o) < reqLen {
|
||||||
o = make([]byte, reqLen)
|
o = make([]byte, reqLen)
|
||||||
} else {
|
} else {
|
||||||
o = o[:reqLen]
|
o = o[:reqLen]
|
||||||
}
|
}
|
||||||
o[0] = KeyTypeBlock
|
o[0] = KeyTypeBlock
|
||||||
copy(o[1:], []byte(folder))
|
binary.BigEndian.PutUint32(o[keyPrefixLen:], folder)
|
||||||
for i := len(folder); i < 64; i++ {
|
copy(o[keyPrefixLen+keyFolderLen:], []byte(hash))
|
||||||
o[1+i] = 0
|
copy(o[keyPrefixLen+keyFolderLen+keyHashLen:], []byte(file))
|
||||||
}
|
|
||||||
copy(o[1+64:], []byte(hash))
|
|
||||||
copy(o[1+64+32:], []byte(file))
|
|
||||||
return o
|
return o
|
||||||
}
|
}
|
||||||
|
|
||||||
func fromBlockKey(data []byte) (string, string) {
|
// blockKeyName returns the file name from the block key
|
||||||
if len(data) < 1+64+32+1 {
|
func blockKeyName(data []byte) string {
|
||||||
|
if len(data) < keyPrefixLen+keyFolderLen+keyHashLen+1 {
|
||||||
panic("Incorrect key length")
|
panic("Incorrect key length")
|
||||||
}
|
}
|
||||||
if data[0] != KeyTypeBlock {
|
if data[0] != KeyTypeBlock {
|
||||||
panic("Incorrect key type")
|
panic("Incorrect key type")
|
||||||
}
|
}
|
||||||
|
|
||||||
file := string(data[1+64+32:])
|
file := string(data[keyPrefixLen+keyFolderLen+keyHashLen:])
|
||||||
|
return file
|
||||||
slice := data[1 : 1+64]
|
|
||||||
izero := bytes.IndexByte(slice, 0)
|
|
||||||
if izero > -1 {
|
|
||||||
return string(slice[:izero]), file
|
|
||||||
}
|
|
||||||
return string(slice), file
|
|
||||||
}
|
}
|
||||||
|
@ -10,6 +10,7 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/syncthing/syncthing/lib/protocol"
|
"github.com/syncthing/syncthing/lib/protocol"
|
||||||
|
"github.com/syndtr/goleveldb/leveldb/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
func genBlocks(n int) []protocol.BlockInfo {
|
func genBlocks(n int) []protocol.BlockInfo {
|
||||||
@ -55,7 +56,7 @@ func setup() (*Instance, *BlockFinder) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func dbEmpty(db *Instance) bool {
|
func dbEmpty(db *Instance) bool {
|
||||||
iter := db.NewIterator(nil, nil)
|
iter := db.NewIterator(util.BytesPrefix([]byte{KeyTypeBlock}), nil)
|
||||||
defer iter.Release()
|
defer iter.Release()
|
||||||
if iter.Next() {
|
if iter.Next() {
|
||||||
return false
|
return false
|
||||||
@ -70,7 +71,7 @@ func TestBlockMapAddUpdateWipe(t *testing.T) {
|
|||||||
t.Fatal("db not empty")
|
t.Fatal("db not empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
m := NewBlockMap(db, "folder1")
|
m := NewBlockMap(db, db.folderIdx.ID([]byte("folder1")))
|
||||||
|
|
||||||
f3.Flags |= protocol.FlagDirectory
|
f3.Flags |= protocol.FlagDirectory
|
||||||
|
|
||||||
@ -152,8 +153,8 @@ func TestBlockMapAddUpdateWipe(t *testing.T) {
|
|||||||
func TestBlockFinderLookup(t *testing.T) {
|
func TestBlockFinderLookup(t *testing.T) {
|
||||||
db, f := setup()
|
db, f := setup()
|
||||||
|
|
||||||
m1 := NewBlockMap(db, "folder1")
|
m1 := NewBlockMap(db, db.folderIdx.ID([]byte("folder1")))
|
||||||
m2 := NewBlockMap(db, "folder2")
|
m2 := NewBlockMap(db, db.folderIdx.ID([]byte("folder2")))
|
||||||
|
|
||||||
err := m1.Add([]protocol.FileInfo{f1})
|
err := m1.Add([]protocol.FileInfo{f1})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -221,7 +222,7 @@ func TestBlockFinderFix(t *testing.T) {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
m := NewBlockMap(db, "folder1")
|
m := NewBlockMap(db, db.folderIdx.ID([]byte("folder1")))
|
||||||
err := m.Add([]protocol.FileInfo{f1})
|
err := m.Add([]protocol.FileInfo{f1})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -42,6 +42,8 @@ const (
|
|||||||
KeyTypeDeviceStatistic
|
KeyTypeDeviceStatistic
|
||||||
KeyTypeFolderStatistic
|
KeyTypeFolderStatistic
|
||||||
KeyTypeVirtualMtime
|
KeyTypeVirtualMtime
|
||||||
|
KeyTypeFolderIdx
|
||||||
|
KeyTypeDeviceIdx
|
||||||
)
|
)
|
||||||
|
|
||||||
type fileVersion struct {
|
type fileVersion struct {
|
||||||
|
114
lib/db/leveldb_convert.go
Normal file
114
lib/db/leveldb_convert.go
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
// Copyright (C) 2015 The Syncthing Authors.
|
||||||
|
//
|
||||||
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
||||||
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
|
||||||
|
package db
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
|
||||||
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
|
)
|
||||||
|
|
||||||
|
// convertKeyFormat converts from the v0.12 to the v0.13 database format, to
|
||||||
|
// avoid having to do rescan. The change is in the key format for folder
|
||||||
|
// labels, so we basically just iterate over the database rewriting keys as
|
||||||
|
// necessary.
|
||||||
|
func convertKeyFormat(from, to *leveldb.DB) error {
|
||||||
|
l.Infoln("Converting database key format")
|
||||||
|
blocks, files, globals, unchanged := 0, 0, 0, 0
|
||||||
|
|
||||||
|
dbi := newDBInstance(to)
|
||||||
|
i := from.NewIterator(nil, nil)
|
||||||
|
for i.Next() {
|
||||||
|
key := i.Key()
|
||||||
|
switch key[0] {
|
||||||
|
case KeyTypeBlock:
|
||||||
|
folder, file := oldFromBlockKey(key)
|
||||||
|
folderIdx := dbi.folderIdx.ID([]byte(folder))
|
||||||
|
hash := key[1+64:]
|
||||||
|
newKey := blockKeyInto(nil, hash, folderIdx, file)
|
||||||
|
if err := to.Put(newKey, i.Value(), nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
blocks++
|
||||||
|
|
||||||
|
case KeyTypeDevice:
|
||||||
|
newKey := dbi.deviceKey(oldDeviceKeyFolder(key), oldDeviceKeyDevice(key), oldDeviceKeyName(key))
|
||||||
|
if err := to.Put(newKey, i.Value(), nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
files++
|
||||||
|
|
||||||
|
case KeyTypeGlobal:
|
||||||
|
newKey := dbi.globalKey(oldGlobalKeyFolder(key), oldGlobalKeyName(key))
|
||||||
|
if err := to.Put(newKey, i.Value(), nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
globals++
|
||||||
|
|
||||||
|
case KeyTypeVirtualMtime:
|
||||||
|
// Cannot be converted, we drop it instead :(
|
||||||
|
|
||||||
|
default:
|
||||||
|
if err := to.Put(key, i.Value(), nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
unchanged++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
l.Infof("Converted %d blocks, %d files, %d globals (%d unchanged).", blocks, files, globals, unchanged)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldDeviceKeyFolder(key []byte) []byte {
|
||||||
|
folder := key[1 : 1+64]
|
||||||
|
izero := bytes.IndexByte(folder, 0)
|
||||||
|
if izero < 0 {
|
||||||
|
return folder
|
||||||
|
}
|
||||||
|
return folder[:izero]
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldDeviceKeyDevice(key []byte) []byte {
|
||||||
|
return key[1+64 : 1+64+32]
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldDeviceKeyName(key []byte) []byte {
|
||||||
|
return key[1+64+32:]
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldGlobalKeyName(key []byte) []byte {
|
||||||
|
return key[1+64:]
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldGlobalKeyFolder(key []byte) []byte {
|
||||||
|
folder := key[1 : 1+64]
|
||||||
|
izero := bytes.IndexByte(folder, 0)
|
||||||
|
if izero < 0 {
|
||||||
|
return folder
|
||||||
|
}
|
||||||
|
return folder[:izero]
|
||||||
|
}
|
||||||
|
|
||||||
|
func oldFromBlockKey(data []byte) (string, string) {
|
||||||
|
if len(data) < 1+64+32+1 {
|
||||||
|
panic("Incorrect key length")
|
||||||
|
}
|
||||||
|
if data[0] != KeyTypeBlock {
|
||||||
|
panic("Incorrect key type")
|
||||||
|
}
|
||||||
|
|
||||||
|
file := string(data[1+64+32:])
|
||||||
|
|
||||||
|
slice := data[1 : 1+64]
|
||||||
|
izero := bytes.IndexByte(slice, 0)
|
||||||
|
if izero > -1 {
|
||||||
|
return string(slice[:izero]), file
|
||||||
|
}
|
||||||
|
return string(slice), file
|
||||||
|
}
|
136
lib/db/leveldb_convert_test.go
Normal file
136
lib/db/leveldb_convert_test.go
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
// Copyright (C) 2015 The Syncthing Authors.
|
||||||
|
//
|
||||||
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
||||||
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
|
||||||
|
package db
|
||||||
|
|
||||||
|
import (
|
||||||
|
"archive/zip"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestLabelConversion(t *testing.T) {
|
||||||
|
os.RemoveAll("testdata/oldformat.db")
|
||||||
|
defer os.RemoveAll("testdata/oldformat.db")
|
||||||
|
os.RemoveAll("testdata/newformat.db")
|
||||||
|
defer os.RemoveAll("testdata/newformat.db")
|
||||||
|
|
||||||
|
if err := unzip("testdata/oldformat.db.zip", "testdata"); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
odb, err := leveldb.OpenFile("testdata/oldformat.db", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ldb, err := leveldb.OpenFile("testdata/newformat.db", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = convertKeyFormat(odb, ldb); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
ldb.Close()
|
||||||
|
odb.Close()
|
||||||
|
|
||||||
|
inst, err := Open("testdata/newformat.db")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
fs := NewFileSet("default", inst)
|
||||||
|
files, deleted, _ := fs.GlobalSize()
|
||||||
|
if files+deleted != 953 {
|
||||||
|
// Expected number of global entries determined by
|
||||||
|
// ../../bin/stindex testdata/oldformat.db/ | grep global | grep -c default
|
||||||
|
t.Errorf("Conversion error, global list differs (%d != 953)", files+deleted)
|
||||||
|
}
|
||||||
|
|
||||||
|
files, deleted, _ = fs.LocalSize()
|
||||||
|
if files+deleted != 953 {
|
||||||
|
t.Errorf("Conversion error, device list differs (%d != 953)", files+deleted)
|
||||||
|
}
|
||||||
|
|
||||||
|
f := NewBlockFinder(inst)
|
||||||
|
// [block] F:"default" H:1c25dea9003cc16216e2a22900be1ec1cc5aaf270442904e2f9812c314e929d8 N:"f/f2/f25f1b3e6e029231b933531b2138796d" I:3
|
||||||
|
h := []byte{0x1c, 0x25, 0xde, 0xa9, 0x00, 0x3c, 0xc1, 0x62, 0x16, 0xe2, 0xa2, 0x29, 0x00, 0xbe, 0x1e, 0xc1, 0xcc, 0x5a, 0xaf, 0x27, 0x04, 0x42, 0x90, 0x4e, 0x2f, 0x98, 0x12, 0xc3, 0x14, 0xe9, 0x29, 0xd8}
|
||||||
|
found := 0
|
||||||
|
f.Iterate([]string{"default"}, h, func(folder, file string, idx int32) bool {
|
||||||
|
if folder == "default" && file == filepath.FromSlash("f/f2/f25f1b3e6e029231b933531b2138796d") && idx == 3 {
|
||||||
|
found++
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
if found != 1 {
|
||||||
|
t.Errorf("Found %d blocks instead of expected 1", found)
|
||||||
|
}
|
||||||
|
|
||||||
|
inst.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func unzip(src, dest string) error {
|
||||||
|
r, err := zip.OpenReader(src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if err := r.Close(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
os.MkdirAll(dest, 0755)
|
||||||
|
|
||||||
|
// Closure to address file descriptors issue with all the deferred .Close() methods
|
||||||
|
extractAndWriteFile := func(f *zip.File) error {
|
||||||
|
rc, err := f.Open()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if err := rc.Close(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
path := filepath.Join(dest, f.Name)
|
||||||
|
|
||||||
|
if f.FileInfo().IsDir() {
|
||||||
|
os.MkdirAll(path, f.Mode())
|
||||||
|
} else {
|
||||||
|
f, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if err := f.Close(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
_, err = io.Copy(f, rc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, f := range r.File {
|
||||||
|
err := extractAndWriteFile(f)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
@ -8,11 +8,15 @@ package db
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/syncthing/syncthing/lib/osutil"
|
||||||
"github.com/syncthing/syncthing/lib/protocol"
|
"github.com/syncthing/syncthing/lib/protocol"
|
||||||
|
"github.com/syncthing/syncthing/lib/sync"
|
||||||
"github.com/syndtr/goleveldb/leveldb"
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
"github.com/syndtr/goleveldb/leveldb/errors"
|
"github.com/syndtr/goleveldb/leveldb/errors"
|
||||||
"github.com/syndtr/goleveldb/leveldb/iterator"
|
"github.com/syndtr/goleveldb/leveldb/iterator"
|
||||||
@ -25,14 +29,33 @@ type deletionHandler func(t readWriteTransaction, folder, device, name []byte, d
|
|||||||
|
|
||||||
type Instance struct {
|
type Instance struct {
|
||||||
*leveldb.DB
|
*leveldb.DB
|
||||||
|
folderIdx *smallIndex
|
||||||
|
deviceIdx *smallIndex
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
keyPrefixLen = 1
|
||||||
|
keyFolderLen = 4 // indexed
|
||||||
|
keyDeviceLen = 4 // indexed
|
||||||
|
keyHashLen = 32
|
||||||
|
)
|
||||||
|
|
||||||
func Open(file string) (*Instance, error) {
|
func Open(file string) (*Instance, error) {
|
||||||
opts := &opt.Options{
|
opts := &opt.Options{
|
||||||
OpenFilesCacheCapacity: 100,
|
OpenFilesCacheCapacity: 100,
|
||||||
WriteBuffer: 4 << 20,
|
WriteBuffer: 4 << 20,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if _, err := os.Stat(file); os.IsNotExist(err) {
|
||||||
|
// The file we are looking to open does not exist. This may be the
|
||||||
|
// first launch so we should look for an old version and try to
|
||||||
|
// convert it.
|
||||||
|
if err := checkConvertDatabase(file); err != nil {
|
||||||
|
l.Infoln("Converting old database:", err)
|
||||||
|
l.Infoln("Will rescan from scratch.")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
db, err := leveldb.OpenFile(file, opts)
|
db, err := leveldb.OpenFile(file, opts)
|
||||||
if leveldbIsCorrupted(err) {
|
if leveldbIsCorrupted(err) {
|
||||||
db, err = leveldb.RecoverFile(file, opts)
|
db, err = leveldb.RecoverFile(file, opts)
|
||||||
@ -60,9 +83,12 @@ func OpenMemory() *Instance {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newDBInstance(db *leveldb.DB) *Instance {
|
func newDBInstance(db *leveldb.DB) *Instance {
|
||||||
return &Instance{
|
i := &Instance{
|
||||||
DB: db,
|
DB: db,
|
||||||
}
|
}
|
||||||
|
i.folderIdx = newSmallIndex(i, []byte{KeyTypeFolderIdx})
|
||||||
|
i.deviceIdx = newSmallIndex(i, []byte{KeyTypeDeviceIdx})
|
||||||
|
return i
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *Instance) Compact() error {
|
func (db *Instance) Compact() error {
|
||||||
@ -72,13 +98,10 @@ func (db *Instance) Compact() error {
|
|||||||
func (db *Instance) genericReplace(folder, device []byte, fs []protocol.FileInfo, localSize, globalSize *sizeTracker, deleteFn deletionHandler) int64 {
|
func (db *Instance) genericReplace(folder, device []byte, fs []protocol.FileInfo, localSize, globalSize *sizeTracker, deleteFn deletionHandler) int64 {
|
||||||
sort.Sort(fileList(fs)) // sort list on name, same as in the database
|
sort.Sort(fileList(fs)) // sort list on name, same as in the database
|
||||||
|
|
||||||
start := db.deviceKey(folder, device, nil) // before all folder/device files
|
|
||||||
limit := db.deviceKey(folder, device, []byte{0xff, 0xff, 0xff, 0xff}) // after all folder/device files
|
|
||||||
|
|
||||||
t := db.newReadWriteTransaction()
|
t := db.newReadWriteTransaction()
|
||||||
defer t.close()
|
defer t.close()
|
||||||
|
|
||||||
dbi := t.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
|
dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, device, nil)[:keyPrefixLen+keyFolderLen+keyDeviceLen]), nil)
|
||||||
defer dbi.Release()
|
defer dbi.Release()
|
||||||
|
|
||||||
moreDb := dbi.Next()
|
moreDb := dbi.Next()
|
||||||
@ -237,13 +260,10 @@ func (db *Instance) updateFiles(folder, device []byte, fs []protocol.FileInfo, l
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (db *Instance) withHave(folder, device []byte, truncate bool, fn Iterator) {
|
func (db *Instance) withHave(folder, device []byte, truncate bool, fn Iterator) {
|
||||||
start := db.deviceKey(folder, device, nil) // before all folder/device files
|
|
||||||
limit := db.deviceKey(folder, device, []byte{0xff, 0xff, 0xff, 0xff}) // after all folder/device files
|
|
||||||
|
|
||||||
t := db.newReadOnlyTransaction()
|
t := db.newReadOnlyTransaction()
|
||||||
defer t.close()
|
defer t.close()
|
||||||
|
|
||||||
dbi := t.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
|
dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, device, nil)[:keyPrefixLen+keyFolderLen+keyDeviceLen]), nil)
|
||||||
defer dbi.Release()
|
defer dbi.Release()
|
||||||
|
|
||||||
for dbi.Next() {
|
for dbi.Next() {
|
||||||
@ -258,13 +278,10 @@ func (db *Instance) withHave(folder, device []byte, truncate bool, fn Iterator)
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (db *Instance) withAllFolderTruncated(folder []byte, fn func(device []byte, f FileInfoTruncated) bool) {
|
func (db *Instance) withAllFolderTruncated(folder []byte, fn func(device []byte, f FileInfoTruncated) bool) {
|
||||||
start := db.deviceKey(folder, nil, nil) // before all folder/device files
|
|
||||||
limit := db.deviceKey(folder, protocol.LocalDeviceID[:], []byte{0xff, 0xff, 0xff, 0xff}) // after all folder/device files
|
|
||||||
|
|
||||||
t := db.newReadWriteTransaction()
|
t := db.newReadWriteTransaction()
|
||||||
defer t.close()
|
defer t.close()
|
||||||
|
|
||||||
dbi := t.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
|
dbi := t.NewIterator(util.BytesPrefix(db.deviceKey(folder, nil, nil)[:keyPrefixLen+keyFolderLen]), nil)
|
||||||
defer dbi.Release()
|
defer dbi.Release()
|
||||||
|
|
||||||
for dbi.Next() {
|
for dbi.Next() {
|
||||||
@ -359,7 +376,10 @@ func (db *Instance) withGlobal(folder, prefix []byte, truncate bool, fn Iterator
|
|||||||
l.Debugf("vl.versions[0].device: %x", vl.versions[0].device)
|
l.Debugf("vl.versions[0].device: %x", vl.versions[0].device)
|
||||||
l.Debugf("name: %q (%x)", name, name)
|
l.Debugf("name: %q (%x)", name, name)
|
||||||
l.Debugf("fk: %q", fk)
|
l.Debugf("fk: %q", fk)
|
||||||
l.Debugf("fk: %x %x %x", fk[1:1+64], fk[1+64:1+64+32], fk[1+64+32:])
|
l.Debugf("fk: %x %x %x",
|
||||||
|
fk[keyPrefixLen:keyPrefixLen+keyFolderLen],
|
||||||
|
fk[keyPrefixLen+keyFolderLen:keyPrefixLen+keyFolderLen+keyDeviceLen],
|
||||||
|
fk[keyPrefixLen+keyFolderLen+keyDeviceLen:])
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -403,13 +423,10 @@ func (db *Instance) availability(folder, file []byte) []protocol.DeviceID {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (db *Instance) withNeed(folder, device []byte, truncate bool, fn Iterator) {
|
func (db *Instance) withNeed(folder, device []byte, truncate bool, fn Iterator) {
|
||||||
start := db.globalKey(folder, nil)
|
|
||||||
limit := db.globalKey(folder, []byte{0xff, 0xff, 0xff, 0xff})
|
|
||||||
|
|
||||||
t := db.newReadOnlyTransaction()
|
t := db.newReadOnlyTransaction()
|
||||||
defer t.close()
|
defer t.close()
|
||||||
|
|
||||||
dbi := t.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
|
dbi := t.NewIterator(util.BytesPrefix(db.globalKey(folder, nil)[:keyPrefixLen+keyFolderLen]), nil)
|
||||||
defer dbi.Release()
|
defer dbi.Release()
|
||||||
|
|
||||||
var fk []byte
|
var fk []byte
|
||||||
@ -546,9 +563,7 @@ func (db *Instance) checkGlobals(folder []byte, globalSize *sizeTracker) {
|
|||||||
t := db.newReadWriteTransaction()
|
t := db.newReadWriteTransaction()
|
||||||
defer t.close()
|
defer t.close()
|
||||||
|
|
||||||
start := db.globalKey(folder, nil)
|
dbi := t.NewIterator(util.BytesPrefix(db.globalKey(folder, nil)[:keyPrefixLen+keyFolderLen]), nil)
|
||||||
limit := db.globalKey(folder, []byte{0xff, 0xff, 0xff, 0xff})
|
|
||||||
dbi := t.NewIterator(&util.Range{Start: start, Limit: limit}, nil)
|
|
||||||
defer dbi.Release()
|
defer dbi.Release()
|
||||||
|
|
||||||
var fk []byte
|
var fk []byte
|
||||||
@ -598,71 +613,72 @@ func (db *Instance) checkGlobals(folder []byte, globalSize *sizeTracker) {
|
|||||||
|
|
||||||
// deviceKey returns a byte slice encoding the following information:
|
// deviceKey returns a byte slice encoding the following information:
|
||||||
// keyTypeDevice (1 byte)
|
// keyTypeDevice (1 byte)
|
||||||
// folder (64 bytes)
|
// folder (4 bytes)
|
||||||
// device (32 bytes)
|
// device (4 bytes)
|
||||||
// name (variable size)
|
// name (variable size)
|
||||||
func (db *Instance) deviceKey(folder, device, file []byte) []byte {
|
func (db *Instance) deviceKey(folder, device, file []byte) []byte {
|
||||||
return db.deviceKeyInto(nil, folder, device, file)
|
return db.deviceKeyInto(nil, folder, device, file)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *Instance) deviceKeyInto(k []byte, folder, device, file []byte) []byte {
|
func (db *Instance) deviceKeyInto(k []byte, folder, device, file []byte) []byte {
|
||||||
reqLen := 1 + 64 + 32 + len(file)
|
reqLen := keyPrefixLen + keyFolderLen + keyDeviceLen + len(file)
|
||||||
if len(k) < reqLen {
|
if len(k) < reqLen {
|
||||||
k = make([]byte, reqLen)
|
k = make([]byte, reqLen)
|
||||||
}
|
}
|
||||||
k[0] = KeyTypeDevice
|
k[0] = KeyTypeDevice
|
||||||
if len(folder) > 64 {
|
binary.BigEndian.PutUint32(k[keyPrefixLen:], db.folderIdx.ID(folder))
|
||||||
panic("folder name too long")
|
binary.BigEndian.PutUint32(k[keyPrefixLen+keyFolderLen:], db.deviceIdx.ID(device))
|
||||||
}
|
copy(k[keyPrefixLen+keyFolderLen+keyDeviceLen:], []byte(file))
|
||||||
copy(k[1:], []byte(folder))
|
|
||||||
copy(k[1+64:], device[:])
|
|
||||||
copy(k[1+64+32:], []byte(file))
|
|
||||||
return k[:reqLen]
|
return k[:reqLen]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// deviceKeyName returns the device ID from the key
|
||||||
func (db *Instance) deviceKeyName(key []byte) []byte {
|
func (db *Instance) deviceKeyName(key []byte) []byte {
|
||||||
return key[1+64+32:]
|
return key[keyPrefixLen+keyFolderLen+keyDeviceLen:]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// deviceKeyFolder returns the folder name from the key
|
||||||
func (db *Instance) deviceKeyFolder(key []byte) []byte {
|
func (db *Instance) deviceKeyFolder(key []byte) []byte {
|
||||||
folder := key[1 : 1+64]
|
folder, ok := db.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
|
||||||
izero := bytes.IndexByte(folder, 0)
|
if !ok {
|
||||||
if izero < 0 {
|
panic("bug: lookup of nonexistent folder ID")
|
||||||
return folder
|
|
||||||
}
|
}
|
||||||
return folder[:izero]
|
return folder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// deviceKeyDevice returns the device ID from the key
|
||||||
func (db *Instance) deviceKeyDevice(key []byte) []byte {
|
func (db *Instance) deviceKeyDevice(key []byte) []byte {
|
||||||
return key[1+64 : 1+64+32]
|
device, ok := db.deviceIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen+keyFolderLen:]))
|
||||||
|
if !ok {
|
||||||
|
panic("bug: lookup of nonexistent device ID")
|
||||||
|
}
|
||||||
|
return device
|
||||||
}
|
}
|
||||||
|
|
||||||
// globalKey returns a byte slice encoding the following information:
|
// globalKey returns a byte slice encoding the following information:
|
||||||
// keyTypeGlobal (1 byte)
|
// keyTypeGlobal (1 byte)
|
||||||
// folder (64 bytes)
|
// folder (4 bytes)
|
||||||
// name (variable size)
|
// name (variable size)
|
||||||
func (db *Instance) globalKey(folder, file []byte) []byte {
|
func (db *Instance) globalKey(folder, file []byte) []byte {
|
||||||
k := make([]byte, 1+64+len(file))
|
k := make([]byte, keyPrefixLen+keyFolderLen+len(file))
|
||||||
k[0] = KeyTypeGlobal
|
k[0] = KeyTypeGlobal
|
||||||
if len(folder) > 64 {
|
binary.BigEndian.PutUint32(k[keyPrefixLen:], db.folderIdx.ID(folder))
|
||||||
panic("folder name too long")
|
copy(k[keyPrefixLen+keyFolderLen:], []byte(file))
|
||||||
}
|
|
||||||
copy(k[1:], []byte(folder))
|
|
||||||
copy(k[1+64:], []byte(file))
|
|
||||||
return k
|
return k
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// globalKeyName returns the filename from the key
|
||||||
func (db *Instance) globalKeyName(key []byte) []byte {
|
func (db *Instance) globalKeyName(key []byte) []byte {
|
||||||
return key[1+64:]
|
return key[keyPrefixLen+keyFolderLen:]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// globalKeyFolder returns the folder name from the key
|
||||||
func (db *Instance) globalKeyFolder(key []byte) []byte {
|
func (db *Instance) globalKeyFolder(key []byte) []byte {
|
||||||
folder := key[1 : 1+64]
|
folder, ok := db.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
|
||||||
izero := bytes.IndexByte(folder, 0)
|
if !ok {
|
||||||
if izero < 0 {
|
panic("bug: lookup of nonexistent folder ID")
|
||||||
return folder
|
|
||||||
}
|
}
|
||||||
return folder[:izero]
|
return folder
|
||||||
}
|
}
|
||||||
|
|
||||||
func unmarshalTrunc(bs []byte, truncate bool) (FileIntf, error) {
|
func unmarshalTrunc(bs []byte, truncate bool) (FileIntf, error) {
|
||||||
@ -692,3 +708,132 @@ func leveldbIsCorrupted(err error) bool {
|
|||||||
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// checkConvertDatabase tries to convert an existing old (v0.11) database to
|
||||||
|
// new (v0.13) format.
|
||||||
|
func checkConvertDatabase(dbFile string) error {
|
||||||
|
oldLoc := filepath.Join(filepath.Dir(dbFile), "index-v0.11.0.db")
|
||||||
|
if _, err := os.Stat(oldLoc); os.IsNotExist(err) {
|
||||||
|
// The old database file does not exist; that's ok, continue as if
|
||||||
|
// everything succeeded.
|
||||||
|
return nil
|
||||||
|
} else if err != nil {
|
||||||
|
// Any other error is weird.
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// There exists a database in the old format. We run a one time
|
||||||
|
// conversion from old to new.
|
||||||
|
|
||||||
|
fromDb, err := leveldb.OpenFile(oldLoc, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
toDb, err := leveldb.OpenFile(dbFile, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = convertKeyFormat(fromDb, toDb)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = toDb.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// We've done this one, we don't want to do it again (if the user runs
|
||||||
|
// -reset or so). We don't care too much about errors any more at this stage.
|
||||||
|
fromDb.Close()
|
||||||
|
osutil.Rename(oldLoc, oldLoc+".converted")
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// A smallIndex is an in memory bidirectional []byte to uint32 map. It gives
|
||||||
|
// fast lookups in both directions and persists to the database. Don't use for
|
||||||
|
// storing more items than fit comfortably in RAM.
|
||||||
|
type smallIndex struct {
|
||||||
|
db *Instance
|
||||||
|
prefix []byte
|
||||||
|
id2val map[uint32]string
|
||||||
|
val2id map[string]uint32
|
||||||
|
nextID uint32
|
||||||
|
mut sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func newSmallIndex(db *Instance, prefix []byte) *smallIndex {
|
||||||
|
idx := &smallIndex{
|
||||||
|
db: db,
|
||||||
|
prefix: prefix,
|
||||||
|
id2val: make(map[uint32]string),
|
||||||
|
val2id: make(map[string]uint32),
|
||||||
|
mut: sync.NewMutex(),
|
||||||
|
}
|
||||||
|
idx.load()
|
||||||
|
return idx
|
||||||
|
}
|
||||||
|
|
||||||
|
// load iterates over the prefix space in the database and populates the in
|
||||||
|
// memory maps.
|
||||||
|
func (i *smallIndex) load() {
|
||||||
|
tr := i.db.newReadOnlyTransaction()
|
||||||
|
it := tr.NewIterator(util.BytesPrefix(i.prefix), nil)
|
||||||
|
for it.Next() {
|
||||||
|
val := string(it.Value())
|
||||||
|
id := binary.BigEndian.Uint32(it.Key()[len(i.prefix):])
|
||||||
|
i.id2val[id] = val
|
||||||
|
i.val2id[val] = id
|
||||||
|
if id >= i.nextID {
|
||||||
|
i.nextID = id + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it.Release()
|
||||||
|
tr.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// ID returns the index number for the given byte slice, allocating a new one
|
||||||
|
// and persisting this to the database if necessary.
|
||||||
|
func (i *smallIndex) ID(val []byte) uint32 {
|
||||||
|
i.mut.Lock()
|
||||||
|
// intentionally avoiding defer here as we want this call to be as fast as
|
||||||
|
// possible in the general case (folder ID already exists). The map lookup
|
||||||
|
// with the conversion of []byte to string is compiler optimized to not
|
||||||
|
// copy the []byte, which is why we don't assign it to a temp variable
|
||||||
|
// here.
|
||||||
|
if id, ok := i.val2id[string(val)]; ok {
|
||||||
|
i.mut.Unlock()
|
||||||
|
return id
|
||||||
|
}
|
||||||
|
|
||||||
|
id := i.nextID
|
||||||
|
i.nextID++
|
||||||
|
|
||||||
|
valStr := string(val)
|
||||||
|
i.val2id[valStr] = id
|
||||||
|
i.id2val[id] = valStr
|
||||||
|
|
||||||
|
key := make([]byte, len(i.prefix)+8) // prefix plus uint32 id
|
||||||
|
copy(key, i.prefix)
|
||||||
|
binary.BigEndian.PutUint32(key[len(i.prefix):], id)
|
||||||
|
i.db.Put(key, val, nil)
|
||||||
|
|
||||||
|
i.mut.Unlock()
|
||||||
|
return id
|
||||||
|
}
|
||||||
|
|
||||||
|
// Val returns the value for the given index number, or (nil, false) if there
|
||||||
|
// is no such index number.
|
||||||
|
func (i *smallIndex) Val(id uint32) ([]byte, bool) {
|
||||||
|
i.mut.Lock()
|
||||||
|
val, ok := i.id2val[id]
|
||||||
|
i.mut.Unlock()
|
||||||
|
if !ok {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
|
return []byte(val), true
|
||||||
|
}
|
||||||
|
@ -16,7 +16,9 @@ func TestDeviceKey(t *testing.T) {
|
|||||||
dev := []byte("device67890123456789012345678901")
|
dev := []byte("device67890123456789012345678901")
|
||||||
name := []byte("name")
|
name := []byte("name")
|
||||||
|
|
||||||
db := &Instance{}
|
db := OpenMemory()
|
||||||
|
db.folderIdx.ID(fld)
|
||||||
|
db.deviceIdx.ID(dev)
|
||||||
|
|
||||||
key := db.deviceKey(fld, dev, name)
|
key := db.deviceKey(fld, dev, name)
|
||||||
|
|
||||||
@ -38,7 +40,8 @@ func TestGlobalKey(t *testing.T) {
|
|||||||
fld := []byte("folder6789012345678901234567890123456789012345678901234567890123")
|
fld := []byte("folder6789012345678901234567890123456789012345678901234567890123")
|
||||||
name := []byte("name")
|
name := []byte("name")
|
||||||
|
|
||||||
db := &Instance{}
|
db := OpenMemory()
|
||||||
|
db.folderIdx.ID(fld)
|
||||||
|
|
||||||
key := db.globalKey(fld, name)
|
key := db.globalKey(fld, name)
|
||||||
|
|
||||||
|
@ -97,7 +97,7 @@ func NewFileSet(folder string, db *Instance) *FileSet {
|
|||||||
localVersion: make(map[protocol.DeviceID]int64),
|
localVersion: make(map[protocol.DeviceID]int64),
|
||||||
folder: folder,
|
folder: folder,
|
||||||
db: db,
|
db: db,
|
||||||
blockmap: NewBlockMap(db, folder),
|
blockmap: NewBlockMap(db, db.folderIdx.ID([]byte(folder))),
|
||||||
mutex: sync.NewMutex(),
|
mutex: sync.NewMutex(),
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -244,7 +244,7 @@ func DropFolder(db *Instance, folder string) {
|
|||||||
db.dropFolder([]byte(folder))
|
db.dropFolder([]byte(folder))
|
||||||
bm := &BlockMap{
|
bm := &BlockMap{
|
||||||
db: db,
|
db: db,
|
||||||
folder: folder,
|
folder: db.folderIdx.ID([]byte(folder)),
|
||||||
}
|
}
|
||||||
bm.Drop()
|
bm.Drop()
|
||||||
NewVirtualMtimeRepo(db, folder).Drop()
|
NewVirtualMtimeRepo(db, folder).Drop()
|
||||||
|
BIN
lib/db/testdata/oldformat.db.zip
vendored
Normal file
BIN
lib/db/testdata/oldformat.db.zip
vendored
Normal file
Binary file not shown.
@ -7,6 +7,7 @@
|
|||||||
package db
|
package db
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@ -24,10 +25,12 @@ type VirtualMtimeRepo struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func NewVirtualMtimeRepo(ldb *Instance, folder string) *VirtualMtimeRepo {
|
func NewVirtualMtimeRepo(ldb *Instance, folder string) *VirtualMtimeRepo {
|
||||||
prefix := string(KeyTypeVirtualMtime) + folder
|
var prefix [5]byte // key type + 4 bytes folder idx number
|
||||||
|
prefix[0] = KeyTypeVirtualMtime
|
||||||
|
binary.BigEndian.PutUint32(prefix[1:], ldb.folderIdx.ID([]byte(folder)))
|
||||||
|
|
||||||
return &VirtualMtimeRepo{
|
return &VirtualMtimeRepo{
|
||||||
ns: NewNamespacedKV(ldb, prefix),
|
ns: NewNamespacedKV(ldb, string(prefix[:])),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,7 +14,6 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"os"
|
"os"
|
||||||
@ -192,6 +191,9 @@ func alterFiles(dir string) error {
|
|||||||
return osutil.TryRename(path, newPath)
|
return osutil.TryRename(path, newPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
This doesn't in fact work. Sometimes it appears to. We need to get this sorted...
|
||||||
|
|
||||||
// Switch between files and directories
|
// Switch between files and directories
|
||||||
case r == 3 && comps > 3 && rand.Float64() < 0.2:
|
case r == 3 && comps > 3 && rand.Float64() < 0.2:
|
||||||
if !info.Mode().IsRegular() {
|
if !info.Mode().IsRegular() {
|
||||||
@ -216,6 +218,7 @@ func alterFiles(dir string) error {
|
|||||||
generateFiles(path, 10, 20, "../LICENSE")
|
generateFiles(path, 10, 20, "../LICENSE")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This fails. Bug?
|
This fails. Bug?
|
||||||
|
Loading…
Reference in New Issue
Block a user