-
-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathsemaphore.go
91 lines (75 loc) · 2.07 KB
/
semaphore.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
// Package semaphore provides an implementation of Semaphore pattern
// with timeout of lock/unlock operations based on channels.
package semaphore
import "errors"
// ReleaseFunc tells a semaphore to release the previously occupied slot
// and ignore an error if it occurs.
type ReleaseFunc func()
// Release calls f().
func (f ReleaseFunc) Release() error {
f()
return nil
}
// New constructs a new thread-safe Semaphore with the given capacity.
func New(capacity int) Semaphore {
return make(semaphore, capacity)
}
// IsEmpty checks if passed error is related to call Release on empty semaphore.
func IsEmpty(err error) bool {
return err == errEmpty
}
// IsNoPlace checks if passed error is related to call Catch on full semaphore.
func IsNoPlace(err error) bool {
return err == errNoPlace
}
// IsTimeout checks if passed error is related to call Acquire on full semaphore.
func IsTimeout(err error) bool {
return err == errTimeout
}
var (
nothing ReleaseFunc = func() {}
errEmpty = errors.New("semaphore is empty")
errNoPlace = errors.New("semaphore has no place")
errTimeout = errors.New("operation timeout")
)
type semaphore chan struct{}
func (semaphore semaphore) Acquire(deadline <-chan struct{}) (ReleaseFunc, error) {
select {
case semaphore <- struct{}{}:
return func() { _ = semaphore.Release() }, nil //nolint: gas
case <-deadline:
return nothing, errTimeout
}
}
func (semaphore semaphore) Catch() (ReleaseFunc, error) {
select {
case semaphore <- struct{}{}:
return func() { _ = semaphore.Release() }, nil //nolint: gas
default:
return nothing, errNoPlace
}
}
func (semaphore semaphore) Capacity() int {
return cap(semaphore)
}
func (semaphore semaphore) Occupied() int {
return len(semaphore)
}
func (semaphore semaphore) Release() error {
select {
case <-semaphore:
return nil
default:
return errEmpty
}
}
func (semaphore semaphore) Signal(deadline <-chan struct{}) <-chan ReleaseFunc {
ch := make(chan ReleaseFunc, 1)
go func() {
if release, err := semaphore.Acquire(deadline); err == nil {
ch <- release
}
close(ch)
}()
return ch
}