mirror of
https://github.com/EchoVault/SugarDB.git
synced 2025-09-26 20:11:15 +08:00
296 lines
8.2 KiB
Go
296 lines
8.2 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 sugardb
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"github.com/echovault/sugardb/internal"
|
|
"io/fs"
|
|
"os"
|
|
"plugin"
|
|
"slices"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
func (server *SugarDB) AddScript(engine string, scriptType string, content string, args []string) error {
|
|
return nil
|
|
}
|
|
|
|
func (server *SugarDB) AddScriptCommand(
|
|
path string,
|
|
args []string,
|
|
) error {
|
|
// Extract the engine from the script file extension
|
|
var engine string
|
|
if strings.HasSuffix(path, ".lua") {
|
|
engine = "lua"
|
|
} else if strings.HasSuffix(path, ".js") {
|
|
engine = "js"
|
|
}
|
|
|
|
// Check if the engine is supported
|
|
supportedEngines := []string{"lua", "js"}
|
|
if !slices.Contains(supportedEngines, strings.ToLower(engine)) {
|
|
return fmt.Errorf("engine %s not supported, only %v engines are supported", engine, supportedEngines)
|
|
}
|
|
|
|
// Initialise VM for the command depending on the engine.
|
|
var vm any
|
|
var commandName string
|
|
var categories []string
|
|
var description string
|
|
var synchronize bool
|
|
var commandType string
|
|
var err error
|
|
|
|
switch strings.ToLower(engine) {
|
|
case "lua":
|
|
vm, commandName, categories, description, synchronize, commandType, err = generateLuaCommandInfo(path)
|
|
case "js":
|
|
vm, commandName, categories, description, synchronize, commandType, err = generateJSCommandInfo(path)
|
|
}
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Save the script's VM to the server's list of VMs.
|
|
server.scriptVMs.Store(commandName, struct {
|
|
vm any
|
|
lock *sync.Mutex
|
|
}{
|
|
vm: vm,
|
|
// lock is the script mutex for the commands.
|
|
// This mutex will be locked everytime the command is executed because
|
|
// the script's VM is not thread safe.
|
|
lock: &sync.Mutex{},
|
|
})
|
|
|
|
// Build the command:
|
|
command := internal.Command{
|
|
Command: commandName,
|
|
Module: path,
|
|
Categories: categories,
|
|
Description: description,
|
|
Sync: synchronize,
|
|
Type: commandType,
|
|
KeyExtractionFunc: func(engine string, args []string) internal.KeyExtractionFunc {
|
|
// Wrapper for the key function
|
|
return func(cmd []string) (internal.KeyExtractionFuncResult, error) {
|
|
switch strings.ToLower(engine) {
|
|
default:
|
|
return internal.KeyExtractionFuncResult{
|
|
Channels: make([]string, 0),
|
|
ReadKeys: make([]string, 0),
|
|
WriteKeys: make([]string, 0),
|
|
}, nil
|
|
case "lua":
|
|
return server.luaKeyExtractionFunc(cmd, args)
|
|
case "js":
|
|
return server.jsKeyExtractionFunc(cmd, args)
|
|
}
|
|
}
|
|
}(engine, args),
|
|
HandlerFunc: func(engine string, args []string) internal.HandlerFunc {
|
|
// Wrapper that generates handler function
|
|
return func(params internal.HandlerFuncParams) ([]byte, error) {
|
|
switch strings.ToLower(engine) {
|
|
default:
|
|
return nil, fmt.Errorf("command %s handler not implemented", commandName)
|
|
case "lua":
|
|
return server.luaHandlerFunc(commandName, args, params)
|
|
case "js":
|
|
return server.jsHandlerFunc(commandName, args, params)
|
|
}
|
|
}
|
|
}(engine, args),
|
|
}
|
|
|
|
// Add the commands to the list of commands.
|
|
server.commands = append(server.commands, command)
|
|
|
|
return nil
|
|
}
|
|
|
|
// LoadModule loads an external module into SugarDB ar runtime.
|
|
//
|
|
// Parameters:
|
|
//
|
|
// `path` - string - The full path to the .so plugin to be loaded.
|
|
//
|
|
// `args` - ...string - A list of args that will be passed unmodified to the plugins command's
|
|
// KeyExtractionFunc and HandlerFunc
|
|
func (server *SugarDB) LoadModule(path string, args ...string) error {
|
|
server.commandsRWMut.Lock()
|
|
defer server.commandsRWMut.Unlock()
|
|
|
|
for _, suffix := range []string{".lua", ".js"} {
|
|
if strings.HasSuffix(path, suffix) {
|
|
return server.AddScriptCommand(path, args)
|
|
}
|
|
}
|
|
|
|
if _, err := os.Stat(path); err != nil {
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("load module: module %s not found", path)
|
|
}
|
|
return fmt.Errorf("load module: %v", err)
|
|
}
|
|
|
|
p, err := plugin.Open(path)
|
|
if err != nil {
|
|
return fmt.Errorf("plugin open: %v", err)
|
|
}
|
|
|
|
commandSymbol, err := p.Lookup("Command")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
command, ok := commandSymbol.(*string)
|
|
if !ok {
|
|
return errors.New("command symbol is not a string")
|
|
}
|
|
|
|
categoriesSymbol, err := p.Lookup("Categories")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
categories, ok := categoriesSymbol.(*[]string)
|
|
if !ok {
|
|
return errors.New("categories symbol not a string slice")
|
|
}
|
|
|
|
descriptionSymbol, err := p.Lookup("Description")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
description, ok := descriptionSymbol.(*string)
|
|
if !ok {
|
|
return errors.New("description symbol is no a string")
|
|
}
|
|
|
|
syncSymbol, err := p.Lookup("Sync")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
synchronize, ok := syncSymbol.(*bool)
|
|
if !ok {
|
|
return errors.New("sync symbol is not a bool")
|
|
}
|
|
|
|
keyExtractionFuncSymbol, err := p.Lookup("KeyExtractionFunc")
|
|
if err != nil {
|
|
return fmt.Errorf("key extraction func symbol: %v", err)
|
|
}
|
|
keyExtractionFunc, ok := keyExtractionFuncSymbol.(func(cmd []string, args ...string) ([]string, []string, error))
|
|
if !ok {
|
|
return errors.New("key extraction function has unexpected signature")
|
|
}
|
|
|
|
handlerFuncSymbol, err := p.Lookup("HandlerFunc")
|
|
if err != nil {
|
|
return fmt.Errorf("handler func symbol: %v", err)
|
|
}
|
|
handlerFunc, ok := handlerFuncSymbol.(func(
|
|
ctx context.Context,
|
|
command []string,
|
|
keysExist func(ctx context.Context, key []string) map[string]bool,
|
|
getValues func(ctx context.Context, key []string) map[string]interface{},
|
|
setValues func(ctx context.Context, entries map[string]interface{}) error,
|
|
args ...string,
|
|
) ([]byte, error))
|
|
if !ok {
|
|
return errors.New("handler function has unexpected signature")
|
|
}
|
|
|
|
// Remove the currently loaded version of this module and replace it with the new one
|
|
server.commands = slices.DeleteFunc(server.commands, func(command internal.Command) bool {
|
|
return strings.EqualFold(command.Module, path)
|
|
})
|
|
|
|
// Add the new command
|
|
server.commands = append(server.commands, internal.Command{
|
|
Command: *command,
|
|
Module: path,
|
|
Categories: func() []string {
|
|
// Convert all the categories to lower case for uniformity
|
|
cats := make([]string, len(*categories))
|
|
for i, cat := range *categories {
|
|
cats[i] = strings.ToLower(cat)
|
|
}
|
|
return cats
|
|
}(),
|
|
Description: *description,
|
|
Sync: *synchronize,
|
|
SubCommands: make([]internal.SubCommand, 0),
|
|
KeyExtractionFunc: func(cmd []string) (internal.KeyExtractionFuncResult, error) {
|
|
readKeys, writeKeys, err := keyExtractionFunc(cmd, args...)
|
|
if err != nil {
|
|
return internal.KeyExtractionFuncResult{}, err
|
|
}
|
|
return internal.KeyExtractionFuncResult{
|
|
Channels: make([]string, 0),
|
|
ReadKeys: readKeys,
|
|
WriteKeys: writeKeys,
|
|
}, nil
|
|
},
|
|
HandlerFunc: func(params internal.HandlerFuncParams) ([]byte, error) {
|
|
return handlerFunc(
|
|
params.Context,
|
|
params.Command,
|
|
params.KeysExist,
|
|
params.GetValues,
|
|
params.SetValues,
|
|
args...,
|
|
)
|
|
},
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
// UnloadModule unloads the provided module
|
|
//
|
|
// Parameters:
|
|
//
|
|
// `module` - string - module name as displayed by the ListModules method.
|
|
func (server *SugarDB) UnloadModule(module string) {
|
|
server.commandsRWMut.Lock()
|
|
defer server.commandsRWMut.Unlock()
|
|
server.commands = slices.DeleteFunc(server.commands, func(command internal.Command) bool {
|
|
return strings.EqualFold(command.Module, module)
|
|
})
|
|
}
|
|
|
|
// ListModules lists the currently loaded modules
|
|
//
|
|
// Returns: a string slice representing all the currently loaded modules.
|
|
func (server *SugarDB) ListModules() []string {
|
|
server.commandsRWMut.RLock()
|
|
defer server.commandsRWMut.RUnlock()
|
|
var modules []string
|
|
for _, command := range server.commands {
|
|
if !slices.ContainsFunc(modules, func(module string) bool {
|
|
return strings.EqualFold(module, command.Module)
|
|
}) {
|
|
modules = append(modules, strings.ToLower(command.Module))
|
|
}
|
|
}
|
|
return modules
|
|
}
|