2015-05-13 14:57:29 +00:00
|
|
|
// 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 (
|
2016-01-03 19:08:19 +01:00
|
|
|
"encoding/binary"
|
2015-05-13 14:57:29 +00:00
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// This type encapsulates a repository of mtimes for platforms where file mtimes
|
|
|
|
// can't be set to arbitrary values. For this to work, we need to store both
|
|
|
|
// the mtime we tried to set (the "actual" mtime) as well as the mtime the file
|
|
|
|
// has when we're done touching it (the "disk" mtime) so that we can tell if it
|
|
|
|
// was changed. So in GetMtime(), it's not sufficient that the record exists --
|
|
|
|
// the argument must also equal the "disk" mtime in the record, otherwise it's
|
|
|
|
// been touched locally and the "disk" mtime is actually correct.
|
|
|
|
|
|
|
|
type VirtualMtimeRepo struct {
|
|
|
|
ns *NamespacedKV
|
|
|
|
}
|
|
|
|
|
2015-10-31 12:31:25 +01:00
|
|
|
func NewVirtualMtimeRepo(ldb *Instance, folder string) *VirtualMtimeRepo {
|
2016-01-03 19:08:19 +01:00
|
|
|
var prefix [5]byte // key type + 4 bytes folder idx number
|
|
|
|
prefix[0] = KeyTypeVirtualMtime
|
|
|
|
binary.BigEndian.PutUint32(prefix[1:], ldb.folderIdx.ID([]byte(folder)))
|
2015-05-13 14:57:29 +00:00
|
|
|
|
|
|
|
return &VirtualMtimeRepo{
|
2016-01-03 19:08:19 +01:00
|
|
|
ns: NewNamespacedKV(ldb, string(prefix[:])),
|
2015-05-13 14:57:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *VirtualMtimeRepo) UpdateMtime(path string, diskMtime, actualMtime time.Time) {
|
2015-10-03 17:25:21 +02:00
|
|
|
l.Debugf("virtual mtime: storing values for path:%s disk:%v actual:%v", path, diskMtime, actualMtime)
|
2015-05-13 14:57:29 +00:00
|
|
|
|
|
|
|
diskBytes, _ := diskMtime.MarshalBinary()
|
|
|
|
actualBytes, _ := actualMtime.MarshalBinary()
|
|
|
|
|
|
|
|
data := append(diskBytes, actualBytes...)
|
|
|
|
|
|
|
|
r.ns.PutBytes(path, data)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *VirtualMtimeRepo) GetMtime(path string, diskMtime time.Time) time.Time {
|
2015-05-14 08:26:21 +02:00
|
|
|
data, exists := r.ns.Bytes(path)
|
|
|
|
if !exists {
|
2015-11-11 21:20:34 -05:00
|
|
|
// Absence of debug print is significant enough in itself here
|
2015-05-14 08:26:21 +02:00
|
|
|
return diskMtime
|
|
|
|
}
|
2015-05-13 14:57:29 +00:00
|
|
|
|
2015-05-14 08:26:21 +02:00
|
|
|
var mtime time.Time
|
|
|
|
if err := mtime.UnmarshalBinary(data[:len(data)/2]); err != nil {
|
|
|
|
panic(fmt.Sprintf("Can't unmarshal stored mtime at path %s: %v", path, err))
|
|
|
|
}
|
2015-05-13 14:57:29 +00:00
|
|
|
|
2015-05-14 08:26:21 +02:00
|
|
|
if mtime.Equal(diskMtime) {
|
|
|
|
if err := mtime.UnmarshalBinary(data[len(data)/2:]); err != nil {
|
|
|
|
panic(fmt.Sprintf("Can't unmarshal stored mtime at path %s: %v", path, err))
|
2015-05-13 14:57:29 +00:00
|
|
|
}
|
|
|
|
|
2015-10-03 17:25:21 +02:00
|
|
|
l.Debugf("virtual mtime: return %v instead of %v for path: %s", mtime, diskMtime, path)
|
2015-05-14 08:26:21 +02:00
|
|
|
return mtime
|
2015-05-13 14:57:29 +00:00
|
|
|
}
|
|
|
|
|
2015-10-03 17:25:21 +02:00
|
|
|
l.Debugf("virtual mtime: record exists, but mismatch inDisk: %v dbDisk: %v for path: %s", diskMtime, mtime, path)
|
2015-05-13 14:57:29 +00:00
|
|
|
return diskMtime
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *VirtualMtimeRepo) DeleteMtime(path string) {
|
|
|
|
r.ns.Delete(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *VirtualMtimeRepo) Drop() {
|
|
|
|
r.ns.Reset()
|
|
|
|
}
|