|
|
@@ -16,96 +16,55 @@
|
|
|
package stats
|
|
|
|
|
|
import (
|
|
|
- "encoding/binary"
|
|
|
"time"
|
|
|
|
|
|
+ "github.com/syncthing/syncthing/internal/db"
|
|
|
"github.com/syndtr/goleveldb/leveldb"
|
|
|
)
|
|
|
|
|
|
-const (
|
|
|
- folderStatisticTypeLastFile = iota
|
|
|
-)
|
|
|
-
|
|
|
-var folderStatisticsTypes = []byte{
|
|
|
- folderStatisticTypeLastFile,
|
|
|
-}
|
|
|
-
|
|
|
type FolderStatistics struct {
|
|
|
- LastFile *LastFile
|
|
|
+ LastFile LastFile
|
|
|
}
|
|
|
|
|
|
type FolderStatisticsReference struct {
|
|
|
- db *leveldb.DB
|
|
|
+ ns *db.NamespacedKV
|
|
|
folder string
|
|
|
}
|
|
|
|
|
|
-func NewFolderStatisticsReference(db *leveldb.DB, folder string) *FolderStatisticsReference {
|
|
|
+type LastFile struct {
|
|
|
+ At time.Time
|
|
|
+ Filename string
|
|
|
+}
|
|
|
+
|
|
|
+func NewFolderStatisticsReference(ldb *leveldb.DB, folder string) *FolderStatisticsReference {
|
|
|
+ prefix := string(db.KeyTypeFolderStatistic) + folder
|
|
|
return &FolderStatisticsReference{
|
|
|
- db: db,
|
|
|
+ ns: db.NewNamespacedKV(ldb, prefix),
|
|
|
folder: folder,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-func (s *FolderStatisticsReference) key(stat byte) []byte {
|
|
|
- k := make([]byte, 1+1+64)
|
|
|
- k[0] = keyTypeFolderStatistic
|
|
|
- k[1] = stat
|
|
|
- copy(k[1+1:], s.folder[:])
|
|
|
- return k
|
|
|
-}
|
|
|
-
|
|
|
-func (s *FolderStatisticsReference) GetLastFile() *LastFile {
|
|
|
- value, err := s.db.Get(s.key(folderStatisticTypeLastFile), nil)
|
|
|
- if err != nil {
|
|
|
- if err != leveldb.ErrNotFound {
|
|
|
- l.Warnln("FolderStatisticsReference: Failed loading last file filename value for", s.folder, ":", err)
|
|
|
- }
|
|
|
- return nil
|
|
|
+func (s *FolderStatisticsReference) GetLastFile() LastFile {
|
|
|
+ at, ok := s.ns.Time("lastFileAt")
|
|
|
+ if !ok {
|
|
|
+ return LastFile{}
|
|
|
}
|
|
|
-
|
|
|
- file := LastFile{}
|
|
|
- err = file.UnmarshalBinary(value)
|
|
|
- if err != nil {
|
|
|
- l.Warnln("FolderStatisticsReference: Failed loading last file value for", s.folder, ":", err)
|
|
|
- return nil
|
|
|
+ file, ok := s.ns.String("lastFileName")
|
|
|
+ if !ok {
|
|
|
+ return LastFile{}
|
|
|
+ }
|
|
|
+ return LastFile{
|
|
|
+ At: at,
|
|
|
+ Filename: file,
|
|
|
}
|
|
|
- return &file
|
|
|
}
|
|
|
|
|
|
func (s *FolderStatisticsReference) ReceivedFile(filename string) {
|
|
|
- f := LastFile{
|
|
|
- Filename: filename,
|
|
|
- At: time.Now(),
|
|
|
- }
|
|
|
if debug {
|
|
|
- l.Debugln("stats.FolderStatisticsReference.ReceivedFile:", s.folder)
|
|
|
- }
|
|
|
-
|
|
|
- value, err := f.MarshalBinary()
|
|
|
- if err != nil {
|
|
|
- l.Warnln("FolderStatisticsReference: Failed serializing last file value for", s.folder, ":", err)
|
|
|
- return
|
|
|
- }
|
|
|
-
|
|
|
- err = s.db.Put(s.key(folderStatisticTypeLastFile), value, nil)
|
|
|
- if err != nil {
|
|
|
- l.Warnln("Failed update last file value for", s.folder, ":", err)
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-// Never called, maybe because it's worth while to keep the data
|
|
|
-// or maybe because we have no easy way of knowing that a folder has been removed.
|
|
|
-func (s *FolderStatisticsReference) Delete() error {
|
|
|
- for _, stype := range folderStatisticsTypes {
|
|
|
- err := s.db.Delete(s.key(stype), nil)
|
|
|
- if debug && err == nil {
|
|
|
- l.Debugln("stats.FolderStatisticsReference.Delete:", s.folder, stype)
|
|
|
- }
|
|
|
- if err != nil && err != leveldb.ErrNotFound {
|
|
|
- return err
|
|
|
- }
|
|
|
+ l.Debugln("stats.FolderStatisticsReference.ReceivedFile:", s.folder, filename)
|
|
|
}
|
|
|
- return nil
|
|
|
+ s.ns.PutTime("lastFileAt", time.Now())
|
|
|
+ s.ns.PutString("lastFileName", filename)
|
|
|
}
|
|
|
|
|
|
func (s *FolderStatisticsReference) GetStatistics() FolderStatistics {
|
|
|
@@ -113,21 +72,3 @@ func (s *FolderStatisticsReference) GetStatistics() FolderStatistics {
|
|
|
LastFile: s.GetLastFile(),
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
-type LastFile struct {
|
|
|
- At time.Time
|
|
|
- Filename string
|
|
|
-}
|
|
|
-
|
|
|
-func (f *LastFile) MarshalBinary() ([]byte, error) {
|
|
|
- buf := make([]byte, 8+len(f.Filename))
|
|
|
- binary.BigEndian.PutUint64(buf[:8], uint64(f.At.Unix()))
|
|
|
- copy(buf[8:], []byte(f.Filename))
|
|
|
- return buf, nil
|
|
|
-}
|
|
|
-
|
|
|
-func (f *LastFile) UnmarshalBinary(buf []byte) error {
|
|
|
- f.At = time.Unix(int64(binary.BigEndian.Uint64(buf[:8])), 0)
|
|
|
- f.Filename = string(buf[8:])
|
|
|
- return nil
|
|
|
-}
|