2014-11-16 21:13:20 +01:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
2014-10-06 21:57:33 +01:00
|
|
|
//
|
2015-03-07 21:36:35 +01:00
|
|
|
// 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,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2014-10-06 21:57:33 +01:00
|
|
|
|
2015-01-12 14:50:30 +01:00
|
|
|
package db
|
2014-10-06 21:57:33 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
2015-06-03 09:47:39 +02:00
|
|
|
"fmt"
|
2014-10-06 21:57:33 +01:00
|
|
|
|
2015-08-06 11:29:25 +02:00
|
|
|
"github.com/syncthing/syncthing/lib/osutil"
|
2014-10-06 21:57:33 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var blockFinder *BlockFinder
|
|
|
|
|
|
|
|
type BlockFinder struct {
|
2019-12-02 08:18:04 +01:00
|
|
|
db *Lowlevel
|
2014-10-06 21:57:33 +01:00
|
|
|
}
|
|
|
|
|
2018-10-10 11:34:24 +02:00
|
|
|
func NewBlockFinder(db *Lowlevel) *BlockFinder {
|
2014-10-06 21:57:33 +01:00
|
|
|
if blockFinder != nil {
|
|
|
|
return blockFinder
|
|
|
|
}
|
|
|
|
|
2019-01-23 10:22:33 +01:00
|
|
|
return &BlockFinder{
|
2019-12-02 08:18:04 +01:00
|
|
|
db: db,
|
2014-10-06 21:57:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-03 09:47:39 +02:00
|
|
|
func (f *BlockFinder) String() string {
|
|
|
|
return fmt.Sprintf("BlockFinder@%p", f)
|
2014-10-06 21:57:33 +01:00
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// Iterate takes an iterator function which iterates over all matching blocks
|
|
|
|
// for the given hash. The iterator function has to return either true (if
|
|
|
|
// they are happy with the block) or false to continue iterating for whatever
|
|
|
|
// reason. The iterator finally returns the result, whether or not a
|
|
|
|
// satisfying block was eventually found.
|
2015-09-04 12:01:00 +02:00
|
|
|
func (f *BlockFinder) Iterate(folders []string, hash []byte, iterFn func(string, string, int32) bool) bool {
|
2019-11-29 09:11:52 +01:00
|
|
|
t, err := f.db.newReadOnlyTransaction()
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
2019-01-23 10:22:33 +01:00
|
|
|
defer t.close()
|
|
|
|
|
2015-10-21 22:55:40 +02:00
|
|
|
var key []byte
|
2014-10-06 21:57:33 +01:00
|
|
|
for _, folder := range folders {
|
2019-11-29 09:11:52 +01:00
|
|
|
key, err = f.db.keyer.GenerateBlockMapKey(key, []byte(folder), hash, nil)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
iter, err := t.NewPrefixIterator(key)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
2014-10-06 21:57:33 +01:00
|
|
|
|
|
|
|
for iter.Next() && iter.Error() == nil {
|
2019-01-23 10:22:33 +01:00
|
|
|
file := string(f.db.keyer.NameFromBlockMapKey(iter.Key()))
|
2015-01-18 02:12:06 +01:00
|
|
|
index := int32(binary.BigEndian.Uint32(iter.Value()))
|
2014-11-05 23:41:51 +00:00
|
|
|
if iterFn(folder, osutil.NativeFilename(file), index) {
|
2019-01-23 10:22:33 +01:00
|
|
|
iter.Release()
|
2014-10-06 21:57:33 +01:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
2019-01-23 10:22:33 +01:00
|
|
|
iter.Release()
|
2014-10-06 21:57:33 +01:00
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|