Skip to content

vadiminshakov/gowal

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

88 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

tests Go Reference Go Report Card

GoWAL - Write-Ahead Logging in Go

GoWAL is a simple, efficient Write-Ahead Log (WAL) library written in Go. It allows you to store logs as an append-only log structure, which is useful for applications that require crash recovery, transaction logging, or high-availability systems. GoWAL is optimized for performance with configurable segment rotation and in-memory indexing.

Features

  • Append-only log structure: Ensures that data is only appended and never modified or deleted, which is crucial for durability and integrity.
  • Segmented logs: Automatically rotates segments after a configurable threshold is reached. Older segments are deleted to save disk space.
  • Efficient lookups: In-memory index allows for quick lookups of log entries by their index.
  • Persistence: Logs and their indexes are stored on disk and reloaded into memory upon initialization.
  • Configurable sync mode: Option to sync logs to disk after every write to ensure data durability, though at the cost of speed.
  • Checksums: Each log segment has an associated checksum file to ensure data integrity.

Installation

go get github.com/vadiminshakov/gowal

Usage

Initialization

To create a new WAL instance, specify the directory to store logs and a prefix for the log files:

import "github.com/vadiminshakov/gowal"

cfg := gowal.Config{
    Dir:    "./log",
    Prefix: "segment_",
    SegmentThreshold: 1000,
    MaxSegments:      100,
    IsInSyncDiskMode: false,
})

wal, err := gowal.NewWAL(cfg)
if err != nil {
    log.Fatal(err)
}
defer wal.Close()

Adding a log entry

You can append a new log entry by providing an index, a key, and a value:

err := wal.Write(1, "myKey", []byte("myValue"))
if err != nil {
    log.Fatal(err)
}

If the entry with the same index already exists, the function will return an error.

Retrieving a log entry

You can retrieve a log entry by its index:

key, value, found := wal.Get(1)
if !found {
    log.Println("Entry not found")
} else {
    log.Printf("Key: %s, Value: %s", key, string(value))
}

Iterating over log entries

You can iterate over all log entries using the Iterate function:

iter := wal.Iterator()
for msg, ok := iter() ; ok; msg, ok = iter() {
    log.Printf("Key: %s, Value: %s\n", msg.Key, string(msg.Value))
}

Closing the WAL

Always ensure that you close the WAL instance to properly flush and close the log files:

err := wal.Close()
if err != nil {
    log.Fatal(err)
}

Recover corrupted WAL

If the WAL is corrupted, you can recover it by calling the UnsafeRecover function:

removedFiles, err := wal.UnsafeRecover("./wal", "segment_")

Configuration

The behavior of the WAL can be configured using several configuration options (Config parameter in the NewWAL function):

  • SegmentThreshold: Maximum number of log entries per segment before rotation occurs. Default is 1000.
  • MaxSegments: Maximum number of segments to keep before the oldest segments are deleted. Default is 5.
  • IsInSyncDiskMode: When set to true, every write is synced to disk, ensuring durability at the cost of performance. Default is false.

Contributing

Feel free to open issues or submit pull requests for improvements and bug fixes. We welcome contributions!

License

This project is licensed under the Apache License.