2021-04-29 22:01:46 +02:00
|
|
|
// Copyright (C) 2021 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 "github.com/syncthing/syncthing/lib/protocol"
|
|
|
|
|
|
|
|
// How many files to send in each Index/IndexUpdate message.
|
|
|
|
const (
|
|
|
|
MaxBatchSizeBytes = 250 * 1024 // Aim for making index messages no larger than 250 KiB (uncompressed)
|
|
|
|
MaxBatchSizeFiles = 1000 // Either way, don't include more files than this
|
|
|
|
)
|
|
|
|
|
|
|
|
// FileInfoBatch is a utility to do file operations on the database in suitably
|
|
|
|
// sized batches.
|
|
|
|
type FileInfoBatch struct {
|
|
|
|
infos []protocol.FileInfo
|
|
|
|
size int
|
|
|
|
flushFn func([]protocol.FileInfo) error
|
2024-08-28 15:00:19 +02:00
|
|
|
error error
|
2021-04-29 22:01:46 +02:00
|
|
|
}
|
|
|
|
|
2024-02-10 19:16:27 +01:00
|
|
|
// NewFileInfoBatch returns a new FileInfoBatch that calls fn when it's time
|
2024-08-28 15:00:19 +02:00
|
|
|
// to flush. Errors from the flush function are considered non-recoverable;
|
|
|
|
// once an error is returned the flush function wil not be called again, and
|
|
|
|
// any further calls to Flush will return the same error (unless Reset is
|
|
|
|
// called).
|
2021-04-29 22:01:46 +02:00
|
|
|
func NewFileInfoBatch(fn func([]protocol.FileInfo) error) *FileInfoBatch {
|
2024-02-10 19:16:27 +01:00
|
|
|
return &FileInfoBatch{flushFn: fn}
|
2021-04-29 22:01:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) SetFlushFunc(fn func([]protocol.FileInfo) error) {
|
|
|
|
b.flushFn = fn
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) Append(f protocol.FileInfo) {
|
2024-08-28 15:00:19 +02:00
|
|
|
if b.error != nil {
|
|
|
|
panic("bug: calling append on a failed batch")
|
|
|
|
}
|
2024-02-10 19:16:27 +01:00
|
|
|
if b.infos == nil {
|
|
|
|
b.infos = make([]protocol.FileInfo, 0, MaxBatchSizeFiles)
|
|
|
|
}
|
2021-04-29 22:01:46 +02:00
|
|
|
b.infos = append(b.infos, f)
|
|
|
|
b.size += f.ProtoSize()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) Full() bool {
|
|
|
|
return len(b.infos) >= MaxBatchSizeFiles || b.size >= MaxBatchSizeBytes
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) FlushIfFull() error {
|
2024-08-28 15:00:19 +02:00
|
|
|
if b.error != nil {
|
|
|
|
return b.error
|
|
|
|
}
|
2021-04-29 22:01:46 +02:00
|
|
|
if b.Full() {
|
|
|
|
return b.Flush()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) Flush() error {
|
2024-08-28 15:00:19 +02:00
|
|
|
if b.error != nil {
|
|
|
|
return b.error
|
|
|
|
}
|
2021-04-29 22:01:46 +02:00
|
|
|
if len(b.infos) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err := b.flushFn(b.infos); err != nil {
|
2024-08-28 15:00:19 +02:00
|
|
|
b.error = err
|
2021-04-29 22:01:46 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.Reset()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) Reset() {
|
2024-02-10 19:16:27 +01:00
|
|
|
b.infos = nil
|
2024-08-28 15:00:19 +02:00
|
|
|
b.error = nil
|
2021-04-29 22:01:46 +02:00
|
|
|
b.size = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *FileInfoBatch) Size() int {
|
|
|
|
return b.size
|
|
|
|
}
|