Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
projectdiscovery
GitHub Repository: projectdiscovery/nuclei
Path: blob/dev/pkg/protocols/http/race/syncedreadcloser.go
2073 views
1
package race
2
3
import (
4
"fmt"
5
"io"
6
"time"
7
)
8
9
// SyncedReadCloser is compatible with io.ReadSeeker and performs
10
// gate-based synced writes to enable race condition testing.
11
type SyncedReadCloser struct {
12
data []byte
13
p int64
14
length int64
15
openGate chan struct{}
16
enableBlocking bool
17
}
18
19
// NewSyncedReadCloser creates a new SyncedReadCloser instance.
20
func NewSyncedReadCloser(r io.ReadCloser) *SyncedReadCloser {
21
var (
22
s SyncedReadCloser
23
err error
24
)
25
s.data, err = io.ReadAll(r)
26
if err != nil {
27
return nil
28
}
29
defer func() {
30
_ = r.Close()
31
}()
32
s.length = int64(len(s.data))
33
s.openGate = make(chan struct{})
34
s.enableBlocking = true
35
return &s
36
}
37
38
// NewOpenGateWithTimeout creates a new open gate with a timeout
39
func NewOpenGateWithTimeout(r io.ReadCloser, d time.Duration) *SyncedReadCloser {
40
s := NewSyncedReadCloser(r)
41
s.OpenGateAfter(d)
42
return s
43
}
44
45
// SetOpenGate sets the status of the blocking gate
46
func (s *SyncedReadCloser) SetOpenGate(status bool) {
47
s.enableBlocking = status
48
}
49
50
// OpenGate opens the gate allowing all requests to be completed
51
func (s *SyncedReadCloser) OpenGate() {
52
s.openGate <- struct{}{}
53
}
54
55
// OpenGateAfter schedules gate to be opened after a duration
56
func (s *SyncedReadCloser) OpenGateAfter(d time.Duration) {
57
time.AfterFunc(d, func() {
58
s.openGate <- struct{}{}
59
})
60
}
61
62
// Seek implements seek method for io.ReadSeeker
63
func (s *SyncedReadCloser) Seek(offset int64, whence int) (int64, error) {
64
var err error
65
switch whence {
66
case io.SeekStart:
67
s.p = 0
68
case io.SeekCurrent:
69
if s.p+offset < s.length {
70
s.p += offset
71
break
72
}
73
err = fmt.Errorf("offset is too big")
74
case io.SeekEnd:
75
if s.length-offset >= 0 {
76
s.p = s.length - offset
77
break
78
}
79
err = fmt.Errorf("offset is too big")
80
}
81
return s.p, err
82
}
83
84
// Read implements read method for io.ReadSeeker
85
func (s *SyncedReadCloser) Read(p []byte) (n int, err error) {
86
// If the data fits in the buffer blocks awaiting the sync instruction
87
if s.p+int64(len(p)) >= s.length && s.enableBlocking {
88
<-s.openGate
89
}
90
n = copy(p, s.data[s.p:])
91
s.p += int64(n)
92
if s.p == s.length {
93
err = io.EOF
94
}
95
return n, err
96
}
97
98
// Close closes an io.ReadSeeker
99
func (s *SyncedReadCloser) Close() error {
100
return nil
101
}
102
103
// Len returns the length of data in reader
104
func (s *SyncedReadCloser) Len() int {
105
return int(s.length)
106
}
107
108