-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfilelock.go
156 lines (145 loc) · 3.56 KB
/
filelock.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
// Package filelock provides locked File struct similar with os.File.
package filelock
import (
"context"
"errors"
"os"
"sync"
"syscall"
"time"
)
// File represents an open file descriptor such as os.File. But File always has Posix write lock.
type File struct {
internalFile
name string
closeOnce sync.Once
}
type internalFile = *os.File
// Open opens the named file with OpenFile for reading.
// When an error occurs, Open returns the error like OpenFile.
// If successful, methods on the returned file can be used for reading; the associated file descriptor has mode os.O_RDONLY.
func Open(name string) (*File, error) {
return OpenFile(name, os.O_RDONLY, 0)
}
// Create creates or opens the named file with OpenFile.
// When an error occurs, Create returns OpenFile error.
// If the file already exists, it is not truncated as with os.Create.
// If the file does not exist, it is created with mode perm (before umask).
// If successful, methods on the returned File can be used for I/O; the associated file descriptor has mode os.O_RDWR.
func Create(name string, perm os.FileMode) (*File, error) {
f, err := OpenFile(name, os.O_RDWR, 0)
if err != nil {
if !errors.Is(err, os.ErrNotExist) {
return nil, err
}
f, err = OpenFile(name, os.O_RDWR|os.O_CREATE|os.O_EXCL, perm)
if err != nil {
if _, ok := err.(*LockError); ok {
_ = os.Remove(name)
}
return nil, err
}
}
return f, nil
}
// OpenFile opens the named file with os.OpenFile, after locks with Posix lock.
// When an error occurs, OpenFile returns os.OpenFile error or LockError.
// If the file created with os.O_CREATE and LockError occurs, OpenFile will not delete created file.
func OpenFile(name string, flag int, perm os.FileMode) (f *File, err error) {
filesMu.Lock()
if _, ok := files[name]; ok {
filesMu.Unlock()
return nil, ErrLocked
}
files[name] = nil
filesMu.Unlock()
defer func() {
if err != nil {
filesMu.Lock()
delete(files, name)
filesMu.Unlock()
}
}()
f2, err := os.OpenFile(name, flag, perm)
if err != nil {
return nil, err
}
defer func() {
if err != nil {
_ = f2.Close()
}
}()
ok, err := posixLock(f2.Fd())
if err != nil {
return nil, err
}
if !ok {
return nil, ErrLocked
}
f = &File{
internalFile: f2,
name: name,
}
filesMu.Lock()
files[name] = f
filesMu.Unlock()
return f, nil
}
// Acquire tries to obtain file lock given period.
func Acquire(ctx context.Context, name string, perm os.FileMode, period time.Duration) (*File, error) {
f, err := Create(name, perm)
if err != ErrLocked {
return f, err
}
tkr := time.NewTicker(period)
defer tkr.Stop()
for {
select {
case <-ctx.Done():
return nil, ctx.Err()
case <-tkr.C:
f, err = Create(name, perm)
if err != ErrLocked {
return f, err
}
}
}
}
// Close closes and unlocks the File.
func (f *File) Close() (err error) {
err = f.internalFile.Close()
f.closeOnce.Do(func() {
filesMu.Lock()
delete(files, f.name)
filesMu.Unlock()
})
return
}
// Release deletes, closes and unlocks the File.
func (f *File) Release() (err error) {
f.closeOnce.Do(func() {
_ = os.Remove(f.name)
err = f.internalFile.Close()
filesMu.Lock()
delete(files, f.name)
filesMu.Unlock()
})
return
}
var files = make(map[string]*File)
var filesMu sync.Mutex
func posixLock(fd uintptr) (ok bool, err error) {
err = syscall.FcntlFlock(fd, syscall.F_SETLK, &syscall.Flock_t{
Type: syscall.F_WRLCK,
Whence: 0,
Start: 0,
Len: 0,
})
if err != nil {
if err != syscall.EWOULDBLOCK {
return false, &LockError{err}
}
return false, nil
}
return true, nil
}