-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathserver.go
More file actions
92 lines (84 loc) · 1.87 KB
/
server.go
File metadata and controls
92 lines (84 loc) · 1.87 KB
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
package tinyRPC
import (
"fmt"
"io"
"log"
"net"
"reflect"
)
type Server struct {
addr string
fns map[string]reflect.Value
}
func NewServer(addr string) *Server {
return &Server{
addr: addr,
fns: make(map[string]reflect.Value),
}
}
func (s *Server) Run() {
l, err := net.Listen("tcp", s.addr)
if err != nil {
log.Printf("listening on %s, error: %v\n", s.addr, err)
return
}
for {
conn, err := l.Accept()
if err != nil {
log.Printf("accept error: %v\n", err)
continue
}
go func() {
transport := NewTransport(conn)
for {
req, err := transport.Receive()
if err != nil {
if err != io.EOF {
log.Printf("receive read err: %v\n", err)
}
return
}
fn, ok := s.fns[req.Name]
if !ok {
// fn doesn'e exist
e := fmt.Sprintf("func %s not exist", req.Name)
log.Printf(e)
if err = transport.Send(Data{Name: req.Name, Err: e}); err != nil {
log.Printf("transport send error: %v\n", err)
}
continue
}
log.Printf("func %s is called\n", req.Name)
args := make([]reflect.Value, len(req.Args))
for i := range req.Args {
args[i] = reflect.ValueOf(req.Args[i])
}
output := fn.Call(args)
// values will ignore error
values := make([]interface{}, len(output)-1)
var e string
for i := 0; i < len(output); i++ {
if i != len(output)-1 {
values[i] = output[i].Interface()
} else {
if _, ok := output[len(output)-1].Interface().(error); !ok {
e = ""
} else {
e = output[len(output)-1].Interface().(error).Error()
}
}
}
err = transport.Send(Data{Name: req.Name, Args: values, Err: e})
if err != nil {
log.Printf("transport send error: %v\n", err)
}
}
}()
}
}
func (s *Server) Register(name string, fn interface{}) {
if _, ok := s.fns[name]; ok {
return
}
s.fns[name] = reflect.ValueOf(fn)
}