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
5fc83902
Commit
5fc83902
authored
Mar 20, 2021
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
8ee4675a
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
67 additions
and
23 deletions
+67
-23
go/zodb/storage/demo/demo.go
go/zodb/storage/demo/demo.go
+66
-22
go/zodb/storage/demo/demo_test.go
go/zodb/storage/demo/demo_test.go
+1
-1
No files found.
go/zodb/storage/demo/demo.go
View file @
5fc83902
...
@@ -30,6 +30,7 @@ import (
...
@@ -30,6 +30,7 @@ import (
"io"
"io"
"net/url"
"net/url"
"regexp"
"regexp"
"sync"
"lab.nexedi.com/kirr/go123/mem"
"lab.nexedi.com/kirr/go123/mem"
"lab.nexedi.com/kirr/go123/xerr"
"lab.nexedi.com/kirr/go123/xerr"
...
@@ -54,6 +55,10 @@ type Storage struct {
...
@@ -54,6 +55,10 @@ type Storage struct {
watchWG
*
xsync
.
WorkGroup
watchWG
*
xsync
.
WorkGroup
watchCancel
func
()
watchCancel
func
()
downOnce
sync
.
Once
down
chan
struct
{}
// ready when storage is down
downErr
error
// reason for shutdown
}
}
// baseMutatedError is reported when Storage.base is detected to change.
// baseMutatedError is reported when Storage.base is detected to change.
...
@@ -78,6 +83,7 @@ func (e *baseError) Error() string {
...
@@ -78,6 +83,7 @@ func (e *baseError) Error() string {
func
(
e
*
baseError
)
Cause
()
error
{
return
e
.
err
}
func
(
e
*
baseError
)
Cause
()
error
{
return
e
.
err
}
func
(
e
*
baseError
)
Unwrap
()
error
{
return
e
.
err
}
func
(
e
*
baseError
)
Unwrap
()
error
{
return
e
.
err
}
// watcher detects base mutation and proxies δ events to user watchq.
// watcher detects base mutation and proxies δ events to user watchq.
// it runs as separate goroutine.
// it runs as separate goroutine.
func
(
d
*
Storage
)
watcher
(
ctx
context
.
Context
)
error
{
func
(
d
*
Storage
)
watcher
(
ctx
context
.
Context
)
error
{
...
@@ -111,11 +117,11 @@ func (d *Storage) watcher(ctx context.Context) error {
...
@@ -111,11 +117,11 @@ func (d *Storage) watcher(ctx context.Context) error {
edown
=
fmt
.
Errorf
(
"base: unexpected event %T"
,
event
)
edown
=
fmt
.
Errorf
(
"base: unexpected event %T"
,
event
)
}
}
d
.
shutdown
(
edown
)
ev
:=
&
zodb
.
EventError
{
edown
}
// XXX + context
ev
:=
&
zodb
.
EventError
{
edown
}
// XXX + context
if
d
.
watchq
!=
nil
{
if
d
.
watchq
!=
nil
{
d
.
watchq
<-
ev
d
.
watchq
<-
ev
}
}
// XXX d.shutdown(edown)
return
edown
return
edown
// event on δ -> proxy to user
// event on δ -> proxy to user
...
@@ -132,6 +138,15 @@ func (d *Storage) watcher(ctx context.Context) error {
...
@@ -132,6 +138,15 @@ func (d *Storage) watcher(ctx context.Context) error {
}
}
}
}
// shutdown marks Storage as no longer being operational due to reason.
func
(
d
*
Storage
)
shutdown
(
reason
error
)
{
d
.
downOnce
.
Do
(
func
()
{
d
.
downErr
=
reason
close
(
d
.
down
)
})
}
var
errClosed
=
errors
.
New
(
"storage is closed"
)
// Close implements zodb.IStorageDriver .
// Close implements zodb.IStorageDriver .
func
(
d
*
Storage
)
Close
()
(
err
error
)
{
func
(
d
*
Storage
)
Close
()
(
err
error
)
{
...
@@ -141,6 +156,7 @@ func (d *Storage) Close() (err error) {
...
@@ -141,6 +156,7 @@ func (d *Storage) Close() (err error) {
}
}
}()
}()
d
.
shutdown
(
errClosed
)
errδ
:=
d
.
δ
.
Close
()
errδ
:=
d
.
δ
.
Close
()
errBase
:=
d
.
base
.
Close
()
errBase
:=
d
.
base
.
Close
()
...
@@ -200,38 +216,51 @@ func (d *Storage) Load(ctx context.Context, xid zodb.Xid) (_ *mem.Buf, _ zodb.Ti
...
@@ -200,38 +216,51 @@ func (d *Storage) Load(ctx context.Context, xid zodb.Xid) (_ *mem.Buf, _ zodb.Ti
}
}
}()
}()
data
,
serial
,
err
:=
d
.
δ
.
Load
(
ctx
,
xid
)
if
ready
(
d
.
down
)
{
if
err
==
nil
{
return
nil
,
zodb
.
InvalidTid
,
d
.
downErr
// object data is present in δ
return
data
,
serial
,
nil
}
}
useBase
:=
false
inδ
:=
false
var
eNoData
*
zodb
.
NoDataError
var
eNoData
*
zodb
.
NoDataError
var
eNoObject
*
zodb
.
NoObjectError
var
eNoObject
*
zodb
.
NoObjectError
switch
{
case
errors
.
As
(
err
,
&
eNoData
)
:
inδ
:=
false
if
eNoData
.
DeletedAt
!=
0
{
if
xid
.
At
>
d
.
baseAt0
{
// object deleted in δ -> whiteout
data
,
serial
,
err
:=
d
.
δ
.
Load
(
ctx
,
xid
)
return
data
,
serial
,
eNoData
if
err
==
nil
{
}
else
{
// object data is present in δ
// object present in δ but not yet created as of xid.at
return
data
,
serial
,
nil
}
useBase
:=
false
switch
{
case
errors
.
As
(
err
,
&
eNoData
)
:
if
eNoData
.
DeletedAt
!=
0
{
// object deleted in δ -> whiteout
return
data
,
serial
,
eNoData
}
else
{
// object present in δ but not yet created as of xid.at
useBase
=
true
inδ
=
true
}
case
errors
.
As
(
err
,
&
eNoObject
)
:
// object not created in δ
useBase
=
true
useBase
=
true
inδ
=
true
}
}
case
errors
.
As
(
err
,
&
eNoObject
)
:
if
!
useBase
{
// object not created in δ
return
data
,
serial
,
err
useBase
=
true
}
}
}
if
!
useBase
{
// cap .at in xid to .baseAt0 (we convert it back on error return, and
return
data
,
serial
,
err
// it makes more robust wrt simultaneous base mutation).
xidBase
:=
xid
if
xid
.
At
>
d
.
baseAt0
{
xidBase
.
At
=
d
.
baseAt0
}
}
// XXX cap .at in xid to .baseAt0 ? (and convert back on error return)
data
,
serial
,
err
:=
d
.
base
.
Load
(
ctx
,
xidBase
)
data
,
serial
,
err
=
d
.
base
.
Load
(
ctx
,
xid
)
if
err
==
nil
{
if
err
==
nil
{
return
data
,
serial
,
nil
return
data
,
serial
,
nil
}
}
...
@@ -350,6 +379,8 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.DriverOptions) (_ zodb
...
@@ -350,6 +379,8 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.DriverOptions) (_ zodb
baseWatchq
:
baseWatchq
,
baseWatchq
:
baseWatchq
,
δWatchq
:
δWatchq
,
δWatchq
:
δWatchq
,
watchq
:
opt
.
Watchq
,
watchq
:
opt
.
Watchq
,
down
:
make
(
chan
struct
{}),
}
}
// spawn watcher to listen on baseWatchq and shutdown storage if base changes.
// spawn watcher to listen on baseWatchq and shutdown storage if base changes.
...
@@ -364,3 +395,16 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.DriverOptions) (_ zodb
...
@@ -364,3 +395,16 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.DriverOptions) (_ zodb
func
init
()
{
func
init
()
{
zodb
.
RegisterDriver
(
"demo"
,
openByURL
)
zodb
.
RegisterDriver
(
"demo"
,
openByURL
)
}
}
// misc
// ready returns whether c is ready.
func
ready
(
c
<-
chan
struct
{})
bool
{
select
{
case
<-
c
:
return
true
default
:
return
false
}
}
go/zodb/storage/demo/demo_test.go
View file @
5fc83902
...
@@ -240,7 +240,7 @@ func TestWatchLoad_vs_BaseMutate(t *testing.T) {
...
@@ -240,7 +240,7 @@ func TestWatchLoad_vs_BaseMutate(t *testing.T) {
t
.
Fatalf
(
"after base mutate: load: unexpected error:
\n
have: %s
\n
want: %s"
,
t
.
Fatalf
(
"after base mutate: load: unexpected error:
\n
have: %s
\n
want: %s"
,
err
,
errOk
)
err
,
errOk
)
}
}
if
!
(
data
==
nil
&&
serial
==
0
)
{
if
!
(
data
==
nil
&&
serial
==
zodb
.
InvalidTid
)
{
t
.
Fatalf
(
"after base mutate: load: unexpected data=%v serial=%v"
,
data
,
serial
)
t
.
Fatalf
(
"after base mutate: load: unexpected data=%v serial=%v"
,
data
,
serial
)
}
}
})
})
...
...
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