-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathonce.go
108 lines (88 loc) · 2.13 KB
/
once.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
/*
* Drop-in replacement for Go's sync featuring generics and channels.
* Copyright (C) 2025 Dviih
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*
*/
package sync
import (
"reflect"
"sync/atomic"
)
type Once[T interface{}] struct {
done atomic.Bool
m Mutex
result []interface{}
r interface{}
}
func (once *Once[T]) Do(fn T) interface{} {
if once.r != nil {
panic(once.r)
}
if once.done.Load() {
switch len(once.result) {
case 0:
return nil
case 1:
return once.result[0]
default:
return once.result[:]
}
}
defer once.m.Unlock()
once.m.Lock()
v := reflect.ValueOf(fn)
if v.Type().NumIn() > 0 {
panic("once: Do func must not have input arguments")
}
defer func() {
once.r = recover()
if !once.done.Load() {
panic(once.r)
}
}()
out := v.Call(nil)
for i := 0; i < len(out); i++ {
once.result = append(once.result, out[i].Interface())
}
once.done.Store(true)
switch len(once.result) {
case 0:
return nil
case 1:
return once.result[0]
default:
return once.result[:]
}
}
func OnceFunc(fn func()) func() {
once := &Once[func()]{}
return func() {
once.Do(fn)
}
}
func OnceValue[T interface{}](fn func() T) func() T {
once := &Once[func() T]{}
return func() T {
return once.Do(fn).(T)
}
}
func OnceValues[T1, T2 interface{}](fn func() (T1, T2)) func() (T1, T2) {
once := &Once[func() (T1, T2)]{}
return func() (T1, T2) {
i := once.Do(fn)
return i.([]interface{})[0].(T1), i.([]interface{})[1].(T2)
}
}