Commit 0f5cc13f authored by Kirill Smelkov's avatar Kirill Smelkov

.

parent 5a7e31ab
...@@ -110,6 +110,10 @@ func (txnh *TxnHeader) decodeErr(format string, a ...interface{}) *ErrTxnRecord ...@@ -110,6 +110,10 @@ func (txnh *TxnHeader) decodeErr(format string, a ...interface{}) *ErrTxnRecord
return txnh.errf("decode", format, a...) return txnh.errf("decode", format, a...)
} }
func (txnh *TxnHeader) bug(format string, a ...interface{}) {
panic(txnh.errf("bug", format, a...))
}
// ErrDataRecord is returned on data record read / decode errors // ErrDataRecord is returned on data record read / decode errors
type ErrDataRecord struct { type ErrDataRecord struct {
Pos int64 // position of data record Pos int64 // position of data record
...@@ -148,14 +152,23 @@ const ( ...@@ -148,14 +152,23 @@ const (
LoadNoStrings = 0x01 // do not load user/desc/ext strings LoadNoStrings = 0x01 // do not load user/desc/ext strings
) )
// it is software bug to pass invalid position to Load*() // // it is software bug to pass invalid position to Load*()
// this error will be paniced // // this error will be panicked
var bugPositionInvalid = errors.New("software bug: invalid position") // var bugPositionInvalid = errors.New("software bug: invalid position")
// Load reads and decodes transaction record header // Load reads and decodes transaction record header
// pos: points to transaction start // pos: points to transaction start
// no requirements are made to previous txnh state // no prerequisite requirements are made to previous txnh state
// TODO describe what happens at EOF and when .LenPrev is still valid // TODO describe what happens at EOF and when .LenPrev is still valid
//
// rules for Len/LenPrev returns:
// Len == 0 transaction header could not be read
// Len == -1 EOF is there when reading forward
// Len >= TxnHeaderFixSize transaction was read normally
//
// LenPrev == 0 error reading
// LenPrev == -1 EOF backward
// LenPrev >= TxnHeaderFixSize LenPrev was read/checked normally
func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLoadFlags) error { func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLoadFlags) error {
if cap(txnh.workMem) < txnXHeaderFixSize { if cap(txnh.workMem) < txnXHeaderFixSize {
txnh.workMem = make([]byte, txnXHeaderFixSize) // XXX or 0, ... ? txnh.workMem = make([]byte, txnXHeaderFixSize) // XXX or 0, ... ?
...@@ -164,40 +177,49 @@ func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLo ...@@ -164,40 +177,49 @@ func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLo
// XXX recheck rules about error exit // XXX recheck rules about error exit
txnh.Pos = pos txnh.Pos = pos
txnh.Len = 0 txnh.Len = 0 // read error
txnh.LenPrev = 0 txnh.LenPrev = 0 // read error
if pos < txnValidFrom { if pos < txnValidFrom {
panic(txnh.err("read", bugPositionInvalid)) txnh.bug("Load() on invalid position")
} }
var n int var n int
var err error var err error
if pos - 8 >= txnValidFrom { if pos - 8 >= txnValidFrom {
// read together with previous's txn record redundand length // read together with previous's txn record redundant length
n, err = r.ReadAt(work, pos - 8) n, err = r.ReadAt(work, pos - 8)
n -= 8 // relative to pos n -= 8 // relative to pos
if n >= 0 { if n >= 0 {
lenPrev := 8 + int64(binary.BigEndian.Uint64(work[8-8:])) lenPrev := 8 + int64(binary.BigEndian.Uint64(work[8-8:]))
if lenPrev < TxnHeaderFixSize { if lenPrev < TxnHeaderFixSize {
return txnh.decodeErr("invalid txn prev record length: %v", lenPrev) return txnh.decodeErr("invalid prev record length: %v", lenPrev)
}
posPrev := txnh.Pos - lenPrev
if posPrev < txnValidFrom {
return txnh.decodeErr("prev record length goes beyond valid area: %v", lenPrev)
}
if posPrev < txnValidFrom + TxnHeaderFixSize && posPrev != txnValidFrom {
return txnh.decodeErr("prev record does not land exactly at valid area start: %v", posPrev)
} }
txnh.LenPrev = lenPrev txnh.LenPrev = lenPrev
} }
} else { } else {
// read only current txn without previous record length // read only current txn without previous record length
n, err = r.ReadAt(work[8:], pos) n, err = r.ReadAt(work[8:], pos)
txnh.LenPrev = -1 // EOF backward
} }
if err != nil { if err != nil {
if err == io.EOF && n == 0 { if err == io.EOF && n == 0 {
txnh.Len = -1 // EOF forward
return err // end of stream return err // end of stream
} }
// EOF after txn header is not good - because at least // EOF after txn header is not good - because at least
// redundand lenght should be also there // redundant length should be also there
return txnh.err("read", noEOF(err)) return txnh.err("read", noEOF(err))
} }
...@@ -210,6 +232,7 @@ func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLo ...@@ -210,6 +232,7 @@ func (txnh *TxnHeader) Load(r io.ReaderAt /* *os.File */, pos int64, flags TxnLo
if tlen < TxnHeaderFixSize { if tlen < TxnHeaderFixSize {
return txnh.decodeErr("invalid txn record length: %v", tlen) return txnh.decodeErr("invalid txn record length: %v", tlen)
} }
// XXX also check tlen to not go beyond file size ?
txnh.Len = tlen txnh.Len = tlen
txnh.Status = zodb.TxnStatus(work[8+16]) txnh.Status = zodb.TxnStatus(work[8+16])
...@@ -263,31 +286,45 @@ func (txnh *TxnHeader) loadStrings(r io.ReaderAt /* *os.File */) error { ...@@ -263,31 +286,45 @@ func (txnh *TxnHeader) loadStrings(r io.ReaderAt /* *os.File */) error {
} }
// LoadPrev reads and decodes previous transaction record header // LoadPrev reads and decodes previous transaction record header
// txnh should be already initialized by previous call to load() // prerequisite: txnh .Pos and .LenPrev should be already initialized:
// - by successful call to Load() initially XXX but EOF also works
// - by subsequent successful calls to LoadPrev / LoadNext XXX recheck
func (txnh *TxnHeader) LoadPrev(r io.ReaderAt, flags TxnLoadFlags) error { func (txnh *TxnHeader) LoadPrev(r io.ReaderAt, flags TxnLoadFlags) error {
lenPrev := txnh.LenPrev lenPrev := txnh.LenPrev
if lenPrev == 0 { switch lenPrev { // XXX recheck states for: 1) LenPrev load error 2) EOF
case 0:
txnh.bug("LoadPrev() when .LenPrev == error")
case -1:
return io.EOF return io.EOF
} }
// here we know: Load already checked txnh.Pos - lenPrev to be valid position
err := txnh.Load(r, txnh.Pos - lenPrev, flags) err := txnh.Load(r, txnh.Pos - lenPrev, flags)
if err != nil { if err != nil {
return err return err
} }
if txnh.Len != lenPrev { if txnh.Len != lenPrev {
return txnh.decodeErr("redundant lengths mismatch: %v, %v", txnh.Len, lenPrev) return txnh.decodeErr("head/tail lengths mismatch: %v, %v", txnh.Len, lenPrev)
} }
return nil return nil
} }
// LoadNext reads and decodes next transaction record header // LoadNext reads and decodes next transaction record header
// txnh should be already initialized by previous call to load() // prerequisite: txnh .Pos and .Len should be already initialized by:
// - previous successful call to Load() initially XXX ^^^
// - TODO
func (txnh *TxnHeader) LoadNext(r io.ReaderAt, flags TxnLoadFlags) error { func (txnh *TxnHeader) LoadNext(r io.ReaderAt, flags TxnLoadFlags) error {
lenCur := txnh.Len lenCur := txnh.Len
switch lenCur {
case 0:
txhn.bug("LoadNext() when .Len == error")
case -1:
return io.EOF
}
err := txnh.Load(r, txnh.Pos + txnh.Len, flags) err := txnh.Load(r, txnh.Pos + lenCur, flags)
// TODO XXX // TODO XXX
if txnh.LenPrev != lenCur { if txnh.LenPrev != lenCur {
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment