Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neoppod
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Levin Zimmermann
neoppod
Commits
9d8e264f
Commit
9d8e264f
authored
Dec 19, 2018
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
5aae345c
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
48 additions
and
16 deletions
+48
-16
go/zodb/storage/fs1/filestorage.go
go/zodb/storage/fs1/filestorage.go
+48
-16
No files found.
go/zodb/storage/fs1/filestorage.go
View file @
9d8e264f
...
@@ -468,9 +468,11 @@ func (fs *FileStorage) Iterate(_ context.Context, tidMin, tidMax zodb.Tid) zodb.
...
@@ -468,9 +468,11 @@ func (fs *FileStorage) Iterate(_ context.Context, tidMin, tidMax zodb.Tid) zodb.
//
//
// watcher is the only place that mutates index and txnh{Min,Max}.
// watcher is the only place that mutates index and txnh{Min,Max}.
// XXX ^^^ will change after commit is implemented.
// XXX ^^^ will change after commit is implemented.
func
(
fs
*
FileStorage
)
watcher
(
w
*
fsnotify
.
Watcher
)
{
//
// if errFirstRead is !nil, the error of reading first transaction header is sent to it.
func
(
fs
*
FileStorage
)
watcher
(
w
*
fsnotify
.
Watcher
,
errFirstRead
chan
<-
error
)
{
defer
w
.
Close
()
// XXX lclose
defer
w
.
Close
()
// XXX lclose
err
:=
fs
.
_watcher
(
w
)
err
:=
fs
.
_watcher
(
w
,
errFirstRead
)
// it is ok if we got read error due to file being closed
// it is ok if we got read error due to file being closed
if
e
,
_
:=
errors
.
Cause
(
err
)
.
(
*
os
.
PathError
);
e
!=
nil
&&
(
e
.
Err
==
os
.
ErrClosed
||
if
e
,
_
:=
errors
.
Cause
(
err
)
.
(
*
os
.
PathError
);
e
!=
nil
&&
(
e
.
Err
==
os
.
ErrClosed
||
// XXX it can also be internal.poll.ErrFileClosing
// XXX it can also be internal.poll.ErrFileClosing
...
@@ -486,8 +488,8 @@ func (fs *FileStorage) watcher(w *fsnotify.Watcher) {
...
@@ -486,8 +488,8 @@ func (fs *FileStorage) watcher(w *fsnotify.Watcher) {
log
.
Print
(
err
)
log
.
Print
(
err
)
}
}
// if watcher failed with e.g. IO error we no longer know what is real
// if watcher failed with e.g. IO error
,
we no longer know what is real
// last_tid and which objects were modified
in the IO error region
.
// last_tid and which objects were modified
after it
.
// -> storage operations have to fail from now on.
// -> storage operations have to fail from now on.
fs
.
shutdown
(
err
)
fs
.
shutdown
(
err
)
...
@@ -498,10 +500,16 @@ func (fs *FileStorage) watcher(w *fsnotify.Watcher) {
...
@@ -498,10 +500,16 @@ func (fs *FileStorage) watcher(w *fsnotify.Watcher) {
// _watcher serves watcher and returns either when fs is closed (ok), or when
// _watcher serves watcher and returns either when fs is closed (ok), or when
// it hits any kind of non-recoverable error.
// it hits any kind of non-recoverable error.
func
(
fs
*
FileStorage
)
_watcher
(
w
*
fsnotify
.
Watcher
)
(
err
error
)
{
func
(
fs
*
FileStorage
)
_watcher
(
w
*
fsnotify
.
Watcher
,
errFirstRead
chan
<-
error
)
(
err
error
)
{
f
:=
fs
.
file
f
:=
fs
.
file
idx
:=
fs
.
index
idx
:=
fs
.
index
defer
xerr
.
Contextf
(
&
err
,
"%s: watcher"
,
f
.
Name
())
defer
xerr
.
Contextf
(
&
err
,
"%s: watcher"
,
f
.
Name
())
defer
func
()
{
if
errFirstRead
!=
nil
{
errFirstRead
<-
err
errFirstRead
=
nil
}
}()
// loop checking f.size vs topPos
// loop checking f.size vs topPos
//
//
...
@@ -606,6 +614,11 @@ mainloop:
...
@@ -606,6 +614,11 @@ mainloop:
// read ok - reset t₀(partial)
// read ok - reset t₀(partial)
t0partial
=
time
.
Time
{}
t0partial
=
time
.
Time
{}
if
errFirstRead
!=
nil
{
errFirstRead
<-
nil
// ok
errFirstRead
=
nil
}
// XXX dup wrt Index.Update
// XXX dup wrt Index.Update
// we could successfully read the transaction header. Try to see now,
// we could successfully read the transaction header. Try to see now,
...
@@ -634,7 +647,7 @@ mainloop:
...
@@ -634,7 +647,7 @@ mainloop:
oidv
=
append
(
oidv
,
it
.
Datah
.
Oid
)
oidv
=
append
(
oidv
,
it
.
Datah
.
Oid
)
}
}
// update index & txnh{MinMax}
// update index & txnh{Min
,
Max}
fs
.
mu
.
Lock
()
fs
.
mu
.
Lock
()
idx
.
TopPos
=
it
.
Txnh
.
Pos
+
it
.
Txnh
.
Len
idx
.
TopPos
=
it
.
Txnh
.
Pos
+
it
.
Txnh
.
Len
for
oid
,
pos
:=
range
update
{
for
oid
,
pos
:=
range
update
{
...
@@ -708,7 +721,7 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
...
@@ -708,7 +721,7 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
fs
.
file
=
f
fs
.
file
=
f
defer
func
()
{
defer
func
()
{
if
err
!=
nil
{
if
err
!=
nil
{
f
.
Close
()
// XXX -> lclose
f
s
.
shutdown
(
err
)
}
}
}()
}()
...
@@ -743,8 +756,13 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
...
@@ -743,8 +756,13 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
err
=
index
.
Update
(
ctx
,
fseq
,
-
1
,
nil
/*no progress; XXX log it? */
)
err
=
index
.
Update
(
ctx
,
fseq
,
-
1
,
nil
/*no progress; XXX log it? */
)
}
}
// XXX cause=ErrUnexpectedEOF -> let watcher decide what was
// it can be either garbage or in-progress transaction.
// it: garbage or in-progress transaction
// defer to watcher to clarify this for us.
checkTailGarbage
:=
false
if
errors
.
Cause
(
err
)
==
io
.
ErrUnexpectedEOF
{
err
=
nil
checkTailGarbage
=
true
}
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
...
@@ -788,17 +806,31 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
...
@@ -788,17 +806,31 @@ func Open(ctx context.Context, path string, opt *zodb.DriverOptions) (_ *FileSto
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
defer
func
()
{
err
=
w
.
Add
(
f
.
Name
())
if
err
!=
nil
{
if
err
!=
nil
{
w
.
Close
()
// XXX lclose
w
.
Close
()
// XXX lclose
return
nil
,
err
}
}
}()
err
=
w
.
Add
(
f
.
Name
())
var
errFirstRead
chan
error
if
checkTailGarbage
{
defer
xerr
.
Contextf
(
&
err
,
"open %s: checking whether it is garbage at @%d"
,
path
,
index
.
TopPos
)
errFirstRead
=
make
(
chan
error
,
1
)
}
go
fs
.
watcher
(
w
,
errFirstRead
)
if
checkTailGarbage
{
select
{
case
<-
ctx
.
Done
()
:
return
nil
,
ctx
.
Err
()
case
err
=
<-
errFirstRead
:
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
// it was garbage
}
}
}
}
go
fs
.
watcher
(
w
)
return
fs
,
nil
return
fs
,
nil
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment