-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathtask.go
117 lines (102 loc) · 2.08 KB
/
task.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
package gossamr
import (
"fmt"
"io"
"log"
"reflect"
)
type Phase uint8
const (
MapPhase Phase = iota
CombinePhase
ReducePhase
)
func GetPhase(name string) (Phase, error) {
switch name {
default:
return 0, fmt.Errorf("Unknown phase %s", name)
case "":
return 0, fmt.Errorf("Missing phase")
case "map":
return MapPhase, nil
case "combine":
return CombinePhase, nil
case "reduce":
return ReducePhase, nil
}
}
type Task struct {
instance interface{}
value reflect.Value
}
func NewTask(instance interface{}) *Task {
value := reflect.ValueOf(instance)
return &Task{
instance: instance,
value: value,
}
}
func (t *Task) Run(phase Phase, r io.Reader, w io.WriteCloser) (err error) {
var input Reader
pairs := NewPairReader(r)
output := NewPairWriter(w)
var m reflect.Value
var ok bool
switch phase {
default:
return fmt.Errorf("Invalid phase %d", phase)
case MapPhase:
input = pairs
m, ok = t.mapper()
case CombinePhase:
input = NewGroupedReader(pairs)
m, ok = t.combiner()
case ReducePhase:
input = NewGroupedReader(pairs)
m, ok = t.reducer()
}
if !ok {
return fmt.Errorf("No phase %d for %s", phase, t.instance)
}
err = t.run(m, input, output)
return
}
func (t *Task) run(m reflect.Value, input Reader, output Writer) (err error) {
collector := NewWriterCollector(output)
colValue := reflect.ValueOf(collector)
defer func() {
if e := output.Close(); e != nil && err == nil {
err = e
}
}()
var k, v interface{}
for {
k, v, err = input.Next()
if err != nil {
if err == io.EOF {
return nil
}
log.Printf("Read error: %s", err)
return
}
m.Call([]reflect.Value{
reflect.ValueOf(k),
reflect.ValueOf(v),
colValue,
})
}
}
func (t *Task) mapper() (reflect.Value, bool) {
return t.methodByName("Map")
}
func (t *Task) combiner() (reflect.Value, bool) {
return t.methodByName("Combine")
}
func (t *Task) reducer() (reflect.Value, bool) {
return t.methodByName("Reduce")
}
func (t *Task) methodByName(name string) (v reflect.Value, ok bool) {
v = t.value.MethodByName(name)
ok = v.Kind() == reflect.Func
return
}