mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-05 16:12:20 +00:00
lib/ur: Track new folder feature usage (#6803)
This commit is contained in:
parent
d7fc7008af
commit
9d4a700829
@ -13,6 +13,8 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/syncthing/syncthing/lib/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Report struct {
|
type Report struct {
|
||||||
@ -119,6 +121,13 @@ type Report struct {
|
|||||||
PullOrder map[string]int `json:"pullOrder,omitempty" since:"3"`
|
PullOrder map[string]int `json:"pullOrder,omitempty" since:"3"`
|
||||||
FilesystemType map[string]int `json:"filesystemType,omitempty" since:"3"`
|
FilesystemType map[string]int `json:"filesystemType,omitempty" since:"3"`
|
||||||
FsWatcherDelays []int `json:"fsWatcherDelays,omitempty" since:"3"`
|
FsWatcherDelays []int `json:"fsWatcherDelays,omitempty" since:"3"`
|
||||||
|
CustomMarkerName int `json:"customMarkerName,omitempty" since:"3"`
|
||||||
|
CopyOwnershipFromParent int `json:"copyOwnershipFromParent,omitempty" since:"3"`
|
||||||
|
ModTimeWindowS []int `json:"modTimeWindowS,omitempty" since:"3"`
|
||||||
|
MaxConcurrentWrites []int `json:"maxConcurrentWrites,omitempty" since:"3"`
|
||||||
|
DisableFsync int `json:"disableFsync,omitempty" since:"3"`
|
||||||
|
BlockPullOrder map[string]int `json:"blockPullOrder,omitempty" since:"3"`
|
||||||
|
CopyRangeMethod map[string]int `json:"copyRangeMethod,omitempty" since:"3"`
|
||||||
} `json:"folderUsesV3,omitempty" since:"3"`
|
} `json:"folderUsesV3,omitempty" since:"3"`
|
||||||
|
|
||||||
GUIStats struct {
|
GUIStats struct {
|
||||||
@ -164,12 +173,7 @@ type Report struct {
|
|||||||
|
|
||||||
func New() *Report {
|
func New() *Report {
|
||||||
r := &Report{}
|
r := &Report{}
|
||||||
r.FolderUsesV3.PullOrder = make(map[string]int)
|
util.FillNil(r)
|
||||||
r.FolderUsesV3.FilesystemType = make(map[string]int)
|
|
||||||
r.GUIStats.Theme = make(map[string]int)
|
|
||||||
r.TransportStats = make(map[string]int)
|
|
||||||
r.RescanIntvs = make([]int, 0)
|
|
||||||
r.FolderUsesV3.FsWatcherDelays = make([]int, 0)
|
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -256,6 +256,19 @@ func (s *Service) reportData(ctx context.Context, urVersion int, preview bool) (
|
|||||||
report.FolderUsesV3.PullOrder[cfg.Order.String()]++
|
report.FolderUsesV3.PullOrder[cfg.Order.String()]++
|
||||||
report.FolderUsesV3.FilesystemType[cfg.FilesystemType.String()]++
|
report.FolderUsesV3.FilesystemType[cfg.FilesystemType.String()]++
|
||||||
report.FolderUsesV3.FsWatcherDelays = append(report.FolderUsesV3.FsWatcherDelays, cfg.FSWatcherDelayS)
|
report.FolderUsesV3.FsWatcherDelays = append(report.FolderUsesV3.FsWatcherDelays, cfg.FSWatcherDelayS)
|
||||||
|
if cfg.MarkerName != config.DefaultMarkerName {
|
||||||
|
report.FolderUsesV3.CustomMarkerName++
|
||||||
|
}
|
||||||
|
if cfg.CopyOwnershipFromParent {
|
||||||
|
report.FolderUsesV3.CopyOwnershipFromParent++
|
||||||
|
}
|
||||||
|
report.FolderUsesV3.ModTimeWindowS = append(report.FolderUsesV3.ModTimeWindowS, int(cfg.ModTimeWindow().Seconds()))
|
||||||
|
report.FolderUsesV3.MaxConcurrentWrites = append(report.FolderUsesV3.MaxConcurrentWrites, cfg.MaxConcurrentWrites)
|
||||||
|
if cfg.DisableFsync {
|
||||||
|
report.FolderUsesV3.DisableFsync++
|
||||||
|
}
|
||||||
|
report.FolderUsesV3.BlockPullOrder[cfg.BlockPullOrder.String()]++
|
||||||
|
report.FolderUsesV3.CopyRangeMethod[cfg.CopyRangeMethod.String()]++
|
||||||
}
|
}
|
||||||
sort.Ints(report.FolderUsesV3.FsWatcherDelays)
|
sort.Ints(report.FolderUsesV3.FsWatcherDelays)
|
||||||
|
|
||||||
|
@ -137,6 +137,38 @@ func UniqueTrimmedStrings(ss []string) []string {
|
|||||||
return us
|
return us
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func FillNil(data interface{}) {
|
||||||
|
s := reflect.ValueOf(data).Elem()
|
||||||
|
for i := 0; i < s.NumField(); i++ {
|
||||||
|
f := s.Field(i)
|
||||||
|
|
||||||
|
for f.Kind() == reflect.Ptr && f.IsZero() && f.CanSet() {
|
||||||
|
newValue := reflect.New(f.Type().Elem())
|
||||||
|
f.Set(newValue)
|
||||||
|
f = f.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
if f.CanSet() {
|
||||||
|
if f.IsZero() {
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.Map:
|
||||||
|
f.Set(reflect.MakeMap(f.Type()))
|
||||||
|
case reflect.Slice:
|
||||||
|
f.Set(reflect.MakeSlice(f.Type(), 0, 0))
|
||||||
|
case reflect.Chan:
|
||||||
|
f.Set(reflect.MakeChan(f.Type(), 0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if f.Kind() == reflect.Struct && f.CanAddr() {
|
||||||
|
if addr := f.Addr(); addr.CanInterface() {
|
||||||
|
FillNil(addr.Interface())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// FillNilSlices sets default value on slices that are still nil.
|
// FillNilSlices sets default value on slices that are still nil.
|
||||||
func FillNilSlices(data interface{}) error {
|
func FillNilSlices(data interface{}) error {
|
||||||
s := reflect.ValueOf(data).Elem()
|
s := reflect.ValueOf(data).Elem()
|
||||||
|
@ -287,3 +287,128 @@ func TestUtilStopTwicePanic(t *testing.T) {
|
|||||||
}()
|
}()
|
||||||
s.Stop()
|
s.Stop()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestFillNil(t *testing.T) {
|
||||||
|
type A struct {
|
||||||
|
Slice []int
|
||||||
|
Map map[string]string
|
||||||
|
Chan chan int
|
||||||
|
}
|
||||||
|
|
||||||
|
type B struct {
|
||||||
|
Slice *[]int
|
||||||
|
Map *map[string]string
|
||||||
|
Chan *chan int
|
||||||
|
}
|
||||||
|
|
||||||
|
type C struct {
|
||||||
|
A A
|
||||||
|
B *B
|
||||||
|
D *****[]int
|
||||||
|
}
|
||||||
|
|
||||||
|
c := C{}
|
||||||
|
FillNil(&c)
|
||||||
|
|
||||||
|
if c.A.Slice == nil {
|
||||||
|
t.Error("c.A.Slice")
|
||||||
|
}
|
||||||
|
if c.A.Map == nil {
|
||||||
|
t.Error("c.A.Slice")
|
||||||
|
}
|
||||||
|
if c.A.Chan == nil {
|
||||||
|
t.Error("c.A.Chan")
|
||||||
|
}
|
||||||
|
if c.B == nil {
|
||||||
|
t.Error("c.B")
|
||||||
|
}
|
||||||
|
if c.B.Slice == nil {
|
||||||
|
t.Error("c.B.Slice")
|
||||||
|
}
|
||||||
|
if c.B.Map == nil {
|
||||||
|
t.Error("c.B.Slice")
|
||||||
|
}
|
||||||
|
if c.B.Chan == nil {
|
||||||
|
t.Error("c.B.Chan")
|
||||||
|
}
|
||||||
|
if *c.B.Slice == nil {
|
||||||
|
t.Error("*c.B.Slice")
|
||||||
|
}
|
||||||
|
if *c.B.Map == nil {
|
||||||
|
t.Error("*c.B.Slice")
|
||||||
|
}
|
||||||
|
if *c.B.Chan == nil {
|
||||||
|
t.Error("*c.B.Chan")
|
||||||
|
}
|
||||||
|
if *****c.D == nil {
|
||||||
|
t.Error("c.D")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFillNilDoesNotBulldozeSetFields(t *testing.T) {
|
||||||
|
type A struct {
|
||||||
|
Slice []int
|
||||||
|
Map map[string]string
|
||||||
|
Chan chan int
|
||||||
|
}
|
||||||
|
|
||||||
|
type B struct {
|
||||||
|
Slice *[]int
|
||||||
|
Map *map[string]string
|
||||||
|
Chan *chan int
|
||||||
|
}
|
||||||
|
|
||||||
|
type C struct {
|
||||||
|
A A
|
||||||
|
B *B
|
||||||
|
D **[]int
|
||||||
|
}
|
||||||
|
|
||||||
|
ch := make(chan int, 10)
|
||||||
|
d := make([]int, 10)
|
||||||
|
dd := &d
|
||||||
|
|
||||||
|
c := C{
|
||||||
|
A: A{
|
||||||
|
Slice: []int{1},
|
||||||
|
Map: map[string]string{
|
||||||
|
"k": "v",
|
||||||
|
},
|
||||||
|
Chan: make(chan int, 10),
|
||||||
|
},
|
||||||
|
B: &B{
|
||||||
|
Slice: &[]int{1},
|
||||||
|
Map: &map[string]string{
|
||||||
|
"k": "v",
|
||||||
|
},
|
||||||
|
Chan: &ch,
|
||||||
|
},
|
||||||
|
D: &dd,
|
||||||
|
}
|
||||||
|
FillNil(&c)
|
||||||
|
|
||||||
|
if len(c.A.Slice) != 1 {
|
||||||
|
t.Error("c.A.Slice")
|
||||||
|
}
|
||||||
|
if len(c.A.Map) != 1 {
|
||||||
|
t.Error("c.A.Slice")
|
||||||
|
}
|
||||||
|
if cap(c.A.Chan) != 10 {
|
||||||
|
t.Error("c.A.Chan")
|
||||||
|
}
|
||||||
|
if c.B == nil {
|
||||||
|
t.Error("c.B")
|
||||||
|
}
|
||||||
|
if len(*c.B.Slice) != 1 {
|
||||||
|
t.Error("c.B.Slice")
|
||||||
|
}
|
||||||
|
if len(*c.B.Map) != 1 {
|
||||||
|
t.Error("c.B.Slice")
|
||||||
|
}
|
||||||
|
if cap(*c.B.Chan) != 10 {
|
||||||
|
t.Error("c.B.Chan")
|
||||||
|
}
|
||||||
|
if cap(**c.D) != 10 {
|
||||||
|
t.Error("c.D")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user