2018-09-18 08:41:06 +00:00
|
|
|
// Copyright (C) 2018 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 https://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
keyPrefixLen = 1
|
|
|
|
keyFolderLen = 4 // indexed
|
|
|
|
keyDeviceLen = 4 // indexed
|
|
|
|
keySequenceLen = 8
|
|
|
|
keyHashLen = 32
|
|
|
|
|
|
|
|
maxInt64 int64 = 1<<63 - 1
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2018-12-10 08:55:04 +00:00
|
|
|
// KeyTypeDevice <int32 folder ID> <int32 device ID> <file name> = FileInfo
|
|
|
|
KeyTypeDevice = 0
|
|
|
|
|
|
|
|
// KeyTypeGlobal <int32 folder ID> <file name> = VersionList
|
|
|
|
KeyTypeGlobal = 1
|
|
|
|
|
|
|
|
// KeyTypeBlock <int32 folder ID> <32 bytes hash> <§file name> = int32 (block index)
|
|
|
|
KeyTypeBlock = 2
|
|
|
|
|
|
|
|
// KeyTypeDeviceStatistic <device ID as string> <some string> = some value
|
2018-09-18 08:41:06 +00:00
|
|
|
KeyTypeDeviceStatistic = 3
|
2018-12-10 08:55:04 +00:00
|
|
|
|
|
|
|
// KeyTypeFolderStatistic <folder ID as string> <some string> = some value
|
2018-09-18 08:41:06 +00:00
|
|
|
KeyTypeFolderStatistic = 4
|
2018-12-10 08:55:04 +00:00
|
|
|
|
|
|
|
// KeyTypeVirtualMtime <int32 folder ID> <file name> = dbMtime
|
|
|
|
KeyTypeVirtualMtime = 5
|
|
|
|
|
|
|
|
// KeyTypeFolderIdx <int32 id> = string value
|
|
|
|
KeyTypeFolderIdx = 6
|
|
|
|
|
|
|
|
// KeyTypeDeviceIdx <int32 id> = string value
|
|
|
|
KeyTypeDeviceIdx = 7
|
|
|
|
|
|
|
|
// KeyTypeIndexID <int32 device ID> <int32 folder ID> = protocol.IndexID
|
|
|
|
KeyTypeIndexID = 8
|
|
|
|
|
|
|
|
// KeyTypeFolderMeta <int32 folder ID> = CountsSet
|
|
|
|
KeyTypeFolderMeta = 9
|
|
|
|
|
|
|
|
// KeyTypeMiscData <some string> = some value
|
|
|
|
KeyTypeMiscData = 10
|
|
|
|
|
|
|
|
// KeyTypeSequence <int32 folder ID> <int64 sequence number> = KeyTypeDevice key
|
|
|
|
KeyTypeSequence = 11
|
|
|
|
|
|
|
|
// KeyTypeNeed <int32 folder ID> <file name> = <nothing>
|
|
|
|
KeyTypeNeed = 12
|
2018-09-18 08:41:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type keyer interface {
|
|
|
|
// device file key stuff
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateDeviceFileKey(key, folder, device, name []byte) (deviceFileKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
NameFromDeviceFileKey(key []byte) []byte
|
|
|
|
DeviceFromDeviceFileKey(key []byte) ([]byte, bool)
|
|
|
|
FolderFromDeviceFileKey(key []byte) ([]byte, bool)
|
|
|
|
|
|
|
|
// global version key stuff
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateGlobalVersionKey(key, folder, name []byte) (globalVersionKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
NameFromGlobalVersionKey(key []byte) []byte
|
|
|
|
FolderFromGlobalVersionKey(key []byte) ([]byte, bool)
|
|
|
|
|
2019-01-23 09:22:33 +00:00
|
|
|
// block map key stuff (former BlockMap)
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateBlockMapKey(key, folder, hash, name []byte) (blockMapKey, error)
|
2019-01-23 09:22:33 +00:00
|
|
|
NameFromBlockMapKey(key []byte) []byte
|
|
|
|
|
2018-09-18 08:41:06 +00:00
|
|
|
// file need index
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateNeedFileKey(key, folder, name []byte) (needFileKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
|
|
|
|
// file sequence index
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateSequenceKey(key, folder []byte, seq int64) (sequenceKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
SequenceFromSequenceKey(key []byte) int64
|
|
|
|
|
|
|
|
// index IDs
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateIndexIDKey(key, device, folder []byte) (indexIDKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
DeviceFromIndexIDKey(key []byte) ([]byte, bool)
|
|
|
|
|
|
|
|
// Mtimes
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateMtimesKey(key, folder []byte) (mtimesKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
|
|
|
|
// Folder metadata
|
2019-11-29 08:11:52 +00:00
|
|
|
GenerateFolderMetaKey(key, folder []byte) (folderMetaKey, error)
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// defaultKeyer implements our key scheme. It needs folder and device
|
|
|
|
// indexes.
|
|
|
|
type defaultKeyer struct {
|
|
|
|
folderIdx *smallIndex
|
|
|
|
deviceIdx *smallIndex
|
|
|
|
}
|
|
|
|
|
|
|
|
func newDefaultKeyer(folderIdx, deviceIdx *smallIndex) defaultKeyer {
|
|
|
|
return defaultKeyer{
|
|
|
|
folderIdx: folderIdx,
|
|
|
|
deviceIdx: deviceIdx,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type deviceFileKey []byte
|
|
|
|
|
2018-10-11 10:09:44 +00:00
|
|
|
func (k deviceFileKey) WithoutNameAndDevice() []byte {
|
|
|
|
return k[:keyPrefixLen+keyFolderLen]
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateDeviceFileKey(key, folder, device, name []byte) (deviceFileKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
deviceID, err := k.deviceIdx.ID(device)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen+keyDeviceLen+len(name))
|
|
|
|
key[0] = KeyTypeDevice
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen+keyFolderLen:], deviceID)
|
2018-09-18 08:41:06 +00:00
|
|
|
copy(key[keyPrefixLen+keyFolderLen+keyDeviceLen:], name)
|
2019-11-29 08:11:52 +00:00
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) NameFromDeviceFileKey(key []byte) []byte {
|
|
|
|
return key[keyPrefixLen+keyFolderLen+keyDeviceLen:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) DeviceFromDeviceFileKey(key []byte) ([]byte, bool) {
|
|
|
|
return k.deviceIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen+keyFolderLen:]))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) FolderFromDeviceFileKey(key []byte) ([]byte, bool) {
|
|
|
|
return k.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
|
|
|
|
}
|
|
|
|
|
|
|
|
type globalVersionKey []byte
|
|
|
|
|
|
|
|
func (k globalVersionKey) WithoutName() []byte {
|
|
|
|
return k[:keyPrefixLen+keyFolderLen]
|
|
|
|
}
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateGlobalVersionKey(key, folder, name []byte) (globalVersionKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen+len(name))
|
|
|
|
key[0] = KeyTypeGlobal
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
2018-09-18 08:41:06 +00:00
|
|
|
copy(key[keyPrefixLen+keyFolderLen:], name)
|
2019-11-29 08:11:52 +00:00
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) NameFromGlobalVersionKey(key []byte) []byte {
|
|
|
|
return key[keyPrefixLen+keyFolderLen:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) FolderFromGlobalVersionKey(key []byte) ([]byte, bool) {
|
|
|
|
return k.folderIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
|
|
|
|
}
|
|
|
|
|
2019-01-23 09:22:33 +00:00
|
|
|
type blockMapKey []byte
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateBlockMapKey(key, folder, hash, name []byte) (blockMapKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-23 09:22:33 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen+keyHashLen+len(name))
|
|
|
|
key[0] = KeyTypeBlock
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
2019-01-23 09:22:33 +00:00
|
|
|
copy(key[keyPrefixLen+keyFolderLen:], hash)
|
|
|
|
copy(key[keyPrefixLen+keyFolderLen+keyHashLen:], name)
|
2019-11-29 08:11:52 +00:00
|
|
|
return key, nil
|
2019-01-23 09:22:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) NameFromBlockMapKey(key []byte) []byte {
|
|
|
|
return key[keyPrefixLen+keyFolderLen+keyHashLen:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k blockMapKey) WithoutHashAndName() []byte {
|
|
|
|
return k[:keyPrefixLen+keyFolderLen]
|
|
|
|
}
|
|
|
|
|
2018-09-18 08:41:06 +00:00
|
|
|
type needFileKey []byte
|
|
|
|
|
|
|
|
func (k needFileKey) WithoutName() []byte {
|
|
|
|
return k[:keyPrefixLen+keyFolderLen]
|
|
|
|
}
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateNeedFileKey(key, folder, name []byte) (needFileKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen+len(name))
|
|
|
|
key[0] = KeyTypeNeed
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
2018-09-18 08:41:06 +00:00
|
|
|
copy(key[keyPrefixLen+keyFolderLen:], name)
|
2019-11-29 08:11:52 +00:00
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type sequenceKey []byte
|
|
|
|
|
|
|
|
func (k sequenceKey) WithoutSequence() []byte {
|
|
|
|
return k[:keyPrefixLen+keyFolderLen]
|
|
|
|
}
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateSequenceKey(key, folder []byte, seq int64) (sequenceKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen+keySequenceLen)
|
|
|
|
key[0] = KeyTypeSequence
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
2018-09-18 08:41:06 +00:00
|
|
|
binary.BigEndian.PutUint64(key[keyPrefixLen+keyFolderLen:], uint64(seq))
|
2019-11-29 08:11:52 +00:00
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) SequenceFromSequenceKey(key []byte) int64 {
|
|
|
|
return int64(binary.BigEndian.Uint64(key[keyPrefixLen+keyFolderLen:]))
|
|
|
|
}
|
|
|
|
|
|
|
|
type indexIDKey []byte
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateIndexIDKey(key, device, folder []byte) (indexIDKey, error) {
|
|
|
|
deviceID, err := k.deviceIdx.ID(device)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyDeviceLen+keyFolderLen)
|
|
|
|
key[0] = KeyTypeIndexID
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], deviceID)
|
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen+keyDeviceLen:], folderID)
|
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (k defaultKeyer) DeviceFromIndexIDKey(key []byte) ([]byte, bool) {
|
|
|
|
return k.deviceIdx.Val(binary.BigEndian.Uint32(key[keyPrefixLen:]))
|
|
|
|
}
|
|
|
|
|
|
|
|
type mtimesKey []byte
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateMtimesKey(key, folder []byte) (mtimesKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen)
|
|
|
|
key[0] = KeyTypeVirtualMtime
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type folderMetaKey []byte
|
|
|
|
|
2019-11-29 08:11:52 +00:00
|
|
|
func (k defaultKeyer) GenerateFolderMetaKey(key, folder []byte) (folderMetaKey, error) {
|
|
|
|
folderID, err := k.folderIdx.ID(folder)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-18 08:41:06 +00:00
|
|
|
key = resize(key, keyPrefixLen+keyFolderLen)
|
|
|
|
key[0] = KeyTypeFolderMeta
|
2019-11-29 08:11:52 +00:00
|
|
|
binary.BigEndian.PutUint32(key[keyPrefixLen:], folderID)
|
|
|
|
return key, nil
|
2018-09-18 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// resize returns a byte slice of the specified size, reusing bs if possible
|
|
|
|
func resize(bs []byte, size int) []byte {
|
|
|
|
if cap(bs) < size {
|
|
|
|
return make([]byte, size)
|
|
|
|
}
|
|
|
|
return bs[:size]
|
|
|
|
}
|