accounting: add limits and listing to stats groups

This commit is contained in:
Aleksandar Jankovic 2019-07-23 17:45:07 +02:00 committed by Nick Craig-Wood
parent 53a1a0e3ef
commit 68e641f6cf
3 changed files with 55 additions and 10 deletions

View file

@ -13,6 +13,14 @@ const globalStats = "global_stats"
var groups *statsGroups var groups *statsGroups
func listStats(ctx context.Context, in rc.Params) (rc.Params, error) {
out := make(rc.Params)
out["groups"] = groups.names()
return out, nil
}
func remoteStats(ctx context.Context, in rc.Params) (rc.Params, error) { func remoteStats(ctx context.Context, in rc.Params) (rc.Params, error) {
// Check to see if we should filter by group. // Check to see if we should filter by group.
group, err := in.GetString("group") group, err := in.GetString("group")
@ -125,6 +133,26 @@ Returns the following values:
} }
] ]
} }
`,
})
rc.Add(rc.Call{
Path: "core/group_list",
Fn: listStats,
Title: "Returns list of stats.",
Help: `
This returns list of stats groups currently in memory.
Returns the following values:
` + "```" + `
{
"groups": an array of group names:
[
"group1",
"group2",
...
]
}
`, `,
}) })
} }
@ -182,6 +210,7 @@ func NewStatsGroup(group string) *StatsInfo {
type statsGroups struct { type statsGroups struct {
mu sync.Mutex mu sync.Mutex
m map[string]*StatsInfo m map[string]*StatsInfo
order []string
} }
// newStatsGroups makes a new statsGroups object // newStatsGroups makes a new statsGroups object
@ -192,17 +221,24 @@ func newStatsGroups() *statsGroups {
} }
// set marks the stats as belonging to a group // set marks the stats as belonging to a group
func (sg *statsGroups) set(group string, acc *StatsInfo) { func (sg *statsGroups) set(group string, stats *StatsInfo) {
sg.mu.Lock() sg.mu.Lock()
defer sg.mu.Unlock() defer sg.mu.Unlock()
sg.m[group] = acc
}
// clear discards reference to group // Limit number of groups kept in memory.
func (sg *statsGroups) clear(group string) { if len(sg.order) >= fs.Config.MaxStatsGroups {
sg.mu.Lock() group := sg.order[0]
defer sg.mu.Unlock() fs.LogPrintf(fs.LogLevelInfo, nil, "Max number of stats groups reached removing %s", group)
delete(sg.m, group) delete(sg.m, group)
r := (len(sg.order) - fs.Config.MaxStatsGroups) + 1
sg.order = sg.order[r:]
}
// Exclude global stats from
if group != globalStats {
sg.order = append(sg.order, group)
}
sg.m[group] = stats
} }
// get gets the stats for group, or nil if not found // get gets the stats for group, or nil if not found
@ -216,6 +252,12 @@ func (sg *statsGroups) get(group string) *StatsInfo {
return stats return stats
} }
func (sg *statsGroups) names() []string {
sg.mu.Lock()
defer sg.mu.Unlock()
return sg.order
}
// get gets the stats for group, or nil if not found // get gets the stats for group, or nil if not found
func (sg *statsGroups) sum() *StatsInfo { func (sg *statsGroups) sum() *StatsInfo {
sg.mu.Lock() sg.mu.Lock()

View file

@ -88,6 +88,7 @@ type ConfigInfo struct {
UseServerModTime bool UseServerModTime bool
MaxTransfer SizeSuffix MaxTransfer SizeSuffix
MaxBacklog int MaxBacklog int
MaxStatsGroups int
StatsOneLine bool StatsOneLine bool
StatsOneLineDate bool // If we want a date prefix at all StatsOneLineDate bool // If we want a date prefix at all
StatsOneLineDateFormat string // If we want to customize the prefix StatsOneLineDateFormat string // If we want to customize the prefix
@ -125,6 +126,7 @@ func NewConfig() *ConfigInfo {
c.BufferSize = SizeSuffix(16 << 20) c.BufferSize = SizeSuffix(16 << 20)
c.UserAgent = "rclone/" + Version c.UserAgent = "rclone/" + Version
c.StreamingUploadCutoff = SizeSuffix(100 * 1024) c.StreamingUploadCutoff = SizeSuffix(100 * 1024)
c.MaxStatsGroups = 1000
c.StatsFileNameLength = 45 c.StatsFileNameLength = 45
c.AskPassword = true c.AskPassword = true
c.TPSLimitBurst = 1 c.TPSLimitBurst = 1

View file

@ -91,6 +91,7 @@ func AddFlags(flagSet *pflag.FlagSet) {
flags.FVarP(flagSet, &fs.Config.Dump, "dump", "", "List of items to dump from: "+fs.DumpFlagsList) flags.FVarP(flagSet, &fs.Config.Dump, "dump", "", "List of items to dump from: "+fs.DumpFlagsList)
flags.FVarP(flagSet, &fs.Config.MaxTransfer, "max-transfer", "", "Maximum size of data to transfer.") flags.FVarP(flagSet, &fs.Config.MaxTransfer, "max-transfer", "", "Maximum size of data to transfer.")
flags.IntVarP(flagSet, &fs.Config.MaxBacklog, "max-backlog", "", fs.Config.MaxBacklog, "Maximum number of objects in sync or check backlog.") flags.IntVarP(flagSet, &fs.Config.MaxBacklog, "max-backlog", "", fs.Config.MaxBacklog, "Maximum number of objects in sync or check backlog.")
flags.IntVarP(flagSet, &fs.Config.MaxStatsGroups, "max-stats-groups", "", fs.Config.MaxStatsGroups, "Maximum number of stats groups to keep in memory. On max oldest is discarded.")
flags.BoolVarP(flagSet, &fs.Config.StatsOneLine, "stats-one-line", "", fs.Config.StatsOneLine, "Make the stats fit on one line.") flags.BoolVarP(flagSet, &fs.Config.StatsOneLine, "stats-one-line", "", fs.Config.StatsOneLine, "Make the stats fit on one line.")
flags.BoolVarP(flagSet, &fs.Config.StatsOneLineDate, "stats-one-line-date", "", fs.Config.StatsOneLineDate, "Enables --stats-one-line and add current date/time prefix.") flags.BoolVarP(flagSet, &fs.Config.StatsOneLineDate, "stats-one-line-date", "", fs.Config.StatsOneLineDate, "Enables --stats-one-line and add current date/time prefix.")
flags.StringVarP(flagSet, &fs.Config.StatsOneLineDateFormat, "stats-one-line-date-format", "", fs.Config.StatsOneLineDateFormat, "Enables --stats-one-line-date and uses custom formatted date. Enclose date string in double quotes (\"). See https://golang.org/pkg/time/#Time.Format") flags.StringVarP(flagSet, &fs.Config.StatsOneLineDateFormat, "stats-one-line-date-format", "", fs.Config.StatsOneLineDateFormat, "Enables --stats-one-line-date and uses custom formatted date. Enclose date string in double quotes (\"). See https://golang.org/pkg/time/#Time.Format")