mirror of https://github.com/k3s-io/k3s
243 lines
5.0 KiB
Go
243 lines
5.0 KiB
Go
// +build dqlite
|
|
|
|
package dqlite
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/canonical/go-dqlite"
|
|
"github.com/canonical/go-dqlite/client"
|
|
"github.com/canonical/go-dqlite/driver"
|
|
"github.com/k3s-io/kine/pkg/drivers/generic"
|
|
"github.com/k3s-io/kine/pkg/drivers/sqlite"
|
|
"github.com/k3s-io/kine/pkg/server"
|
|
"github.com/pkg/errors"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
Dialer = client.DefaultDialFunc
|
|
Logger = client.DefaultLogFunc
|
|
)
|
|
|
|
func init() {
|
|
// We assume SQLite will be used multi-threaded
|
|
if err := dqlite.ConfigMultiThread(); err != nil {
|
|
panic(errors.Wrap(err, "failed to set dqlite multithreaded mode"))
|
|
}
|
|
}
|
|
|
|
type opts struct {
|
|
peers []client.NodeInfo
|
|
peerFile string
|
|
dsn string
|
|
}
|
|
|
|
func AddPeers(ctx context.Context, nodeStore client.NodeStore, additionalPeers ...client.NodeInfo) error {
|
|
existing, err := nodeStore.Get(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var peers []client.NodeInfo
|
|
|
|
outer:
|
|
for _, peer := range additionalPeers {
|
|
for _, check := range existing {
|
|
if check.Address == peer.Address {
|
|
continue outer
|
|
}
|
|
}
|
|
peers = append(peers, peer)
|
|
}
|
|
|
|
if len(peers) > 0 {
|
|
err = nodeStore.Set(ctx, append(existing, peers...))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func New(ctx context.Context, datasourceName string, connPoolConfig generic.ConnectionPoolConfig) (server.Backend, error) {
|
|
opts, err := parseOpts(datasourceName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var nodeStore client.NodeStore
|
|
if opts.peerFile != "" {
|
|
nodeStore, err = client.DefaultNodeStore(opts.peerFile)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "opening peerfile")
|
|
}
|
|
} else {
|
|
nodeStore = client.NewInmemNodeStore()
|
|
}
|
|
|
|
if err := AddPeers(ctx, nodeStore, opts.peers...); err != nil {
|
|
return nil, errors.Wrap(err, "add peers")
|
|
}
|
|
|
|
d, err := driver.New(nodeStore,
|
|
driver.WithLogFunc(Logger),
|
|
driver.WithContext(ctx),
|
|
driver.WithDialFunc(Dialer))
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "new dqlite driver")
|
|
}
|
|
|
|
sql.Register("dqlite", d)
|
|
backend, generic, err := sqlite.NewVariant(ctx, "dqlite", opts.dsn, connPoolConfig)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "sqlite client")
|
|
}
|
|
if err := migrate(ctx, generic.DB); err != nil {
|
|
return nil, errors.Wrap(err, "failed to migrate DB from sqlite")
|
|
}
|
|
|
|
generic.LockWrites = true
|
|
generic.Retry = func(err error) bool {
|
|
if err, ok := err.(driver.Error); ok {
|
|
return err.Code == driver.ErrBusy
|
|
}
|
|
return false
|
|
}
|
|
generic.TranslateErr = func(err error) error {
|
|
if strings.Contains(err.Error(), "UNIQUE constraint") {
|
|
return server.ErrKeyExists
|
|
}
|
|
return err
|
|
}
|
|
|
|
return backend, nil
|
|
}
|
|
|
|
func migrate(ctx context.Context, newDB *sql.DB) (exitErr error) {
|
|
row := newDB.QueryRowContext(ctx, "SELECT COUNT(*) FROM kine")
|
|
var count int64
|
|
if err := row.Scan(&count); err != nil {
|
|
return err
|
|
}
|
|
if count > 0 {
|
|
return nil
|
|
}
|
|
|
|
if _, err := os.Stat("./db/state.db"); err != nil {
|
|
return nil
|
|
}
|
|
|
|
oldDB, err := sql.Open("sqlite3", "./db/state.db")
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
defer oldDB.Close()
|
|
|
|
oldData, err := oldDB.QueryContext(ctx, "SELECT id, name, created, deleted, create_revision, prev_revision, lease, value, old_value FROM kine")
|
|
if err != nil {
|
|
logrus.Errorf("failed to find old data to migrate: %v", err)
|
|
return nil
|
|
}
|
|
defer oldData.Close()
|
|
|
|
tx, err := newDB.BeginTx(ctx, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if exitErr == nil {
|
|
exitErr = tx.Commit()
|
|
} else {
|
|
tx.Rollback()
|
|
}
|
|
}()
|
|
|
|
for oldData.Next() {
|
|
row := []interface{}{
|
|
new(int),
|
|
new(string),
|
|
new(int),
|
|
new(int),
|
|
new(int),
|
|
new(int),
|
|
new(int),
|
|
new([]byte),
|
|
new([]byte),
|
|
}
|
|
if err := oldData.Scan(row...); err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := newDB.ExecContext(ctx, "INSERT INTO kine(id, name, created, deleted, create_revision, prev_revision, lease, value, old_value) values(?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
row...); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := oldData.Err(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func parseOpts(dsn string) (opts, error) {
|
|
result := opts{
|
|
dsn: dsn,
|
|
}
|
|
|
|
parts := strings.SplitN(dsn, "?", 2)
|
|
if len(parts) == 1 {
|
|
return result, nil
|
|
}
|
|
|
|
values, err := url.ParseQuery(parts[1])
|
|
if err != nil {
|
|
return result, err
|
|
}
|
|
|
|
for k, vs := range values {
|
|
if len(vs) == 0 {
|
|
continue
|
|
}
|
|
|
|
switch k {
|
|
case "peer":
|
|
for _, v := range vs {
|
|
parts := strings.SplitN(v, ":", 3)
|
|
if len(parts) != 3 {
|
|
return result, fmt.Errorf("must be ID:IP:PORT format got: %s", v)
|
|
}
|
|
id, err := strconv.ParseUint(parts[0], 10, 64)
|
|
if err != nil {
|
|
return result, errors.Wrapf(err, "failed to parse %s", parts[0])
|
|
}
|
|
result.peers = append(result.peers, client.NodeInfo{
|
|
ID: id,
|
|
Address: parts[1] + ":" + parts[2],
|
|
})
|
|
}
|
|
delete(values, k)
|
|
case "peer-file":
|
|
result.peerFile = vs[0]
|
|
delete(values, k)
|
|
}
|
|
}
|
|
|
|
if len(values) == 0 {
|
|
result.dsn = parts[0]
|
|
} else {
|
|
result.dsn = fmt.Sprintf("%s?%s", parts[0], values.Encode())
|
|
}
|
|
|
|
return result, nil
|
|
}
|