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
1
Issues
1
List
Boards
Labels
Milestones
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
neoppod
Commits
b0b596a2
Commit
b0b596a2
authored
Jul 25, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
38a9a71d
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
101 additions
and
64 deletions
+101
-64
go/zodb/storage/fs1/filestorage.go
go/zodb/storage/fs1/filestorage.go
+38
-0
go/zodb/storage/fs1/fs1tools/dump.go
go/zodb/storage/fs1/fs1tools/dump.go
+63
-64
No files found.
go/zodb/storage/fs1/filestorage.go
View file @
b0b596a2
...
@@ -786,6 +786,44 @@ func Iterate(r io.ReaderAt, posStart int64, dir IterDir) *Iter {
...
@@ -786,6 +786,44 @@ func Iterate(r io.ReaderAt, posStart int64, dir IterDir) *Iter {
}
}
// IterateFile opens file @ path and creates Iter to iterate over it.
// The iteration will use buffering over os.File optimized for sequential access.
// You are responsible to eventually close the file after the iteration is done.
func
IterateFile
(
path
string
,
dir
IterDir
)
(
iter
*
Iter
,
file
*
os
.
File
,
err
error
)
{
f
,
err
:=
os
.
Open
(
path
)
if
err
!=
nil
{
return
nil
,
nil
,
err
}
// close file in case we return with an error
defer
func
()
{
if
err
!=
nil
{
f
.
Close
()
}
}()
// use IO optimized for sequential access when iterating
fSeq
:=
xbufio
.
NewSeqReaderAt
(
f
)
switch
dir
{
case
IterForward
:
return
Iterate
(
fSeq
,
txnValidFrom
,
IterForward
),
f
,
nil
case
IterBackward
:
// get file size as topPos
fi
,
err
:=
f
.
Stat
()
if
err
!=
nil
{
return
nil
,
nil
,
err
}
topPos
:=
fi
.
Size
()
return
Iterate
(
fSeq
,
topPos
,
IterBackward
),
f
,
nil
default
:
panic
(
"dir invalid"
)
}
}
// --- FileStorage ---
// --- FileStorage ---
// FileStorage is a ZODB storage which stores data in simple append-only file
// FileStorage is a ZODB storage which stores data in simple append-only file
...
...
go/zodb/storage/fs1/fs1tools/dump.go
View file @
b0b596a2
...
@@ -46,22 +46,23 @@ func Dump(w io.Writer, path string, options DumpOptions) (err error) {
...
@@ -46,22 +46,23 @@ func Dump(w io.Writer, path string, options DumpOptions) (err error) {
d
=
&
dumper1
{}
d
=
&
dumper1
{}
}
}
return
dump
(
w
,
path
,
d
)
return
dump
(
w
,
path
,
fs1
.
IterForward
,
d
)
// XXX hardcoded
}
}
type
DumpOptions
struct
{
type
DumpOptions
struct
{
Verbose
bool
// dump in verbose mode
Verbose
bool
// dump in verbose mode
}
}
func
dump
(
w
io
.
Writer
,
path
string
,
d
dumper
)
(
err
error
)
{
func
dump
(
w
io
.
Writer
,
path
string
,
d
ir
fs1
.
IterDir
,
d
dumper
)
(
err
error
)
{
defer
xerr
.
Contextf
(
&
err
,
"%s: fsdump"
,
path
)
// XXX ok?
defer
xerr
.
Contextf
(
&
err
,
"%s: fsdump"
,
path
)
// XXX ok?
fs
,
err
:=
fs1
.
Open
(
path
,
read
-
only
,
no
-
index
)
it
,
f
,
err
:=
fs1
.
IterateFile
(
path
,
dir
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
err
}
}
defer
func
()
{
defer
func
()
{
err2
:=
f
s
.
Close
()
err2
:=
f
.
Close
()
err
=
xerr
.
First
(
err
,
err2
)
err
=
xerr
.
First
(
err
,
err2
)
}()
}()
...
@@ -79,10 +80,16 @@ func dump(w io.Writer, path string, d dumper) (err error) {
...
@@ -79,10 +80,16 @@ func dump(w io.Writer, path string, d dumper) (err error) {
err
=
xerr
.
First
(
err
,
err2
)
err
=
xerr
.
First
(
err
,
err2
)
}()
}()
// TODO d.dumpFileHeader
// file header
var
fh
fs1
.
FileHeader
err
=
fh
.
Load
(
it
.
R
)
if
err
!=
nil
{
return
err
}
d
.
dumpFileHeader
(
buf
,
&
fh
)
it
:=
fs
.
IterateRaw
(
fs1
.
IterForward
)
// iter over txn/data
for
{
for
{
err
=
it
.
NextTxn
(
fs1
.
LoadAll
)
err
=
it
.
NextTxn
(
fs1
.
LoadAll
)
if
err
!=
nil
{
if
err
!=
nil
{
...
@@ -92,24 +99,14 @@ func dump(w io.Writer, path string, d dumper) (err error) {
...
@@ -92,24 +99,14 @@ func dump(w io.Writer, path string, d dumper) (err error) {
return
err
return
err
}
}
d
.
dumpTxn
(
buf
,
it
.
Txnh
)
// XXX err
err
=
d
.
dumpTxn
(
buf
,
it
)
if
err
!=
nil
{
for
{
if
err
==
io
.
EOF
{
err
=
it
.
NextData
()
err
=
nil
// XXX -> okEOF(err)
if
err
!=
nil
{
if
err
==
io
.
EOF
{
err
=
nil
// XXX -> okEOF(err)
break
}
return
err
}
}
return
err
d
.
dumpData
(
buf
,
it
.
Datah
)
// XXX err
}
}
d
.
dumpTxnPost
(
buf
,
it
.
Txnh
)
// XXX err
err
=
flushBuf
()
err
=
flushBuf
()
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
err
...
@@ -121,14 +118,13 @@ func dump(w io.Writer, path string, d dumper) (err error) {
...
@@ -121,14 +118,13 @@ func dump(w io.Writer, path string, d dumper) (err error) {
type
dumper
interface
{
type
dumper
interface
{
dumpFileHeader
(
*
xfmt
.
Buffer
,
*
fs1
.
FileHeader
)
error
dumpFileHeader
(
*
xfmt
.
Buffer
,
*
fs1
.
FileHeader
)
error
dumpTxn
(
*
xfmt
.
Buffer
,
*
fs1
.
Iter
)
error
dumpTxn
(
*
xfmt
.
Buffer
,
*
fs1
.
Iter
)
error
dumpData
(
*
xfmt
.
Buffer
,
*
fs1
.
Iter
)
error
//
dumpData(*xfmt.Buffer, *fs1.Iter) error
dumpTxnPost
(
*
xfmt
.
Buffer
,
*
fs1
.
Iter
)
error
//
dumpTxnPost(*xfmt.Buffer, *fs1.Iter) error
}
}
// "normal" dumper
// "normal" dumper
XXX link to zodb/py
type
dumper1
struct
{
type
dumper1
struct
{
ntxn
int
// current transaction record #
ntxn
int
// current transaction record #
ndata
int
// current data record # inside current transaction
}
}
func
(
d
*
dumper1
)
dumpFileHeader
(
buf
*
xfmt
.
Buffer
,
fh
*
fs1
.
FileHeader
)
error
{
func
(
d
*
dumper1
)
dumpFileHeader
(
buf
*
xfmt
.
Buffer
,
fh
*
fs1
.
FileHeader
)
error
{
...
@@ -144,51 +140,52 @@ func (d *dumper1) dumpTxn(buf *xfmt.Buffer, it *fs1.Iter) error {
...
@@ -144,51 +140,52 @@ func (d *dumper1) dumpTxn(buf *xfmt.Buffer, it *fs1.Iter) error {
buf
.
S
(
"
\n
status="
)
.
Qpycb
(
byte
(
txnh
.
Status
))
buf
.
S
(
"
\n
status="
)
.
Qpycb
(
byte
(
txnh
.
Status
))
buf
.
S
(
" user="
)
.
Qpyb
(
txnh
.
User
)
buf
.
S
(
" user="
)
.
Qpyb
(
txnh
.
User
)
buf
.
S
(
" description="
)
.
Qpyb
(
txnh
.
Description
)
.
S
(
"
\n
"
)
buf
.
S
(
" description="
)
.
Qpyb
(
txnh
.
Description
)
.
S
(
"
\n
"
)
d
.
ntxn
++
d
.
ntxn
++
d
.
ndata
=
0
return
nil
}
func
(
d
*
dumper1
)
dumpData
(
buf
*
xfmt
.
Buffer
,
it
*
fs1
.
Iter
)
error
{
dh
:=
&
it
.
Datah
buf
.
S
(
" data #"
)
buf
.
S
(
fmt
.
Sprintf
(
"%05d"
,
d
.
ndata
))
// XXX -> .D_f("05", d.ndata)
buf
.
S
(
" oid="
)
.
V
(
dh
.
Oid
)
if
dh
.
DataLen
==
0
{
for
j
:=
0
;
;
j
++
{
buf
.
S
(
" class=undo or abort of object creation"
)
err
:=
it
.
NextData
()
backPos
,
err
:=
dh
.
LoadBackRef
(
it
.
R
)
if
err
!=
nil
{
if
err
!=
nil
{
// XXX
if
err
==
io
.
EOF
{
break
}
return
err
}
}
if
backPos
!=
0
{
dh
:=
&
it
.
Datah
buf
.
S
(
" bp="
)
.
X016
(
uint64
(
backPos
))
buf
.
S
(
" data #"
)
}
buf
.
S
(
fmt
.
Sprintf
(
"%05d"
,
j
))
// XXX -> .D_f("05", j)
}
else
{
buf
.
S
(
" oid="
)
.
V
(
dh
.
Oid
)
// XXX Datah.LoadData()
//modname, classname = zodb.GetPickleMetadata(...) // XXX
//fullclass = "%s.%s" % (modname, classname)
fullclass
:=
"AAA.BBB"
// FIXME stub
buf
.
S
(
" size="
)
.
D64
(
dh
.
DataLen
)
if
dh
.
DataLen
==
0
{
buf
.
S
(
" class="
)
.
S
(
fullclass
)
buf
.
S
(
" class=undo or abort of object creation"
)
}
buf
.
S
(
"
\n
"
)
backPos
,
err
:=
dh
.
LoadBackRef
(
it
.
R
)
if
err
!=
nil
{
// XXX
}
d
.
ndata
++
if
backPos
!=
0
{
return
nil
buf
.
S
(
" bp="
)
.
X016
(
uint64
(
backPos
))
}
}
}
else
{
// XXX Datah.LoadData()
//modname, classname = zodb.GetPickleMetadata(...) // XXX
//fullclass = "%s.%s" % (modname, classname)
fullclass
:=
"AAA.BBB"
// FIXME stub
buf
.
S
(
" size="
)
.
D64
(
dh
.
DataLen
)
buf
.
S
(
" class="
)
.
S
(
fullclass
)
}
buf
.
S
(
"
\n
"
)
}
func
(
d
*
dumper1
)
dumpTxnPost
(
buf
*
xfmt
.
Buffer
,
it
*
fs1
.
Iter
)
error
{
return
nil
return
nil
}
}
// ----------------------------------------
// ----------------------------------------
// verbose dumper with output identical to fsdump.Dumper in zodb/py
type
dumperVerbose
struct
{
type
dumperVerbose
struct
{
}
}
...
@@ -215,6 +212,16 @@ func (d *dumperVerbose) dumpTxn(buf *xfmt.Buffer, it *fs1.Iter) error {
...
@@ -215,6 +212,16 @@ func (d *dumperVerbose) dumpTxn(buf *xfmt.Buffer, it *fs1.Iter) error {
buf
.
S
(
"
\n
description: "
)
.
Qpyb
(
txnh
.
Description
)
buf
.
S
(
"
\n
description: "
)
.
Qpyb
(
txnh
.
Description
)
buf
.
S
(
"
\n
len(extra): "
)
.
D
(
len
(
txnh
.
Extension
))
buf
.
S
(
"
\n
len(extra): "
)
.
D
(
len
(
txnh
.
Extension
))
buf
.
S
(
"
\n
"
)
buf
.
S
(
"
\n
"
)
err
:=
d
.
dumpData
(
buf
,
it
)
if
err
!=
nil
{
return
err
}
// NOTE printing the same .Len twice
// we do not print/check redundant len here because our
// FileStorage code checks/reports this itself
buf
.
S
(
"redundant trec len: "
)
.
D64
(
it
.
Txnh
.
Len
)
.
S
(
"
\n
"
)
return
nil
return
nil
}
}
...
@@ -242,11 +249,3 @@ func (d *dumperVerbose) dumpData(buf *xfmt.Buffer, it *fs1.Iter) error {
...
@@ -242,11 +249,3 @@ func (d *dumperVerbose) dumpData(buf *xfmt.Buffer, it *fs1.Iter) error {
buf
.
S
(
"
\n
"
)
buf
.
S
(
"
\n
"
)
return
nil
return
nil
}
}
func
(
d
*
dumperVerbose
)
dumpTxnPost
(
buf
*
xfmt
.
Buffer
,
it
*
fs1
.
Iter
)
error
{
// NOTE printing the same .Len twice
// we do not print/check redundant len here because our
// FileStorage code checks/reports this itself
buf
.
S
(
"redundant trec len: "
)
.
D64
(
it
.
Txnh
.
Len
)
.
S
(
"
\n
"
)
return
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