-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathbind.go
More file actions
executable file
·87 lines (74 loc) · 1.5 KB
/
bind.go
File metadata and controls
executable file
·87 lines (74 loc) · 1.5 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
package di
import (
"errors"
"fmt"
"reflect"
)
// ResolveT devuelve un valor del inyector usando genéricos.
//
// cfg, err := di.ResolveT[*Config](inj)
func ResolveT[T any](inj Injector) (T, error) {
var zero T
var dst T // se pasa la dirección
// Resolve requiere un "pointer al destino".
if err := inj.Resolve(&dst); err != nil {
return zero, err
}
return dst, nil
}
func (inj *injector) Resolve(v ...Pointer) error {
for _, p := range v {
r := reflect.ValueOf(p)
if r.Kind() != reflect.Ptr || r.IsNil() {
return errors.New("invalid pointer")
}
elem := r.Elem()
t := elem.Type()
val, ok := inj.Get(t)
if !ok {
return errors.New("can't find injector for " + t.String())
}
if !elem.CanSet() {
return errors.New("cannot set value")
}
elem.Set(val)
}
return nil
}
func (inj *injector) Apply(p Pointer) error {
v := reflect.ValueOf(p)
for v.Kind() == reflect.Ptr {
v = v.Elem()
}
if !v.CanSet() {
return errors.New("cannot set value")
}
typ := v.Type()
valInject, ok := inj.Get(typ)
if ok {
v.Set(valInject)
return nil
}
if v.Kind() != reflect.Struct {
valInject, ok := inj.Get(typ)
if !ok {
return nil
}
v.Set(valInject)
return nil
}
for i := 0; i < v.NumField(); i++ {
f := v.Field(i)
structField := typ.Field(i)
_, ok := structField.Tag.Lookup("di")
if f.CanSet() && ok {
ft := f.Type()
valInject, ok := inj.Get(ft)
if !ok {
return fmt.Errorf("value not found for type %v", ft)
}
f.Set(valInject)
}
}
return nil
}