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
|
package module
import (
"tunnel/pkg/server/queue"
"tunnel/pkg/server/opts"
"tunnel/pkg/server/env"
"fmt"
"log"
)
type moduleInitFunc func (opts.Opts, env.Env) (module, error)
var modules = map[string]moduleInitFunc{}
type module interface {
Open(env env.Env) (Pipe, Pipe)
}
type M interface {
module
String() string
}
type Pipe func (rq, wq queue.Q) error
func (p Pipe) Open(env env.Env) (Pipe, Pipe) {
return p, nil
}
type reverse struct {
M
}
func Reverse(m M) M {
return &reverse{m}
}
func (r *reverse) Open(env env.Env) (Pipe, Pipe) {
p1, p2 := r.M.Open(env)
return p2, p1
}
type named struct {
name string
module
}
func (m *named) String() string {
return fmt.Sprintf("module:%s", m.name)
}
func register(name string, f moduleInitFunc) {
if _, ok := modules[name]; ok {
log.Panicf("duplicate module name '%s'", name)
}
modules[name] = f
}
func registerPipe(name string, p Pipe) {
register(name, func (opts.Opts, env.Env) (module, error) {
return p, nil
})
}
func New(desc string, env env.Env) (M, error) {
name, opts := opts.Parse(desc)
if f, ok := modules[name]; !ok {
return nil, fmt.Errorf("unknown module '%s'", name)
} else if m, err := f(opts, env); err != nil {
return nil, err
} else {
return &named{name: name, module: m}, nil
}
}
|