Files
SugarDB/internal/raft/fsm.go

174 lines
4.6 KiB
Go

// Copyright 2024 Kelvin Clement Mwinuka
//
// 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.
package raft
import (
"context"
"encoding/json"
"fmt"
"github.com/echovault/echovault/internal"
"github.com/echovault/echovault/internal/config"
"github.com/hashicorp/raft"
"io"
"log"
"net"
"strings"
"time"
)
type FSMOpts struct {
Config config.Config
GetState func() map[string]internal.KeyData
GetCommand func(command string) (internal.Command, error)
CreateKeyAndLock func(ctx context.Context, key string) (bool, error)
SetValue func(ctx context.Context, key string, value interface{}) error
SetExpiry func(ctx context.Context, key string, expire time.Time, touch bool)
KeyUnlock func(ctx context.Context, key string)
DeleteKey func(ctx context.Context, key string) error
StartSnapshot func()
FinishSnapshot func()
SetLatestSnapshotTime func(msec int64)
GetHandlerFuncParams func(ctx context.Context, cmd []string, conn *net.Conn) internal.HandlerFuncParams
}
type FSM struct {
options FSMOpts
}
func NewFSM(opts FSMOpts) raft.FSM {
return raft.FSM(&FSM{
options: opts,
})
}
// Apply Implements raft.FSM interface
func (fsm *FSM) Apply(log *raft.Log) interface{} {
switch log.Type {
default:
// No-Op
case raft.LogCommand:
var request internal.ApplyRequest
if err := json.Unmarshal(log.Data, &request); err != nil {
return internal.ApplyResponse{
Error: err,
Response: nil,
}
}
ctx := context.WithValue(context.Background(), internal.ContextServerID("ServerID"), request.ServerID)
ctx = context.WithValue(ctx, internal.ContextConnID("ConnectionID"), request.ConnectionID)
switch strings.ToLower(request.Type) {
default:
return internal.ApplyResponse{
Error: fmt.Errorf("unsupported raft command type %s", request.Type),
Response: nil,
}
case "delete-key":
if err := fsm.options.DeleteKey(ctx, request.Key); err != nil {
return internal.ApplyResponse{
Error: err,
Response: nil,
}
}
return internal.ApplyResponse{
Error: nil,
Response: []byte("OK"),
}
case "command":
// Handle command
command, err := fsm.options.GetCommand(request.CMD[0])
if err != nil {
return internal.ApplyResponse{
Error: err,
Response: nil,
}
}
handler := command.HandlerFunc
subCommand, ok := internal.GetSubCommand(command, request.CMD).(internal.SubCommand)
if ok {
handler = subCommand.HandlerFunc
}
if res, err := handler(fsm.options.GetHandlerFuncParams(ctx, request.CMD, nil)); err != nil {
return internal.ApplyResponse{
Error: err,
Response: nil,
}
} else {
return internal.ApplyResponse{
Error: nil,
Response: res,
}
}
}
}
return nil
}
// Snapshot implements raft.FSM interface
func (fsm *FSM) Snapshot() (raft.FSMSnapshot, error) {
return NewFSMSnapshot(SnapshotOpts{
config: fsm.options.Config,
startSnapshot: fsm.options.StartSnapshot,
finishSnapshot: fsm.options.FinishSnapshot,
setLatestSnapshotTime: fsm.options.SetLatestSnapshotTime,
data: fsm.options.GetState(),
}), nil
}
// Restore implements raft.FSM interface
func (fsm *FSM) Restore(snapshot io.ReadCloser) error {
b, err := io.ReadAll(snapshot)
if err != nil {
log.Fatal(err)
return err
}
data := internal.SnapshotObject{
State: make(map[string]internal.KeyData),
LatestSnapshotMilliseconds: 0,
}
if err = json.Unmarshal(b, &data); err != nil {
log.Fatal(err)
return err
}
// Set state
ctx := context.Background()
for k, v := range internal.FilterExpiredKeys(data.State) {
if _, err = fsm.options.CreateKeyAndLock(ctx, k); err != nil {
log.Fatal(err)
}
if err = fsm.options.SetValue(ctx, k, v.Value); err != nil {
log.Fatal(err)
}
fsm.options.SetExpiry(ctx, k, v.ExpireAt, false)
fsm.options.KeyUnlock(ctx, k)
}
// Set latest snapshot milliseconds
fsm.options.SetLatestSnapshotTime(data.LatestSnapshotMilliseconds)
return nil
}