Finished index with its tests.
This commit is contained in:
parent
093422a2f9
commit
525abb53fd
3 changed files with 166 additions and 0 deletions
100
internal/log/index.go
Normal file
100
internal/log/index.go
Normal file
|
|
@ -0,0 +1,100 @@
|
|||
package log
|
||||
|
||||
import (
|
||||
"io"
|
||||
"os"
|
||||
|
||||
"github.com/tysontate/gommap"
|
||||
)
|
||||
|
||||
var (
|
||||
offWidth uint64 = 4
|
||||
posWidth uint64 = 8
|
||||
entWidth = offWidth + posWidth
|
||||
)
|
||||
|
||||
type index struct {
|
||||
file *os.File
|
||||
mmap gommap.MMap
|
||||
size uint64
|
||||
}
|
||||
|
||||
func newIndex(f *os.File, c Config) (*index, error) {
|
||||
idx := &index{
|
||||
file: f,
|
||||
}
|
||||
fi, err := os.Stat(f.Name())
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// Get and store the file size.
|
||||
idx.size = uint64(fi.Size())
|
||||
// Grow the file to its max size so it can be memory mapped as a whole.
|
||||
if err = os.Truncate(
|
||||
f.Name(), int64(c.Segment.MaxIndexBytes),
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
// Map the grown file to memory.
|
||||
if idx.mmap, err = gommap.Map(
|
||||
idx.file.Fd(),
|
||||
gommap.PROT_READ|gommap.PROT_WRITE,
|
||||
gommap.MAP_SHARED,
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return idx, nil
|
||||
}
|
||||
|
||||
func (i *index) Read(in int64) (out uint32, pos uint64, err error) {
|
||||
if i.size == 0 {
|
||||
return 0, 0, io.EOF
|
||||
}
|
||||
if in == -1 {
|
||||
// The last entity was requested
|
||||
out = uint32((i.size / entWidth) - 1)
|
||||
} else {
|
||||
out = uint32(in)
|
||||
}
|
||||
|
||||
pos = uint64(out) * entWidth
|
||||
|
||||
// Check that the index actually contains the record with the given offset.
|
||||
if i.size < pos+entWidth {
|
||||
return 0, 0, io.EOF
|
||||
}
|
||||
|
||||
out = enc.Uint32(i.mmap[pos : pos+offWidth])
|
||||
pos = enc.Uint64(i.mmap[pos+offWidth : pos+entWidth])
|
||||
return out, pos, nil
|
||||
}
|
||||
|
||||
func (i *index) Write(off uint32, pos uint64) error {
|
||||
if uint64(len(i.mmap)) < i.size+entWidth {
|
||||
return io.EOF
|
||||
}
|
||||
enc.PutUint32(i.mmap[i.size:i.size+offWidth], off)
|
||||
enc.PutUint64(i.mmap[i.size+offWidth:i.size+entWidth], pos)
|
||||
i.size += entWidth
|
||||
return nil
|
||||
}
|
||||
|
||||
func (i *index) Name() string {
|
||||
return i.file.Name()
|
||||
}
|
||||
|
||||
func (i *index) Close() error {
|
||||
if err := i.mmap.Sync(gommap.MS_SYNC); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := i.file.Sync(); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := i.file.Truncate(int64(i.size)); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return i.file.Close()
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue