forked from TrueCloudLab/rclone
Introduce AtExit to fix --cpuprofile and --memprofile to write profiles at end of run
This commit is contained in:
parent
6e0e1ad9cb
commit
2da6cd7f84
2 changed files with 53 additions and 3 deletions
45
cmd/atexit.go
Normal file
45
cmd/atexit.go
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
package cmd
|
||||||
|
|
||||||
|
// Atexit handling
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
atExitFns []func()
|
||||||
|
atExitOnce sync.Once
|
||||||
|
atExitRegisterOnce sync.Once
|
||||||
|
)
|
||||||
|
|
||||||
|
// AtExit registers a function to be added on exit
|
||||||
|
func AtExit(fn func()) {
|
||||||
|
atExitFns = append(atExitFns, fn)
|
||||||
|
// Run AtExit handlers on SIGINT or SIGTERM so everything gets
|
||||||
|
// tidied up properly
|
||||||
|
atExitRegisterOnce.Do(func() {
|
||||||
|
go func() {
|
||||||
|
ch := make(chan os.Signal, 1)
|
||||||
|
signal.Notify(ch, os.Interrupt) // syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT
|
||||||
|
sig := <-ch
|
||||||
|
fs.Infof(nil, "Signal received: %s", sig)
|
||||||
|
runAtExitFunctions()
|
||||||
|
fs.Infof(nil, "Exiting...")
|
||||||
|
os.Exit(0)
|
||||||
|
}()
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Runs all the AtExit functions if they haven't been run already
|
||||||
|
func runAtExitFunctions() {
|
||||||
|
atExitOnce.Do(func() {
|
||||||
|
for _, fn := range atExitFns {
|
||||||
|
fn()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
11
cmd/cmd.go
11
cmd/cmd.go
|
@ -69,6 +69,9 @@ and configuration walkthroughs.
|
||||||
|
|
||||||
* http://rclone.org/
|
* http://rclone.org/
|
||||||
`,
|
`,
|
||||||
|
PersistentPostRun: func(cmd *cobra.Command, args []string) {
|
||||||
|
runAtExitFunctions()
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// runRoot implements the main rclone command with no subcommands
|
// runRoot implements the main rclone command with no subcommands
|
||||||
|
@ -341,12 +344,14 @@ func initConfig() {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
defer pprof.StopCPUProfile()
|
AtExit(func() {
|
||||||
|
pprof.StopCPUProfile()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Setup memory profiling if desired
|
// Setup memory profiling if desired
|
||||||
if *memProfile != "" {
|
if *memProfile != "" {
|
||||||
defer func() {
|
AtExit(func() {
|
||||||
fs.Infof(nil, "Saving Memory profile %q\n", *memProfile)
|
fs.Infof(nil, "Saving Memory profile %q\n", *memProfile)
|
||||||
f, err := os.Create(*memProfile)
|
f, err := os.Create(*memProfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -363,7 +368,7 @@ func initConfig() {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
}()
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if m, _ := regexp.MatchString("^(bits|bytes)$", *dataRateUnit); m == false {
|
if m, _ := regexp.MatchString("^(bits|bytes)$", *dataRateUnit); m == false {
|
||||||
|
|
Loading…
Reference in a new issue