etcd/wal/wal.go

220 lines
4.2 KiB
Go
Raw Normal View History

2014-07-28 07:58:39 +04:00
/*
Copyright 2014 CoreOS Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
2014-07-26 01:20:21 +04:00
package wal
import (
"bufio"
2014-07-26 11:10:59 +04:00
"bytes"
2014-07-26 01:20:21 +04:00
"encoding/binary"
"fmt"
2014-07-26 11:10:59 +04:00
"io"
2014-08-03 20:51:10 +04:00
"log"
2014-07-26 01:20:21 +04:00
"os"
"github.com/coreos/etcd/raft"
)
var (
infoType = int64(1)
entryType = int64(2)
stateType = int64(3)
)
type WAL struct {
2014-07-28 07:28:41 +04:00
f *os.File
bw *bufio.Writer
buf *bytes.Buffer
2014-07-26 01:20:21 +04:00
}
2014-07-28 07:32:23 +04:00
func newWAL(f *os.File) *WAL {
return &WAL{f, bufio.NewWriter(f), new(bytes.Buffer)}
}
2014-07-26 01:20:21 +04:00
func New(path string) (*WAL, error) {
2014-08-03 20:51:10 +04:00
log.Printf("path=%s wal.new", path)
2014-07-26 01:54:08 +04:00
f, err := os.Open(path)
if err == nil {
f.Close()
return nil, os.ErrExist
}
f, err = os.Create(path)
2014-07-26 01:20:21 +04:00
if err != nil {
return nil, err
}
2014-07-28 07:32:23 +04:00
return newWAL(f), nil
2014-07-26 01:20:21 +04:00
}
2014-07-26 11:10:59 +04:00
func Open(path string) (*WAL, error) {
2014-08-03 20:51:10 +04:00
log.Printf("path=%s wal.open", path)
f, err := os.OpenFile(path, os.O_RDWR, 0)
2014-07-26 11:10:59 +04:00
if err != nil {
return nil, err
}
2014-07-28 07:32:23 +04:00
return newWAL(f), nil
2014-07-26 11:10:59 +04:00
}
2014-07-30 01:47:36 +04:00
func (w *WAL) Sync() error {
if err := w.bw.Flush(); err != nil {
return err
}
return w.f.Sync()
2014-07-28 08:01:39 +04:00
}
2014-07-26 01:54:08 +04:00
func (w *WAL) Close() {
2014-08-03 20:51:10 +04:00
log.Printf("path=%s wal.close", w.f.Name())
2014-07-26 01:54:08 +04:00
if w.f != nil {
2014-07-30 01:47:36 +04:00
w.Sync()
2014-07-26 01:54:08 +04:00
w.f.Close()
}
}
2014-08-04 07:44:02 +04:00
func (w *WAL) SaveInfo(i *raft.Info) error {
log.Printf("path=%s wal.saveInfo id=%d", w.f.Name(), i.Id)
2014-07-26 11:10:59 +04:00
if err := w.checkAtHead(); err != nil {
2014-07-26 01:20:21 +04:00
return err
}
2014-08-04 07:44:02 +04:00
b, err := i.Marshal()
2014-07-28 04:23:04 +04:00
if err != nil {
panic(err)
2014-07-26 01:20:21 +04:00
}
2014-08-05 02:46:12 +04:00
rec := &Record{Type: infoType, Data: b}
return writeRecord(w.bw, rec)
2014-07-26 01:20:21 +04:00
}
2014-07-28 04:23:04 +04:00
func (w *WAL) SaveEntry(e *raft.Entry) error {
2014-08-03 05:21:25 +04:00
b, err := e.Marshal()
2014-07-26 01:20:21 +04:00
if err != nil {
2014-07-28 04:23:04 +04:00
panic(err)
2014-07-26 01:20:21 +04:00
}
2014-08-05 02:46:12 +04:00
rec := &Record{Type: entryType, Data: b}
return writeRecord(w.bw, rec)
2014-07-26 01:20:21 +04:00
}
2014-07-28 04:23:04 +04:00
func (w *WAL) SaveState(s *raft.State) error {
2014-08-03 20:51:10 +04:00
log.Printf("path=%s wal.saveState state=\"%+v\"", w.f.Name(), s)
2014-08-04 04:38:53 +04:00
b, err := s.Marshal()
2014-07-28 04:23:04 +04:00
if err != nil {
panic(err)
2014-07-26 01:20:21 +04:00
}
2014-08-05 02:46:12 +04:00
rec := &Record{Type: stateType, Data: b}
return writeRecord(w.bw, rec)
2014-07-26 01:20:21 +04:00
}
2014-07-26 11:10:59 +04:00
func (w *WAL) checkAtHead() error {
o, err := w.f.Seek(0, os.SEEK_CUR)
if err != nil {
return err
}
if o != 0 || w.bw.Buffered() != 0 {
return fmt.Errorf("cannot write info at %d, expect 0", max(o, int64(w.bw.Buffered())))
}
return nil
}
type Node struct {
Id int64
Ents []raft.Entry
State raft.State
}
2014-07-28 04:23:04 +04:00
func (w *WAL) LoadNode() (*Node, error) {
2014-08-03 20:51:10 +04:00
log.Printf("path=%s wal.loadNode", w.f.Name())
2014-07-26 11:10:59 +04:00
if err := w.checkAtHead(); err != nil {
return nil, err
}
br := bufio.NewReader(w.f)
2014-08-05 02:46:12 +04:00
rec := &Record{}
2014-07-26 11:10:59 +04:00
2014-08-05 02:46:12 +04:00
err := readRecord(br, rec)
2014-07-26 11:10:59 +04:00
if err != nil {
return nil, err
}
2014-08-05 02:46:12 +04:00
if rec.Type != infoType {
return nil, fmt.Errorf("the first block of wal is not infoType but %d", rec.Type)
2014-07-28 04:23:04 +04:00
}
2014-08-05 02:46:12 +04:00
i, err := loadInfo(rec.Data)
2014-07-28 04:23:04 +04:00
if err != nil {
return nil, err
2014-07-26 11:10:59 +04:00
}
ents := make([]raft.Entry, 0)
var state raft.State
2014-08-05 02:46:12 +04:00
for err = readRecord(br, rec); err == nil; err = readRecord(br, rec) {
switch rec.Type {
2014-07-26 11:10:59 +04:00
case entryType:
2014-08-05 02:46:12 +04:00
e, err := loadEntry(rec.Data)
2014-07-26 11:10:59 +04:00
if err != nil {
return nil, err
}
2014-08-02 08:58:18 +04:00
ents = append(ents[:e.Index-1], e)
2014-07-26 11:10:59 +04:00
case stateType:
2014-08-05 02:46:12 +04:00
s, err := loadState(rec.Data)
2014-07-26 11:10:59 +04:00
if err != nil {
return nil, err
}
state = s
default:
2014-08-05 02:46:12 +04:00
return nil, fmt.Errorf("unexpected block type %d", rec.Type)
2014-07-26 11:10:59 +04:00
}
}
2014-07-28 04:23:04 +04:00
if err != io.EOF {
return nil, err
}
2014-08-04 07:44:02 +04:00
return &Node{i.Id, ents, state}, nil
2014-07-26 11:10:59 +04:00
}
2014-08-04 07:44:02 +04:00
func loadInfo(d []byte) (raft.Info, error) {
var i raft.Info
err := i.Unmarshal(d)
if err != nil {
panic(err)
2014-07-26 11:10:59 +04:00
}
2014-08-04 07:44:02 +04:00
return i, err
2014-07-26 11:10:59 +04:00
}
2014-07-28 04:23:04 +04:00
func loadEntry(d []byte) (raft.Entry, error) {
2014-07-26 11:10:59 +04:00
var e raft.Entry
2014-08-03 05:21:25 +04:00
err := e.Unmarshal(d)
if err != nil {
panic(err)
}
2014-07-26 11:10:59 +04:00
return e, err
}
2014-07-28 04:23:04 +04:00
func loadState(d []byte) (raft.State, error) {
2014-07-26 11:10:59 +04:00
var s raft.State
2014-08-04 04:38:53 +04:00
err := s.Unmarshal(d)
2014-07-26 11:10:59 +04:00
return s, err
}
2014-07-28 04:23:04 +04:00
func writeInt64(w io.Writer, n int64) error {
return binary.Write(w, binary.LittleEndian, n)
2014-07-26 11:10:59 +04:00
}
func readInt64(r io.Reader) (int64, error) {
var n int64
err := binary.Read(r, binary.LittleEndian, &n)
return n, err
}
2014-07-26 02:21:04 +04:00
func max(a, b int64) int64 {
if a > b {
return a
}
return b
}