Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
Kirill Smelkov
neo
Commits
23870baf
Commit
23870baf
authored
Nov 09, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
fe8d33cd
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
65 additions
and
55 deletions
+65
-55
go/neo/client/client.go
go/neo/client/client.go
+9
-1
go/zodb/storage/fs1/filestorage.go
go/zodb/storage/fs1/filestorage.go
+44
-49
go/zodb/storage/fs1/url.go
go/zodb/storage/fs1/url.go
+5
-1
go/zodb/zodb.go
go/zodb/zodb.go
+5
-2
go/zodb/zodbtools/dump.go
go/zodb/zodbtools/dump.go
+1
-1
go/zodb/zodbtools/info.go
go/zodb/zodbtools/info.go
+1
-1
No files found.
go/neo/client/client.go
View file @
23870baf
...
@@ -67,6 +67,8 @@ type Client struct {
...
@@ -67,6 +67,8 @@ type Client struct {
// protected by .node.StateMu
// protected by .node.StateMu
operational
bool
// XXX <- somehow move to NodeApp?
operational
bool
// XXX <- somehow move to NodeApp?
opReady
chan
struct
{}
// reinitialized each time state becomes non-operational
opReady
chan
struct
{}
// reinitialized each time state becomes non-operational
url
*
url
.
URL
// original URL we were opened with
}
}
var
_
zodb
.
IStorage
=
(
*
Client
)(
nil
)
var
_
zodb
.
IStorage
=
(
*
Client
)(
nil
)
...
@@ -75,6 +77,10 @@ func (c *Client) StorageName() string {
...
@@ -75,6 +77,10 @@ func (c *Client) StorageName() string {
return
fmt
.
Sprintf
(
"neo(%s)"
,
c
.
node
.
ClusterName
)
return
fmt
.
Sprintf
(
"neo(%s)"
,
c
.
node
.
ClusterName
)
}
}
func
(
c
*
Client
)
URL
()
string
{
return
c
.
url
.
String
()
}
// NewClient creates new client node.
// NewClient creates new client node.
//
//
// It will connect to master @masterAddr and identify with sepcified cluster name.
// It will connect to master @masterAddr and identify with sepcified cluster name.
...
@@ -521,7 +527,9 @@ func openClientByURL(ctx context.Context, u *url.URL, opt *zodb.OpenOptions) (zo
...
@@ -521,7 +527,9 @@ func openClientByURL(ctx context.Context, u *url.URL, opt *zodb.OpenOptions) (zo
// XXX we are not passing ctx to NewClient - right?
// XXX we are not passing ctx to NewClient - right?
// as ctx for open can be done after open finishes - not covering
// as ctx for open can be done after open finishes - not covering
// whole storage working lifetime.
// whole storage working lifetime.
return
NewClient
(
u
.
User
.
Username
(),
u
.
Host
,
net
),
nil
c
:=
NewClient
(
u
.
User
.
Username
(),
u
.
Host
,
net
)
c
.
url
=
u
// FIXME move this inside NewClient
return
c
,
nil
}
}
func
init
()
{
func
init
()
{
...
...
go/zodb/storage/fs1/filestorage.go
View file @
23870baf
...
@@ -68,6 +68,7 @@ import (
...
@@ -68,6 +68,7 @@ import (
"fmt"
"fmt"
"io"
"io"
"log"
"log"
"net/url"
"os"
"os"
"sync"
"sync"
...
@@ -89,6 +90,8 @@ type FileStorage struct {
...
@@ -89,6 +90,8 @@ type FileStorage struct {
// XXX keep loaded with LoadNoStrings ?
// XXX keep loaded with LoadNoStrings ?
txnhMin
TxnHeader
txnhMin
TxnHeader
txnhMax
TxnHeader
txnhMax
TxnHeader
url
*
url
.
URL
// original URL we were opened with
}
}
// IStorage
// IStorage
...
@@ -98,6 +101,10 @@ func (fs *FileStorage) StorageName() string {
...
@@ -98,6 +101,10 @@ func (fs *FileStorage) StorageName() string {
return
"FileStorage v1"
return
"FileStorage v1"
}
}
func
(
fs
*
FileStorage
)
URL
()
string
{
return
fs
.
url
.
String
()
}
func
(
fs
*
FileStorage
)
LastTid
(
_
context
.
Context
)
(
zodb
.
Tid
,
error
)
{
func
(
fs
*
FileStorage
)
LastTid
(
_
context
.
Context
)
(
zodb
.
Tid
,
error
)
{
// XXX check we have transactions at all - what to return if not?
// XXX check we have transactions at all - what to return if not?
...
@@ -418,7 +425,7 @@ func (fs *FileStorage) Iterate(tidMin, tidMax zodb.Tid) zodb.ITxnIterator {
...
@@ -418,7 +425,7 @@ func (fs *FileStorage) Iterate(tidMin, tidMax zodb.Tid) zodb.ITxnIterator {
return
ziter
return
ziter
}
}
// --- open + rebuild index ---
TODO review completely
// --- open + rebuild index ---
// open opens FileStorage without loading index
// open opens FileStorage without loading index
func
open
(
path
string
)
(
_
*
FileStorage
,
err
error
)
{
func
open
(
path
string
)
(
_
*
FileStorage
,
err
error
)
{
...
@@ -445,8 +452,6 @@ func open(path string) (_ *FileStorage, err error) {
...
@@ -445,8 +452,6 @@ func open(path string) (_ *FileStorage, err error) {
// FIXME rework opening logic to support case when last txn was committed only partially
// FIXME rework opening logic to support case when last txn was committed only partially
// determine topPos from file size
// determine topPos from file size
// if it is invalid (e.g. a transaction committed only half-way) we'll catch it
// while loading/recreating index XXX recheck this logic
fi
,
err
:=
f
.
Stat
()
fi
,
err
:=
f
.
Stat
()
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
...
@@ -454,13 +459,13 @@ func open(path string) (_ *FileStorage, err error) {
...
@@ -454,13 +459,13 @@ func open(path string) (_ *FileStorage, err error) {
topPos
:=
fi
.
Size
()
topPos
:=
fi
.
Size
()
// read tidMin/tidMax
// read tidMin/tidMax
// FIXME support empty file case -> then both txnhMin and txnhMax stays invalid
err
=
fs
.
txnhMin
.
Load
(
f
,
txnValidFrom
,
LoadAll
)
err
=
fs
.
txnhMin
.
Load
(
f
,
txnValidFrom
,
LoadAll
)
// XXX txnValidFrom here -> ?
err
=
okEOF
(
err
)
// e.g. it is EOF when file is empty
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
err
=
fs
.
txnhMax
.
Load
(
f
,
topPos
,
LoadAll
)
err
=
fs
.
txnhMax
.
Load
(
f
,
topPos
,
LoadAll
)
// expect EOF
but .LenPrev must be goo
d
// expect EOF
forwar
d
// FIXME ^^^ it will be no EOF if a txn was committed only partially
// FIXME ^^^ it will be no EOF if a txn was committed only partially
if
err
!=
io
.
EOF
{
if
err
!=
io
.
EOF
{
if
err
==
nil
{
if
err
==
nil
{
...
@@ -468,15 +473,21 @@ func open(path string) (_ *FileStorage, err error) {
...
@@ -468,15 +473,21 @@ func open(path string) (_ *FileStorage, err error) {
}
}
return
nil
,
fmt
.
Errorf
(
"%s: %s"
,
f
.
Name
(),
err
)
return
nil
,
fmt
.
Errorf
(
"%s: %s"
,
f
.
Name
(),
err
)
}
}
if
fs
.
txnhMax
.
LenPrev
<=
0
{
// .LenPrev must be good or EOF backward
switch
fs
.
txnhMax
.
LenPrev
{
case
0
:
return
nil
,
fmt
.
Errorf
(
"%s: could not read LenPrev @%d (last transaction)"
,
f
.
Name
(),
fs
.
txnhMax
.
Pos
)
return
nil
,
fmt
.
Errorf
(
"%s: could not read LenPrev @%d (last transaction)"
,
f
.
Name
(),
fs
.
txnhMax
.
Pos
)
}
case
-
1
:
// ok - EOF backward
default
:
// .LenPrev is ok - read last previous record
err
=
fs
.
txnhMax
.
LoadPrev
(
f
,
LoadAll
)
err
=
fs
.
txnhMax
.
LoadPrev
(
f
,
LoadAll
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
}
return
fs
,
nil
return
fs
,
nil
}
}
...
@@ -496,59 +507,60 @@ func Open(ctx context.Context, path string) (_ *FileStorage, err error) {
...
@@ -496,59 +507,60 @@ func Open(ctx context.Context, path string) (_ *FileStorage, err error) {
}
}
}()
}()
// load
/
rebuild index
// load
-verify /
rebuild index
err
=
fs
.
loadIndex
()
err
=
fs
.
loadIndex
(
ctx
)
if
err
!=
nil
{
if
err
!=
nil
{
log
.
Print
(
err
)
log
.
Print
(
err
)
log
.
Printf
(
"%s: index recompute..."
,
path
)
log
.
Printf
(
"%s: index recompute..."
,
path
)
// XXX if !ro -> .reindex() which saves it
fs
.
index
,
err
=
fs
.
computeIndex
(
ctx
)
fs
.
index
,
err
=
fs
.
computeIndex
(
ctx
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
}
// TODO verify index is sane / topPos matches
// TODO if opened !ro -> .saveIndex()
// XXX zodb/py iirc scans 10 transactions back and verifies index against it
// XXX also if index is not good - it has to be just rebuild without open error
if
fs
.
index
.
TopPos
!=
fs
.
txnhMax
.
Pos
+
fs
.
txnhMax
.
Len
{
return
nil
,
fmt
.
Errorf
(
"%s: inconsistent index topPos (TODO rebuild index)"
,
path
)
}
}
return
fs
,
nil
return
fs
,
nil
}
}
func
(
fs
*
FileStorage
)
Close
()
error
{
func
(
fs
*
FileStorage
)
Close
()
error
{
// TODO dump index if !ro
err
:=
fs
.
file
.
Close
()
err
:=
fs
.
file
.
Close
()
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
err
}
}
fs
.
file
=
nil
fs
.
file
=
nil
// TODO if opened !ro -> .saveIndex()
return
nil
return
nil
}
}
func
(
fs
*
FileStorage
)
computeIndex
(
ctx
context
.
Context
)
(
index
*
Index
,
err
error
)
{
func
(
fs
*
FileStorage
)
computeIndex
(
ctx
context
.
Context
)
(
index
*
Index
,
err
error
)
{
// XXX lock?
// XXX lock?
fsSeq
:=
xbufio
.
NewSeqReaderAt
(
fs
.
file
)
fsSeq
:=
xbufio
.
NewSeqReaderAt
(
fs
.
file
)
return
BuildIndex
(
ctx
,
fsSeq
,
nil
/*
XXX no progress
*/
)
return
BuildIndex
(
ctx
,
fsSeq
,
nil
/*
no progress; XXX somehow log it?
*/
)
}
}
// loadIndex loads on-disk index to RAM
// loadIndex loads on-disk index to RAM
and verifies it against data lightly
func
(
fs
*
FileStorage
)
loadIndex
()
(
err
error
)
{
func
(
fs
*
FileStorage
)
loadIndex
(
ctx
context
.
Context
)
(
err
error
)
{
// XXX lock?
// XXX lock?
// XXX LoadIndexFile already contains "%s: index load"
defer
xerr
.
Contextf
(
&
err
,
"%s"
,
fs
.
file
.
Name
())
defer
xerr
.
Contextf
(
&
err
,
"%s"
,
fs
.
file
.
Name
())
index
,
err
:=
LoadIndexFile
(
fs
.
file
.
Name
()
+
".index"
)
index
,
err
:=
LoadIndexFile
(
fs
.
file
.
Name
()
+
".index"
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
// XXX err ctx
return
err
}
topPos
:=
fs
.
txnhMax
.
Pos
+
fs
.
txnhMax
.
Len
if
index
.
TopPos
!=
topPos
{
return
fmt
.
Errorf
(
"inconsistent index topPos: data=%d index=%d"
,
topPos
,
index
.
TopPos
)
}
}
// XXX here?
// quickly verify index sanity for last 100 transactions
// TODO verify index sane / topPos matches
fsSeq
:=
xbufio
.
NewSeqReaderAt
(
fs
.
file
)
if
index
.
TopPos
!=
fs
.
txnhMax
.
Pos
+
fs
.
txnhMax
.
Len
{
_
,
err
=
index
.
Verify
(
ctx
,
fsSeq
,
100
,
nil
/*no progress*/
)
panic
(
"inconsistent index topPos"
)
// XXX
if
err
!=
nil
{
return
err
}
}
fs
.
index
=
index
fs
.
index
=
index
...
@@ -558,7 +570,7 @@ func (fs *FileStorage) loadIndex() (err error) {
...
@@ -558,7 +570,7 @@ func (fs *FileStorage) loadIndex() (err error) {
// saveIndex flushes in-RAM index to disk
// saveIndex flushes in-RAM index to disk
func
(
fs
*
FileStorage
)
saveIndex
()
(
err
error
)
{
func
(
fs
*
FileStorage
)
saveIndex
()
(
err
error
)
{
// XXX lock?
// XXX lock?
defer
xerr
.
Contextf
(
&
err
,
"%s
: index save
"
,
fs
.
file
.
Name
())
defer
xerr
.
Contextf
(
&
err
,
"%s"
,
fs
.
file
.
Name
())
err
=
fs
.
index
.
SaveFile
(
fs
.
file
.
Name
()
+
".index"
)
err
=
fs
.
index
.
SaveFile
(
fs
.
file
.
Name
()
+
".index"
)
if
err
!=
nil
{
if
err
!=
nil
{
...
@@ -568,20 +580,3 @@ func (fs *FileStorage) saveIndex() (err error) {
...
@@ -568,20 +580,3 @@ func (fs *FileStorage) saveIndex() (err error) {
// XXX fsync here?
// XXX fsync here?
return
nil
return
nil
}
}
// Reindex rebuilds the index
//
// XXX -> not exported @ fs1
func
(
fs
*
FileStorage
)
reindex
(
ctx
context
.
Context
)
error
{
// XXX lock appends?
index
,
err
:=
fs
.
computeIndex
(
ctx
)
if
err
!=
nil
{
return
err
}
fs
.
index
=
index
err
=
fs
.
saveIndex
()
return
err
// XXX ok?
}
go/zodb/storage/fs1/url.go
View file @
23870baf
...
@@ -39,7 +39,11 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.OpenOptions) (zodb.ISt
...
@@ -39,7 +39,11 @@ func openByURL(ctx context.Context, u *url.URL, opt *zodb.OpenOptions) (zodb.ISt
return
nil
,
fmt
.
Errorf
(
"fs1: %s: TODO write mode not implemented"
,
path
)
return
nil
,
fmt
.
Errorf
(
"fs1: %s: TODO write mode not implemented"
,
path
)
}
}
return
Open
(
ctx
,
path
)
fs
,
err
:=
Open
(
ctx
,
path
)
if
fs
!=
nil
{
fs
.
url
=
u
// FIXME move this inside Open
}
return
fs
,
err
}
}
func
init
()
{
func
init
()
{
...
...
go/zodb/zodb.go
View file @
23870baf
...
@@ -134,8 +134,11 @@ func (e *ErrXidMissing) Error() string {
...
@@ -134,8 +134,11 @@ func (e *ErrXidMissing) Error() string {
// IStorage is the interface provided by ZODB storages
// IStorage is the interface provided by ZODB storages
type
IStorage
interface
{
type
IStorage
interface
{
// StorageName returns storage name
// URL returns URL of this storage
StorageName
()
string
URL
()
string
// XXX also +StorageName() with storage driver name?
// Close closes storage
// Close closes storage
Close
()
error
Close
()
error
...
...
go/zodb/zodbtools/dump.go
View file @
23870baf
...
@@ -214,7 +214,7 @@ func (d *dumper) Dump(ctx context.Context, stor zodb.IStorage, tidMin, tidMax zo
...
@@ -214,7 +214,7 @@ func (d *dumper) Dump(ctx context.Context, stor zodb.IStorage, tidMin, tidMax zo
}
}
if
err
!=
nil
{
if
err
!=
nil
{
return
fmt
.
Errorf
(
"%s: dump
ing %v..%v: %v"
,
stor
,
tidMin
,
tidMax
,
err
)
return
fmt
.
Errorf
(
"%s: dump
%v..%v: %v"
,
stor
.
URL
()
,
tidMin
,
tidMax
,
err
)
}
}
return
nil
return
nil
...
...
go/zodb/zodbtools/info.go
View file @
23870baf
...
@@ -38,7 +38,7 @@ type paramFunc func(ctx context.Context, stor zodb.IStorage) (string, error)
...
@@ -38,7 +38,7 @@ type paramFunc func(ctx context.Context, stor zodb.IStorage) (string, error)
var
infov
=
[]
struct
{
name
string
;
getParam
paramFunc
}
{
var
infov
=
[]
struct
{
name
string
;
getParam
paramFunc
}
{
// XXX e.g. stor.LastTid() should return err itself
// XXX e.g. stor.LastTid() should return err itself
{
"name"
,
func
(
ctx
context
.
Context
,
stor
zodb
.
IStorage
)
(
string
,
error
)
{
{
"name"
,
func
(
ctx
context
.
Context
,
stor
zodb
.
IStorage
)
(
string
,
error
)
{
return
stor
.
StorageName
(),
nil
return
stor
.
URL
(),
nil
}},
}},
// TODO reenable size
// TODO reenable size
// {"size", func(stor zodb.IStorage) (string, error) { return stor.StorageSize(), nil }},
// {"size", func(stor zodb.IStorage) (string, error) { return stor.StorageSize(), 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