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
3be18ddb
Commit
3be18ddb
authored
Jul 25, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
558c79cf
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
118 additions
and
95 deletions
+118
-95
go/zodb/storage/fs1/filestorage.go
go/zodb/storage/fs1/filestorage.go
+118
-95
No files found.
go/zodb/storage/fs1/filestorage.go
View file @
3be18ddb
...
...
@@ -140,13 +140,19 @@ const (
Magic
=
"FS21"
// every FileStorage file starts with this
// on-disk sizes
FileHeaderSize
=
4
TxnHeaderFixSize
=
8
+
8
+
1
+
2
+
2
+
2
// without user/desc/ext strings
txnXHeaderFixSize
=
8
+
TxnHeaderFixSize
// ^^^ with trail LenPrev from previous record
DataHeaderSize
=
8
+
8
+
8
+
8
+
2
+
8
// txn/data pos that are < vvv are for sure invalid
txnValidFrom
=
int64
(
len
(
Magic
))
txnValidFrom
=
int64
(
len
(
Magic
))
// XXX = FileHeaderSize
dataValidFrom
=
txnValidFrom
+
TxnHeaderFixSize
// invalid length that indicates start of iteration for TxnHeader LoadNext/LoadPrev
// NOTE load routines check loaded fields to be valid and .Len in particular,
// so it can never come to us from outside to be this.
lenIterStart
int64
=
-
0x1111111111111112
// = 0xeeeeeeeeeeeeeeee if unsigned
)
// ErrTxnRecord is returned on transaction record read / decode errors
...
...
@@ -435,6 +441,19 @@ func (txnh *TxnHeader) LoadPrev(r io.ReaderAt, flags TxnLoadFlags) error {
bug
(
txnh
,
"LoadPrev() when .LenPrev == error"
)
case
-
1
:
return
io
.
EOF
case
lenIterStart
:
// start of iteration backward:
// read LenPrev @pos, then tail to LoadPrev
pos
:=
txnh
.
Pos
err
:=
txnh
.
Load
(
r
,
pos
,
flags
)
if
txnh
.
LenPrev
==
0
{
if
err
==
nil
{
panic
(
"nil err with txnh.LenPrev = error"
)
}
return
err
}
return
txnh
.
LoadPrev
(
r
,
flags
)
}
lenCur
:=
txnh
.
Len
...
...
@@ -471,6 +490,10 @@ func (txnh *TxnHeader) LoadNext(r io.ReaderAt, flags TxnLoadFlags) error {
bug
(
txnh
,
"LoadNext() when .Len == error"
)
case
-
1
:
return
io
.
EOF
case
lenIterStart
:
// start of iteration forward
return
txnh
.
Load
(
r
,
posCur
,
flags
)
}
// valid .Len means txnh was read ok
...
...
@@ -729,6 +752,63 @@ func (dh *DataHeader) LoadData(r io.ReaderAt, buf *[]byte) error {
return
nil
}
// --- raw iteration ---
// Iter is combined 2-level iterator over transaction and data records
type
Iter
struct
{
R
io
.
ReaderAt
Dir
IterDir
Txnh
TxnHeader
// current transaction record information
Datah
DataHeader
// current data record information
}
type
IterDir
int
const
(
IterForward
IterDir
=
iota
IterBackward
)
// NextTxn iterates to next/previous transaction record according to iteration direction.
// The data header is reset to iterate inside transaction record that becomes current.
func
(
it
*
Iter
)
NextTxn
(
flags
TxnLoadFlags
)
error
{
var
err
error
switch
it
.
Dir
{
case
IterForward
:
err
=
it
.
Txnh
.
LoadNext
(
it
.
R
,
flags
)
case
IterBackward
:
err
=
it
.
Txnh
.
LoadPrev
(
it
.
R
,
flags
)
default
:
panic
(
"Iter.Dir invalid"
)
}
//fmt.Println("loaded:", it.Txnh.Tid)
if
err
!=
nil
{
// reset .Datah to be invalid (just in case)
it
.
Datah
.
Pos
=
0
it
.
Datah
.
DataLen
=
0
}
else
{
// set .Datah to iterate over .Txnh
it
.
Datah
.
Pos
=
it
.
Txnh
.
DataPos
()
it
.
Datah
.
DataLen
=
-
DataHeaderSize
// first iteration will go to first data record
}
return
err
}
// NextData iterates to next data record header inside current transaction
func
(
it
*
Iter
)
NextData
()
error
{
return
it
.
Datah
.
LoadNext
(
it
.
R
,
&
it
.
Txnh
)
}
// Iterate creates Iter to iterate over r starting from posStart in direction dir
func
Iterate
(
r
io
.
ReaderAt
,
posStart
int64
,
dir
IterDir
)
*
Iter
{
it
:=
&
Iter
{
R
:
r
,
Dir
:
dir
,
Txnh
:
TxnHeader
{
Pos
:
posStart
,
Len
:
lenIterStart
}}
return
it
}
// --- FileStorage ---
func
(
fs
*
FileStorage
)
StorageName
()
string
{
...
...
@@ -746,13 +826,10 @@ func open(path string) (*FileStorage, error) {
fs
.
file
=
f
// check file magic
var
xxx
[
len
(
Magic
)]
byte
_
,
err
=
f
.
ReadAt
(
xxx
[
:
],
0
)
fh
:=
FileHeader
{}
err
=
fh
.
Load
(
f
)
if
err
!=
nil
{
return
nil
,
err
// XXX err more context
}
if
string
(
xxx
[
:
])
!=
Magic
{
return
nil
,
fmt
.
Errorf
(
"%s: invalid magic %q"
,
path
,
xxx
)
// XXX err?
return
nil
,
err
}
/*
...
...
@@ -914,75 +991,15 @@ func (fs *FileStorage) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error)
return
data
,
tid
,
nil
}
// --- raw iteration ---
// Iter is combined 2-level iterator over transaction and data records
type
Iter
struct
{
fsSeq
*
xbufio
.
SeqReaderAt
Flags
iterFlags
// XXX iterate forward (> 0) / backward (< 0) / EOF reached (== 0)
Txnh
TxnHeader
// current transaction record information
Datah
DataHeader
// current data record information
}
// NextTxn iterates to next/previous transaction record according to iteration direction.
// The data header is reset to iterate inside transaction record that became current.
func
(
it
*
Iter
)
NextTxn
(
flags
TxnLoadFlags
)
error
{
var
err
error
if
it
.
Flags
&
iterDir
!=
0
{
err
=
it
.
Txnh
.
LoadNext
(
it
.
fsSeq
,
flags
)
}
else
{
err
=
it
.
Txnh
.
LoadPrev
(
it
.
fsSeq
,
flags
)
}
//fmt.Println("loaded:", it.Txnh.Tid)
if
err
!=
nil
{
// reset .Datah to be invalid (just in case)
it
.
Datah
.
Pos
=
0
it
.
Datah
.
DataLen
=
0
}
else
{
// set .Datah to iterate over .Txnh
it
.
Datah
.
Pos
=
it
.
Txnh
.
DataPos
()
it
.
Datah
.
DataLen
=
-
DataHeaderSize
// first iteration will go to first data record
}
return
err
}
// NextData iterates to next data record header inside current transaction
func
(
it
*
Iter
)
NextData
()
error
{
return
it
.
Datah
.
LoadNext
(
it
.
fsSeq
,
&
it
.
Txnh
)
}
// IterateRaw ... XXX
func
(
fs
*
FileStorage
)
IterateRaw
(
dir
/*XXX fwd/back*/
)
*
Iter
{
// when iterating use IO optimized for sequential access
fsSeq
:=
xbufio
.
NewSeqReaderAt
(
fs
.
file
)
// XXX setup .TxnIter.dir and start
it
:=
&
Iter
{
fsSeq
:
fsSeq
}
// XXX ...
//it.DataIter.Txnh = &iter.txnIter.Txnh
return
it
}
// --- zodb.IStorage iteration ---
type
iterFlags
int
const
(
iterDir
iterFlags
=
1
<<
iota
// iterate forward (1) or backward (0)
iterEOF
// EOF reached
iterPreloaded
// data for this iteration was already preloaded
)
// zIter is transaction/data-records iterator as specified by zodb.IStorage
// zIter is combined transaction/data-records iterator as specified by zodb.IStorage
type
zIter
struct
{
iter
Iter
TidStop
zodb
.
Tid
// iterate up to tid <= tidStop | tid >= tidStop depending on .dir
Flags
i
terFlags
zFlags
zI
terFlags
// data header for data loading
// ( NOTE: need to use separate dh because x.LoadData() changes x state
...
...
@@ -995,17 +1012,23 @@ type zIter struct {
dataBuf
[]
byte
}
type
zIterFlags
int
const
(
zIterEOF
zIterFlags
=
1
<<
iota
// EOF reached
zIterPreloaded
// data for this iteration was already preloaded
)
// NextTxn iterates to next/previous transaction record according to iteration direction
func
(
zi
*
zIter
)
NextTxn
()
(
*
zodb
.
TxnInfo
,
zodb
.
IStorageRecordIterator
,
error
)
{
switch
{
case
zi
.
Flags
&
i
terEOF
!=
0
:
case
zi
.
zFlags
&
zI
terEOF
!=
0
:
//println("already eof")
return
nil
,
nil
,
io
.
EOF
// XXX needed?
case
zi
.
Flags
&
i
terPreloaded
!=
0
:
case
zi
.
zFlags
&
zI
terPreloaded
!=
0
:
// first element is already there - preloaded by who initialized TxnIter
zi
.
Flags
&=
^
i
terPreloaded
zi
.
zFlags
&=
^
zI
terPreloaded
//fmt.Println("preloaded:", zi.Txnh.Tid)
default
:
...
...
@@ -1017,10 +1040,10 @@ func (zi *zIter) NextTxn() (*zodb.TxnInfo, zodb.IStorageRecordIterator, error) {
}
// XXX how to make sure last good txnh is preserved?
if
(
zi
.
Flags
&
iterDir
!=
0
&&
zi
.
iter
.
Txnh
.
Tid
>
zi
.
TidStop
)
||
(
zi
.
Flags
&
iterDir
==
0
&&
zi
.
iter
.
Txnh
.
Tid
<
zi
.
TidStop
)
{
if
(
zi
.
iter
.
Dir
==
IterForward
&&
zi
.
iter
.
Txnh
.
Tid
>
zi
.
TidStop
)
||
(
zi
.
iter
.
Dir
==
IterBackward
&&
zi
.
iter
.
Txnh
.
Tid
<
zi
.
TidStop
)
{
//println("-> EOF")
zi
.
Flags
|=
i
terEOF
zi
.
zFlags
|=
zI
terEOF
return
nil
,
nil
,
io
.
EOF
}
...
...
@@ -1067,10 +1090,11 @@ type iterStartError struct {
err
error
}
func
(
e
*
iterStartError
)
NextTxn
(
*
zodb
.
TxnInfo
,
zodb
.
IStorage
Iterator
,
error
)
{
func
(
e
*
iterStartError
)
NextTxn
(
)
(
*
zodb
.
TxnInfo
,
zodb
.
IStorageRecord
Iterator
,
error
)
{
return
nil
,
nil
,
e
.
err
}
// Iterate creates zodb-level iterator for tidMin..tidMax range
func
(
fs
*
FileStorage
)
Iterate
(
tidMin
,
tidMax
zodb
.
Tid
)
zodb
.
IStorageIterator
{
//fmt.Printf("iterate %v..%v\n", tidMin, tidMax)
// FIXME case when only 0 or 1 txn present
...
...
@@ -1081,35 +1105,34 @@ func (fs *FileStorage) Iterate(tidMin, tidMax zodb.Tid) zodb.IStorageIterator {
tidMax
=
fs
.
txnhMax
.
Tid
}
// XXX naming -> ziter?
ziter
:=
&
zIter
{
iter
}
ziter
:=
&
zIter
{
iter
:
Iter
{}}
// when iterating use IO optimized for sequential access
// XXX -> IterateRaw
// XXX -> IterateRaw
?
fsSeq
:=
xbufio
.
NewSeqReaderAt
(
fs
.
file
)
iter
.
txnIter
.
fsSeq
=
fsSeq
iter
.
dataIter
.
fsSeq
=
fsSeq
iter
.
dataIter
.
Txnh
=
&
iter
.
txnIter
.
Txnh
ziter
.
iter
.
fsSeq
=
fsSeq
if
tidMin
>
tidMax
{
Iter
.
txnIter
.
Flags
|=
i
terEOF
// empty
return
&
I
ter
ziter
.
zFlags
|=
zI
terEOF
// empty
return
zi
ter
}
// scan either from file start or end, depending which way it is likely closer, to tidMin
// XXX put iter into ptr to Iter ^^^
iter
:=
&
Iter
.
txnI
ter
iter
:=
&
ziter
.
i
ter
if
(
tidMin
-
fs
.
txnhMin
.
Tid
)
<
(
fs
.
txnhMax
.
Tid
-
tidMin
)
{
//fmt.Printf("forward %.1f%%\n", 100 * float64(tidMin - fs.txnhMin.Tid) / float64(fs.txnhMax.Tid - fs.txnhMin.Tid))
iter
.
Flags
=
1
*
iterDir
|
iterPreloade
d
iter
.
Dir
=
IterForwar
d
iter
.
Txnh
.
CloneFrom
(
&
fs
.
txnhMin
)
iter
.
TidStop
=
tidMin
-
1
// XXX overflow
ziter
.
zFlags
=
zIterPreloaded
ziter
.
TidStop
=
tidMin
-
1
// XXX overflow
}
else
{
//fmt.Printf("backward %.1f%%\n", 100 * float64(tidMin - fs.txnhMin.Tid) / float64(fs.txnhMax.Tid - fs.txnhMin.Tid))
iter
.
Flags
=
0
*
iterDir
|
iterPreloade
d
iter
.
Dir
=
IterBackwar
d
iter
.
Txnh
.
CloneFrom
(
&
fs
.
txnhMax
)
iter
.
TidStop
=
tidMin
ziter
.
zFlags
=
zIterPreloaded
ziter
.
TidStop
=
tidMin
}
// XXX recheck how we enter loop
...
...
@@ -1130,19 +1153,19 @@ func (fs *FileStorage) Iterate(tidMin, tidMax zodb.Tid) zodb.IStorageIterator {
// where to start around tidMin found - let's reinitialize iter to
// iterate appropriately forward up to tidMax
iter
.
Flags
&=
^
i
terEOF
if
iter
.
Flags
&
iterDir
!=
0
{
ziter
.
zFlags
&=
^
zI
terEOF
if
iter
.
Dir
==
IterForward
{
// when ^^^ we were searching forward first txn was already found
err
=
iter
.
Txnh
.
loadStrings
(
fsSeq
)
// XXX ok? XXX -> move NextTxn() ?
if
err
!=
nil
{
return
&
iterStartError
{
err
}
}
iter
.
Flags
|=
i
terPreloaded
ziter
.
zFlags
|=
zI
terPreloaded
}
iter
.
Flags
|=
iterDir
iter
.
TidStop
=
tidMax
iter
.
Dir
=
IterForward
z
iter
.
TidStop
=
tidMax
return
&
I
ter
return
zi
ter
}
// computeIndex builds new in-memory index for FileStorage
...
...
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