package code import ( ice "github.com/shylinux/icebergs" "github.com/shylinux/icebergs/base/mdb" "github.com/shylinux/icebergs/base/web" kit "github.com/shylinux/toolkits" "github.com/shylinux/toolkits/util/bench" "io" "io/ioutil" "net/http" "os" "strings" "sync/atomic" ) func _bench_list(m *ice.Message, zone string, id string, field ...interface{}) { m.Richs(BENCH, nil, kit.Select(kit.MDB_FOREACH, zone), func(key string, val map[string]interface{}) { val = val[kit.MDB_META].(map[string]interface{}) if zone = kit.Format(kit.Value(val, kit.MDB_ZONE)); id == "" { m.Grows(BENCH, kit.Keys(kit.MDB_HASH, key), "", "", func(index int, value map[string]interface{}) { // 查看信息 m.Push("操作", m.Cmdx(mdb.RENDER, web.RENDER.Button, "运行")) m.Push(zone, value, []string{ kit.MDB_ZONE, kit.MDB_ID, kit.MDB_TYPE, kit.MDB_NAME, NCONN, NREQS, kit.MDB_TEXT, }, val) }) return } m.Grows(BENCH, kit.Keys(kit.MDB_HASH, key), kit.MDB_ID, id, func(index int, value map[string]interface{}) { // 查看信息 m.Push("detail", value) m.Push(kit.MDB_KEY, "操作") m.Push(kit.MDB_VALUE, m.Cmdx(mdb.RENDER, web.RENDER.Button, "运行")) }) }) } func _bench_show(m *ice.Message, nconn, nreq int64, list []*http.Request) { m.Log_CONF(NCONN, nconn, NREQS, nreq) var body int64 s, e := bench.HTTP(nconn, nreq, list, func(req *http.Request, res *http.Response) { n, _ := io.Copy(ioutil.Discard, res.Body) atomic.AddInt64(&body, n) }) m.Assert(e) m.Echo(s.Show()) m.Echo("body: %d\n", body) } func _bench_engine(m *ice.Message, kind, name, target string, arg ...string) { for i := 0; i < len(arg); i += 2 { m.Option(arg[i], arg[i+1]) } nconn := kit.Int64(kit.Select("10", m.Option(NCONN))) nreqs := kit.Int64(kit.Select("1000", m.Option(NREQS))) m.Echo("nconn: %d nreqs: %d\n", nconn, nreqs*nconn) list := []*http.Request{} for _, v := range strings.Split(target, ",") { switch ls := kit.Split(v); ls[0] { case http.MethodPost: if f, e := os.Open(ls[2]); m.Assert(e) { defer f.Close() req, err := http.NewRequest(http.MethodPost, ls[1], f) m.Assert(err) list = append(list, req) } default: req, err := http.NewRequest(http.MethodGet, v, nil) m.Assert(err) list = append(list, req) } } _bench_show(m, nconn, nreqs, list) } func _bench_modify(m *ice.Message, zone, id, k, v, old string) { m.Richs(BENCH, nil, zone, func(key string, val map[string]interface{}) { switch k { case kit.MDB_ZONE, kit.MDB_ID, kit.MDB_TIME: m.Warn(true, mdb.ErrDenyModify, k) return } m.Grows(BENCH, kit.Keys(kit.MDB_HASH, key), kit.MDB_ID, id, func(index int, value map[string]interface{}) { // 修改信息 m.Log_MODIFY(kit.MDB_ZONE, zone, kit.MDB_ID, id, kit.MDB_KEY, k, kit.MDB_VALUE, v, "old", old) kit.Value(value, k, v) }) }) } func _bench_insert(m *ice.Message, zone, kind, name, text string, nconn, nreqs string, arg ...string) { m.Richs(BENCH, nil, zone, func(key string, value map[string]interface{}) { id := m.Grow(BENCH, kit.Keys(kit.MDB_HASH, key), kit.Dict( kit.MDB_TYPE, kind, kit.MDB_NAME, name, kit.MDB_TEXT, text, NCONN, nconn, NREQS, nreqs, kit.MDB_EXTRA, kit.Dict(arg), )) m.Log_INSERT(kit.MDB_ZONE, zone, kit.MDB_ID, id, kit.MDB_TYPE, kind, kit.MDB_NAME, name) m.Echo("%d", id) }) } func _bench_create(m *ice.Message, zone string, arg ...string) { m.Rich(BENCH, nil, kit.Data(kit.MDB_ZONE, zone, arg)) m.Log_CREATE(kit.MDB_ZONE, zone) } const BENCH = "bench" const ( NCONN = "nconn" NREQS = "nreqs" ) func init() { Index.Merge(&ice.Context{ Configs: map[string]*ice.Config{ BENCH: {Name: "bench", Help: "性能压测", Value: kit.Data(kit.MDB_SHORT, kit.MDB_ZONE)}, }, Commands: map[string]*ice.Command{ BENCH: {Name: "bench zone=auto id=auto auto", Help: "性能压测", Action: map[string]*ice.Action{ mdb.CREATE: {Name: "create zone", Help: "创建", Hand: func(m *ice.Message, arg ...string) { _bench_create(m, arg[0]) }}, mdb.INSERT: {Name: "insert zone type name text nconn nreqs", Help: "插入", Hand: func(m *ice.Message, arg ...string) { _bench_insert(m, arg[0], arg[1], arg[2], kit.Select("http://localhost:9020/code/bench?cmd="+arg[2], arg, 3), kit.Select("3", arg, 4), kit.Select("10", arg, 5)) }}, mdb.MODIFY: {Name: "modify key value old", Help: "编辑", Hand: func(m *ice.Message, arg ...string) { _bench_modify(m, m.Option(kit.MDB_ZONE), m.Option(kit.MDB_ID), arg[0], arg[1], kit.Select("", arg, 2)) }}, mdb.ENGINE: {Name: "engine type name text arg...", Help: "引擎", Hand: func(m *ice.Message, arg ...string) { _bench_engine(m, arg[0], arg[1], arg[2], arg[3:]...) }}, kit.MDB_SHOW: {Name: "show type name text arg...", Help: "运行", Hand: func(m *ice.Message, arg ...string) { if len(arg) < 4 { m.Richs(BENCH, nil, m.Option(kit.MDB_ZONE), func(key string, val map[string]interface{}) { m.Grows(BENCH, kit.Keys(kit.MDB_HASH, key), kit.MDB_ID, m.Option(kit.MDB_ID), func(index int, value map[string]interface{}) { arg = kit.Simple(value[kit.MDB_TYPE], value[kit.MDB_NAME], value[kit.MDB_TEXT], value[kit.MDB_EXTRA]) }) }) } if len(arg) > 2 { m.Option(kit.MDB_TYPE, arg[0]) m.Option(kit.MDB_NAME, arg[1]) m.Option(kit.MDB_TEXT, arg[2]) for i := 3; i < len(arg)-1; i++ { m.Option(arg[i], arg[i+1]) } } m.Cmdy(mdb.ENGINE, m.Option(kit.MDB_TYPE), m.Option(kit.MDB_NAME), m.Option(kit.MDB_TEXT)) }}, }, Hand: func(m *ice.Message, c *ice.Context, cmd string, arg ...string) { _bench_list(m, kit.Select(kit.MDB_FOREACH, arg, 0), kit.Select("", arg, 1)) }}, "/" + BENCH: {Name: "/bench cmd...", Help: "性能压测", Hand: func(m *ice.Message, c *ice.Context, cmd string, arg ...string) { m.Cmdy(m.Optionv("cmd")) }}, }, }, nil) }