188 lines
4.5 KiB
Go
188 lines
4.5 KiB
Go
|
|
/*
|
||
|
|
* Copyright (c) 2019 Huawei Technologies Co., Ltd.
|
||
|
|
* A-Tune is licensed under the Mulan PSL v1.
|
||
|
|
* You can use this software according to the terms and conditions of the Mulan PSL v1.
|
||
|
|
* You may obtain a copy of Mulan PSL v1 at:
|
||
|
|
* http://license.coscl.org.cn/MulanPSL
|
||
|
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
||
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
||
|
|
* PURPOSE.
|
||
|
|
* See the Mulan PSL v1 for more details.
|
||
|
|
* Create: 2019-10-29
|
||
|
|
*/
|
||
|
|
|
||
|
|
package profile
|
||
|
|
|
||
|
|
import (
|
||
|
|
PB "atune/api/profile"
|
||
|
|
"atune/common/client"
|
||
|
|
SVC "atune/common/service"
|
||
|
|
"atune/common/utils"
|
||
|
|
"fmt"
|
||
|
|
"io"
|
||
|
|
"path/filepath"
|
||
|
|
|
||
|
|
"github.com/urfave/cli"
|
||
|
|
CTX "golang.org/x/net/context"
|
||
|
|
)
|
||
|
|
|
||
|
|
var collectionCommand = cli.Command{
|
||
|
|
Name: "collection",
|
||
|
|
Usage: "collect the system data for training",
|
||
|
|
UsageText: "atune-adm collection OPTIONS",
|
||
|
|
Flags: []cli.Flag{
|
||
|
|
cli.StringFlag{
|
||
|
|
Name: "filename,f",
|
||
|
|
Usage: "filename of the generated data file",
|
||
|
|
Value: "",
|
||
|
|
},
|
||
|
|
cli.IntFlag{
|
||
|
|
Name: "interval,i",
|
||
|
|
Usage: "time interval for collecting data",
|
||
|
|
Value: 5,
|
||
|
|
},
|
||
|
|
cli.IntFlag{
|
||
|
|
Name: "duration,d",
|
||
|
|
Usage: "the duration for collecting data",
|
||
|
|
Value: 1200,
|
||
|
|
},
|
||
|
|
cli.StringFlag{
|
||
|
|
Name: "output_path,o",
|
||
|
|
Usage: "the output path of the collecting data",
|
||
|
|
Value: "",
|
||
|
|
},
|
||
|
|
cli.StringFlag{
|
||
|
|
Name: "disk,b",
|
||
|
|
Usage: "the disk to be collected",
|
||
|
|
Value: "",
|
||
|
|
},
|
||
|
|
cli.StringFlag{
|
||
|
|
Name: "network,n",
|
||
|
|
Usage: "the network to be collected",
|
||
|
|
Value: "",
|
||
|
|
},
|
||
|
|
cli.StringFlag{
|
||
|
|
Name: "workload_type,t",
|
||
|
|
Usage: "the workload type of the collected data",
|
||
|
|
Value: "",
|
||
|
|
},
|
||
|
|
},
|
||
|
|
Description: func() string {
|
||
|
|
desc := `
|
||
|
|
collect data for train machine learning model, you must set the command options
|
||
|
|
which has no default value, the output_path must be a absolute path.
|
||
|
|
example: atune-adm collection -f mysql -i 5 -d 1200 -o /home -b sda -n eth0 -t idle`
|
||
|
|
return desc
|
||
|
|
}(),
|
||
|
|
Action: collection,
|
||
|
|
}
|
||
|
|
|
||
|
|
func init() {
|
||
|
|
svc := SVC.ProfileService{
|
||
|
|
Name: "opt.profile.collection",
|
||
|
|
Desc: "opt profile system",
|
||
|
|
NewInst: newCollectionCmd,
|
||
|
|
}
|
||
|
|
if err := SVC.AddService(&svc); err != nil {
|
||
|
|
fmt.Printf("Failed to load collection service : %s\n", err)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
func newCollectionCmd(ctx *cli.Context, opts ...interface{}) (interface{}, error) {
|
||
|
|
return collectionCommand, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func checkCollectionCtx(ctx *cli.Context) error {
|
||
|
|
if ctx.String("filename") == "" {
|
||
|
|
cli.ShowCommandHelp(ctx, "collection")
|
||
|
|
return fmt.Errorf("error: filename must be specified")
|
||
|
|
}
|
||
|
|
|
||
|
|
if len(ctx.String("filename")) > 255 {
|
||
|
|
return fmt.Errorf("error: filename length is longer than 255 charaters")
|
||
|
|
}
|
||
|
|
|
||
|
|
if ctx.String("disk") == "" {
|
||
|
|
cli.ShowCommandHelp(ctx, "collection")
|
||
|
|
return fmt.Errorf("error: disk block must be specified")
|
||
|
|
}
|
||
|
|
if ctx.String("network") == "" {
|
||
|
|
cli.ShowCommandHelp(ctx, "collection")
|
||
|
|
return fmt.Errorf("error: network must be specified")
|
||
|
|
}
|
||
|
|
|
||
|
|
if ctx.String("workload_type") == "" {
|
||
|
|
cli.ShowCommandHelp(ctx, "collection")
|
||
|
|
return fmt.Errorf("error: workload type must be specified")
|
||
|
|
}
|
||
|
|
|
||
|
|
if ctx.String("output_path") == "" {
|
||
|
|
cli.ShowCommandHelp(ctx, "collection")
|
||
|
|
return fmt.Errorf("error: output_path must be specified")
|
||
|
|
}
|
||
|
|
|
||
|
|
if ctx.Int64("interval") < 1 || ctx.Int64("interval") > 60 {
|
||
|
|
return fmt.Errorf("error: collection interval value must be between 1 and 60 seconds")
|
||
|
|
}
|
||
|
|
|
||
|
|
if ctx.Int64("duration") < ctx.Int64("interval")*10 {
|
||
|
|
return fmt.Errorf("error: collection duration value must be bigger than interval*10")
|
||
|
|
}
|
||
|
|
|
||
|
|
if !filepath.IsAbs(ctx.String("output_path")) {
|
||
|
|
return fmt.Errorf("error: output path must be absolute path")
|
||
|
|
}
|
||
|
|
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func collection(ctx *cli.Context) error {
|
||
|
|
if err := checkCollectionCtx(ctx); err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
|
||
|
|
c, err := client.NewClientFromContext(ctx)
|
||
|
|
if err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
defer c.Close()
|
||
|
|
|
||
|
|
outputPath := ctx.String("output_path")
|
||
|
|
outputPath, err = filepath.Abs(outputPath)
|
||
|
|
if err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
message := PB.CollectFlag{
|
||
|
|
Interval: ctx.Int64("interval"),
|
||
|
|
Duration: ctx.Int64("duration"),
|
||
|
|
Workload: ctx.String("filename"),
|
||
|
|
OutputPath: outputPath,
|
||
|
|
Block: ctx.String("disk"),
|
||
|
|
Network: ctx.String("network"),
|
||
|
|
Type: ctx.String("workload_type"),
|
||
|
|
}
|
||
|
|
|
||
|
|
svc := PB.NewProfileMgrClient(c.Connection())
|
||
|
|
stream, err := svc.Collection(CTX.Background(), &message)
|
||
|
|
if err != nil {
|
||
|
|
fmt.Println(err)
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
|
||
|
|
for {
|
||
|
|
reply, err := stream.Recv()
|
||
|
|
|
||
|
|
if err == io.EOF {
|
||
|
|
break
|
||
|
|
}
|
||
|
|
|
||
|
|
if err != nil {
|
||
|
|
return err
|
||
|
|
}
|
||
|
|
utils.Print(reply)
|
||
|
|
}
|
||
|
|
|
||
|
|
return nil
|
||
|
|
}
|