-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcontroller.go
125 lines (111 loc) · 3.78 KB
/
controller.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package agent
import (
"fmt"
"log"
"reflect"
"strconv"
"strings"
"github.com/ArieShout/websocket-bench/benchmark"
)
// SubjectMap defines the mapping from a string name to the given testing subject implementation.
var SubjectMap = map[string]benchmark.Subject{
// signalr core
"signalr:json:echo": &benchmark.SignalrCoreJsonEcho{},
"signalr:json:broadcast": &benchmark.SignalrCoreJsonBroadcast{},
"signalr:msgpack:echo": &benchmark.SignalrCoreMsgpackEcho{},
"signalr:msgpack:broadcast": &benchmark.SignalrCoreMsgpackBroadcast{},
// signalr service
"signalr:service:json:echo": &benchmark.SignalrServiceJsonEcho{},
"signalr:service:msgpack:echo": &benchmark.SignalrServiceMsgpackEcho{},
"signalr:service:json:broadcast": &benchmark.SignalrServiceJsonBroadcast{},
"signalr:service:msgpack:broadcast": &benchmark.SignalrServiceMsgpackBroadcast{},
}
// Controller stands for a single agent and exposes management interfaces.
type Controller struct {
Subject benchmark.Subject
}
// Invocation represents a command invocation from the master to the agent controller.
type Invocation struct {
Command string
Arguments []string
}
func argError(pos int, command string, expected string, given string) error {
return fmt.Errorf("The %dth argument for command '%s' is %s, but it cannot be parsed from '%s'", pos, command, expected, given)
}
func (c *Controller) Setup(config *benchmark.Config, reply *struct{}) error {
subject, ok := SubjectMap[config.Subject]
if !ok {
return fmt.Errorf("Cannot find subject: " + config.Subject)
}
c.Subject = subject
return c.Subject.Setup(config)
}
func (c *Controller) CollectCounters(args *struct{}, result *map[string]int64) error {
for k, v := range c.Subject.Counters() {
(*result)[k] += v
}
return nil
}
// Invoke calls the method on the agent controller with the name Do<Command>.
func (c *Controller) Invoke(invocation *Invocation, reply *struct{}) error {
if invocation == nil {
return fmt.Errorf("nil Invocation")
}
subject := reflect.ValueOf(c.Subject)
method := subject.MethodByName("Do" + invocation.Command)
if !method.IsValid() {
return fmt.Errorf("Command '%s' was not found", invocation.Command)
}
log.Printf("%s(%s)", invocation.Command, strings.Join(invocation.Arguments, ", "))
argsCount := method.Type().NumIn()
if len(invocation.Arguments) != argsCount {
return fmt.Errorf("Command '%s' needs %d arguments, %d provided", invocation.Command, argsCount, len(invocation.Arguments))
}
in := make([]reflect.Value, argsCount)
for i := 0; i < argsCount; i++ {
stringArg := invocation.Arguments[i]
t := method.Type().In(i)
var arg interface{}
var err error
switch t.Name() {
case "string":
arg = stringArg
case "bool":
arg, err = strconv.ParseBool(stringArg)
if err != nil {
return argError(i, invocation.Command, t.Name(), stringArg)
}
case "int":
arg, err = strconv.Atoi(stringArg)
if err != nil {
return argError(i, invocation.Command, t.Name(), stringArg)
}
case "int32":
tmp, err := strconv.ParseInt(stringArg, 10, 32)
if err != nil {
return argError(i, invocation.Command, t.Name(), stringArg)
}
arg = int32(tmp)
case "float32":
tmp, err := strconv.ParseFloat(stringArg, 32)
if err != nil {
return argError(i, invocation.Command, t.Name(), stringArg)
}
arg = float32(tmp)
case "float64":
arg, err = strconv.ParseFloat(stringArg, 64)
if err != nil {
return argError(i, invocation.Command, t.Name(), stringArg)
}
// TODO: Support more types
default:
return fmt.Errorf("The %dth argument type %s for command '%s' is not supported", i, t.Name(), invocation.Command)
}
in[i] = reflect.ValueOf(arg)
}
response := method.Call(in)
if response[0].Interface() == nil {
return nil
}
return response[0].Interface().(error)
}