-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathenum.go
199 lines (176 loc) · 5.13 KB
/
enum.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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
package enum
import (
"fmt"
"math/rand"
"strings"
"time"
)
// Member is an enum member, a specific value bound to a variable.
type Member[T comparable] struct {
Value T
}
// Equaler check if the two values of the same type are equal.
type Equaler[V comparable] interface {
Equal(other V) bool
comparable
}
// iMember is the type constraint for Member used by Enum.
//
// We can't use Member directly in type constraints
// because the users create a new subtype from Member
// instead of using it directly.
//
// We also can't use a normal interface because new types
// don't inherit methods of their base type.
type iMember[T comparable] interface {
~struct{ Value T }
}
// Enum is a collection of enum members.
//
// Use [New] to construct a new Enum from a list of members.
type Enum[M iMember[V], V comparable] struct {
members []M
v2m map[V]*M
}
// New constructs a new [Enum] wrapping the given enum members.
func New[V comparable, M iMember[V]](members ...M) Enum[M, V] {
e := Enum[M, V]{members, nil}
e.v2m = make(map[V]*M)
for i, m := range e.members {
v := e.Value(m)
e.v2m[v] = &e.members[i]
}
return e
}
// TypeName is a string representation of the wrapped type.
func (Enum[M, V]) TypeName() string {
return fmt.Sprintf("%T", *new(V))
}
// Empty returns true if the enum doesn't have any members.
func (e Enum[M, V]) Empty() bool {
return len(e.members) == 0
}
// Len returns how many members the enum has.
func (e Enum[M, V]) Len() int {
return len(e.members)
}
// Contains returns true if the enum has the given member.
func (e Enum[M, V]) Contains(member M) bool {
for _, m := range e.members {
if m == member {
return true
}
}
return false
}
// Parse converts a raw value into a member of the enum.
//
// If none of the enum members has the given value, nil is returned.
func (e Enum[M, V]) Parse(value V) *M {
return e.v2m[value]
}
// Value returns the wrapped value of the given enum member.
func (e Enum[M, V]) Value(member M) V {
return Member[V](member).Value
}
// Index returns the index of the given member in the enum.
//
// If the given member is not in the enum, it panics.
// Use [Enum.Contains] first if you don't know for sure
// if the member belongs to the enum.
func (e Enum[M, V]) Index(member M) int {
for i, m := range e.members {
if e.Value(m) == e.Value(member) {
return i
}
}
panic("the given Member does not belong to this Enum")
}
// Members returns a slice of the members in the enum.
func (e Enum[M, V]) Members() []M {
return e.members
}
// Choice returns a randomly selected member of the enum.
//
// A random seed can be given (or be 0 to use time.Now().UnixNano() as the seed).
// nil is returned only if the Enum contains no members.
func (e Enum[M, V]) Choice(seed int64) *M {
lenMembers := len(e.members)
// Enum is empty
if lenMembers == 0 {
return nil
}
if seed == 0 {
seed = time.Now().UnixNano()
}
// nolint: gosec
r := rand.New(rand.NewSource(seed))
return &(e.members[r.Intn(lenMembers)])
}
// Values returns a slice of values of all members of the enum.
func (e Enum[M, V]) Values() []V {
res := make([]V, 0, len(e.members))
for _, m := range e.members {
res = append(res, e.Value(m))
}
return res
}
// String implements [fmt.Stringer] interface.
//
// It returns a comma-separated list of values of the enum members.
func (e Enum[M, V]) String() string {
values := make([]string, 0, len(e.members))
for _, m := range e.members {
values = append(values, fmt.Sprintf("%v", e.Value(m)))
}
return strings.Join(values, ", ")
}
// GoString implements [fmt.GoStringer] interface.
//
// When you print a member using "%#v" format,
// it will show the enum representation as a valid Go syntax.
func (e Enum[M, V]) GoString() string {
values := make([]string, 0, len(e.members))
for _, m := range e.members {
values = append(values, fmt.Sprintf("%T{%#v}", m, e.Value(m)))
}
joined := strings.Join(values, ", ")
return fmt.Sprintf("enum.New(%s)", joined)
}
// Parse is like [Enum.Parse] but finds the member for the value using [Equaler] comparator.
func Parse[M iMember[V], V Equaler[V]](e Enum[M, V], value V) *M {
for v, m := range e.v2m {
if v.Equal(value) {
return m
}
}
return nil
}
// Builder is a constructor for an [Enum].
//
// Use [Builder.Add] to add new members to the future enum
// and then call [Builder.Enum] to create a new [Enum] with all added members.
//
// Builder is useful for when you have lots of enum members, and new ones
// are added over time, as the project grows. In such scenario, it's easy to forget
// to add in the [Enum] a newly created [Member].
// The builder is designed to prevent that.
type Builder[M iMember[V], V comparable] struct {
members []M
finished bool
}
// NewBuilder creates a new [Builder], a constructor for an [Enum].
func NewBuilder[V comparable, M iMember[V]]() Builder[M, V] {
return Builder[M, V]{make([]M, 0), false}
}
// Add registers a new [Member] in the builder.
func (b *Builder[M, V]) Add(m M) M {
b.members = append(b.members, m)
return m
}
// Enum creates a new [Enum] with all members registered using [Builder.Add].
func (b *Builder[M, V]) Enum() Enum[M, V] {
b.finished = true
e := New(b.members...)
return e
}