mirror of https://github.com/k3s-io/k3s
245 lines
5.9 KiB
Go
245 lines
5.9 KiB
Go
package bindings
|
|
|
|
/*
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <assert.h>
|
|
#include <stdint.h>
|
|
#include <signal.h>
|
|
|
|
#include <dqlite.h>
|
|
#include <raft.h>
|
|
#include <sqlite3.h>
|
|
|
|
#define EMIT_BUF_LEN 1024
|
|
|
|
typedef unsigned long long nanoseconds_t;
|
|
|
|
// Duplicate a file descriptor and prevent it from being cloned into child processes.
|
|
static int dupCloexec(int oldfd) {
|
|
int newfd = -1;
|
|
|
|
newfd = dup(oldfd);
|
|
if (newfd < 0) {
|
|
return -1;
|
|
}
|
|
|
|
if (fcntl(newfd, F_SETFD, FD_CLOEXEC) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
return newfd;
|
|
}
|
|
|
|
// C to Go trampoline for custom connect function.
|
|
int connectWithDial(uintptr_t handle, char *address, int *fd);
|
|
|
|
// Wrapper to call the Go trampoline.
|
|
static int connectTrampoline(void *data, const char *address, int *fd) {
|
|
uintptr_t handle = (uintptr_t)(data);
|
|
return connectWithDial(handle, (char*)address, fd);
|
|
}
|
|
|
|
// Configure a custom connect function.
|
|
static int configConnectFunc(dqlite_node *t, uintptr_t handle) {
|
|
return dqlite_node_set_connect_func(t, connectTrampoline, (void*)handle);
|
|
}
|
|
|
|
static int initializeSQLite()
|
|
{
|
|
int rc;
|
|
|
|
// Configure SQLite for single-thread mode. This is a global config.
|
|
rc = sqlite3_config(SQLITE_CONFIG_SINGLETHREAD);
|
|
if (rc != SQLITE_OK) {
|
|
assert(rc == SQLITE_MISUSE);
|
|
return DQLITE_MISUSE;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static dqlite_node_info *makeInfos(int n) {
|
|
return calloc(n, sizeof(dqlite_node_info));
|
|
}
|
|
|
|
static void setInfo(dqlite_node_info *infos, unsigned i, unsigned id, const char *address) {
|
|
dqlite_node_info *info = &infos[i];
|
|
info->id = id;
|
|
info->address = address;
|
|
}
|
|
|
|
*/
|
|
import "C"
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"sync"
|
|
"time"
|
|
"unsafe"
|
|
|
|
"github.com/canonical/go-dqlite/internal/protocol"
|
|
)
|
|
|
|
type Node C.dqlite_node
|
|
|
|
// Init initializes dqlite global state.
|
|
func Init() error {
|
|
// FIXME: ignore SIGPIPE, see https://github.com/joyent/libuv/issues/1254
|
|
C.signal(C.SIGPIPE, C.SIG_IGN)
|
|
return nil
|
|
}
|
|
|
|
// NewNode creates a new Node instance.
|
|
func NewNode(id uint64, address string, dir string) (*Node, error) {
|
|
var server *C.dqlite_node
|
|
cid := C.unsigned(id)
|
|
|
|
caddress := C.CString(address)
|
|
defer C.free(unsafe.Pointer(caddress))
|
|
|
|
cdir := C.CString(dir)
|
|
defer C.free(unsafe.Pointer(cdir))
|
|
|
|
if rc := C.dqlite_node_create(cid, caddress, cdir, &server); rc != 0 {
|
|
return nil, fmt.Errorf("failed to create task object")
|
|
}
|
|
|
|
return (*Node)(unsafe.Pointer(server)), nil
|
|
}
|
|
|
|
func (s *Node) SetDialFunc(dial protocol.DialFunc) error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
connectLock.Lock()
|
|
defer connectLock.Unlock()
|
|
connectIndex++
|
|
connectRegistry[connectIndex] = dial
|
|
if rc := C.configConnectFunc(server, connectIndex); rc != 0 {
|
|
return fmt.Errorf("failed to set connect func")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Node) SetBindAddress(address string) error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
caddress := C.CString(address)
|
|
defer C.free(unsafe.Pointer(caddress))
|
|
if rc := C.dqlite_node_set_bind_address(server, caddress); rc != 0 {
|
|
return fmt.Errorf("failed to set bind address")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Node) SetNetworkLatency(nanoseconds uint64) error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
cnanoseconds := C.nanoseconds_t(nanoseconds)
|
|
if rc := C.dqlite_node_set_network_latency(server, cnanoseconds); rc != 0 {
|
|
return fmt.Errorf("failed to set network latency")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Node) GetBindAddress() string {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
return C.GoString(C.dqlite_node_get_bind_address(server))
|
|
}
|
|
|
|
func (s *Node) Start() error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
if rc := C.dqlite_node_start(server); rc != 0 {
|
|
return fmt.Errorf("failed to start task")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Node) Stop() error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
if rc := C.dqlite_node_stop(server); rc != 0 {
|
|
return fmt.Errorf("task stopped with error code %d", rc)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Close the server releasing all used resources.
|
|
func (s *Node) Close() {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
C.dqlite_node_destroy(server)
|
|
}
|
|
|
|
func (s *Node) Recover(cluster []protocol.NodeInfo) error {
|
|
server := (*C.dqlite_node)(unsafe.Pointer(s))
|
|
n := C.int(len(cluster))
|
|
infos := C.makeInfos(n)
|
|
defer C.free(unsafe.Pointer(infos))
|
|
for i, info := range cluster {
|
|
cid := C.unsigned(info.ID)
|
|
caddress := C.CString(info.Address)
|
|
defer C.free(unsafe.Pointer(caddress))
|
|
C.setInfo(infos, C.unsigned(i), cid, caddress)
|
|
}
|
|
if rc := C.dqlite_node_recover(server, infos, n); rc != 0 {
|
|
return fmt.Errorf("recover failed with error code %d", rc)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Extract the underlying socket from a connection.
|
|
func connToSocket(conn net.Conn) (C.int, error) {
|
|
file, err := conn.(fileConn).File()
|
|
if err != nil {
|
|
return C.int(-1), err
|
|
}
|
|
|
|
fd1 := C.int(file.Fd())
|
|
|
|
// Duplicate the file descriptor, in order to prevent Go's finalizer to
|
|
// close it.
|
|
fd2 := C.dupCloexec(fd1)
|
|
if fd2 < 0 {
|
|
return C.int(-1), fmt.Errorf("failed to dup socket fd")
|
|
}
|
|
|
|
conn.Close()
|
|
|
|
return fd2, nil
|
|
}
|
|
|
|
// Interface that net.Conn must implement in order to extract the underlying
|
|
// file descriptor.
|
|
type fileConn interface {
|
|
File() (*os.File, error)
|
|
}
|
|
|
|
//export connectWithDial
|
|
func connectWithDial(handle C.uintptr_t, address *C.char, fd *C.int) C.int {
|
|
connectLock.Lock()
|
|
defer connectLock.Unlock()
|
|
dial := connectRegistry[handle]
|
|
// TODO: make timeout customizable.
|
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
|
defer cancel()
|
|
conn, err := dial(ctx, C.GoString(address))
|
|
if err != nil {
|
|
return C.RAFT_NOCONNECTION
|
|
}
|
|
socket, err := connToSocket(conn)
|
|
if err != nil {
|
|
return C.RAFT_NOCONNECTION
|
|
}
|
|
*fd = socket
|
|
return C.int(0)
|
|
}
|
|
|
|
// Use handles to avoid passing Go pointers to C.
|
|
var connectRegistry = make(map[C.uintptr_t]protocol.DialFunc)
|
|
var connectIndex C.uintptr_t = 100
|
|
var connectLock = sync.Mutex{}
|
|
|
|
// ErrNodeStopped is returned by Node.Handle() is the server was stopped.
|
|
var ErrNodeStopped = fmt.Errorf("server was stopped")
|
|
|
|
// To compare bool values.
|
|
var cfalse C.bool
|