mirror of https://github.com/k3s-io/k3s
665 lines
18 KiB
Go
665 lines
18 KiB
Go
// Copyright 2017 Canonical Ltd.
|
|
//
|
|
// 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 driver
|
|
|
|
import (
|
|
"context"
|
|
"database/sql/driver"
|
|
"io"
|
|
"net"
|
|
"reflect"
|
|
"time"
|
|
|
|
"github.com/Rican7/retry/backoff"
|
|
"github.com/Rican7/retry/strategy"
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/canonical/go-dqlite/client"
|
|
"github.com/canonical/go-dqlite/internal/bindings"
|
|
"github.com/canonical/go-dqlite/internal/protocol"
|
|
)
|
|
|
|
// Driver perform queries against a dqlite server.
|
|
type Driver struct {
|
|
log client.LogFunc // Log function to use
|
|
store client.NodeStore // Holds addresses of dqlite servers
|
|
context context.Context // Global cancellation context
|
|
connectionTimeout time.Duration // Max time to wait for a new connection
|
|
contextTimeout time.Duration // Default client context timeout.
|
|
clientConfig protocol.Config // Configuration for dqlite client instances
|
|
}
|
|
|
|
// Error is returned in case of database errors.
|
|
type Error = bindings.Error
|
|
|
|
// Error codes. Values here mostly overlap with native SQLite codes.
|
|
const (
|
|
ErrBusy = 5
|
|
errIoErr = 10
|
|
errIoErrNotLeader = errIoErr | 32<<8
|
|
errIoErrLeadershipLost = errIoErr | (33 << 8)
|
|
)
|
|
|
|
// Option can be used to tweak driver parameters.
|
|
type Option func(*options)
|
|
|
|
// NodeStore is a convenience alias of client.NodeStore.
|
|
type NodeStore = client.NodeStore
|
|
|
|
// NodeInfo is a convenience alias of client.NodeInfo.
|
|
type NodeInfo = client.NodeInfo
|
|
|
|
// DefaultNodeStore is a convenience alias of client.DefaultNodeStore.
|
|
var DefaultNodeStore = client.DefaultNodeStore
|
|
|
|
// WithLogFunc sets a custom logging function.
|
|
func WithLogFunc(log client.LogFunc) Option {
|
|
return func(options *options) {
|
|
options.Log = log
|
|
}
|
|
}
|
|
|
|
// DialFunc is a function that can be used to establish a network connection
|
|
// with a dqlite node.
|
|
type DialFunc = protocol.DialFunc
|
|
|
|
// WithDialFunc sets a custom dial function.
|
|
func WithDialFunc(dial DialFunc) Option {
|
|
return func(options *options) {
|
|
options.Dial = protocol.DialFunc(dial)
|
|
}
|
|
}
|
|
|
|
// WithConnectionTimeout sets the connection timeout.
|
|
//
|
|
// If not used, the default is 5 seconds.
|
|
func WithConnectionTimeout(timeout time.Duration) Option {
|
|
return func(options *options) {
|
|
options.ConnectionTimeout = timeout
|
|
}
|
|
}
|
|
|
|
// WithConnectionBackoffFactor sets the exponential backoff factor for retrying
|
|
// failed connection attempts.
|
|
//
|
|
// If not used, the default is 50 milliseconds.
|
|
func WithConnectionBackoffFactor(factor time.Duration) Option {
|
|
return func(options *options) {
|
|
options.ConnectionBackoffFactor = factor
|
|
}
|
|
}
|
|
|
|
// WithConnectionBackoffCap sets the maximum connection retry backoff value,
|
|
// (regardless of the backoff factor) for retrying failed connection attempts.
|
|
//
|
|
// If not used, the default is 1 second.
|
|
func WithConnectionBackoffCap(cap time.Duration) Option {
|
|
return func(options *options) {
|
|
options.ConnectionBackoffCap = cap
|
|
}
|
|
}
|
|
|
|
// WithContext sets a global cancellation context.
|
|
func WithContext(context context.Context) Option {
|
|
return func(options *options) {
|
|
options.Context = context
|
|
}
|
|
}
|
|
|
|
// WithContextTimeout sets the default client context timeout when no context
|
|
// deadline is provided.
|
|
//
|
|
// If not used, the default is 5 seconds.
|
|
func WithContextTimeout(timeout time.Duration) Option {
|
|
return func(options *options) {
|
|
options.ContextTimeout = timeout
|
|
}
|
|
}
|
|
|
|
// NewDriver creates a new dqlite driver, which also implements the
|
|
// driver.Driver interface.
|
|
func New(store client.NodeStore, options ...Option) (*Driver, error) {
|
|
o := defaultOptions()
|
|
|
|
for _, option := range options {
|
|
option(o)
|
|
}
|
|
|
|
driver := &Driver{
|
|
log: o.Log,
|
|
store: store,
|
|
context: o.Context,
|
|
connectionTimeout: o.ConnectionTimeout,
|
|
contextTimeout: o.ContextTimeout,
|
|
}
|
|
|
|
driver.clientConfig.Dial = o.Dial
|
|
driver.clientConfig.AttemptTimeout = 5 * time.Second
|
|
driver.clientConfig.RetryStrategies = []strategy.Strategy{
|
|
driverConnectionRetryStrategy(
|
|
o.ConnectionBackoffFactor,
|
|
o.ConnectionBackoffCap,
|
|
),
|
|
}
|
|
|
|
return driver, nil
|
|
}
|
|
|
|
// Hold configuration options for a dqlite driver.
|
|
type options struct {
|
|
Log client.LogFunc
|
|
Dial protocol.DialFunc
|
|
ConnectionTimeout time.Duration
|
|
ContextTimeout time.Duration
|
|
ConnectionBackoffFactor time.Duration
|
|
ConnectionBackoffCap time.Duration
|
|
Context context.Context
|
|
}
|
|
|
|
// Create a options object with sane defaults.
|
|
func defaultOptions() *options {
|
|
return &options{
|
|
Log: client.DefaultLogFunc,
|
|
Dial: client.DefaultDialFunc,
|
|
ConnectionTimeout: 15 * time.Second,
|
|
ContextTimeout: 2 * time.Second,
|
|
ConnectionBackoffFactor: 50 * time.Millisecond,
|
|
ConnectionBackoffCap: time.Second,
|
|
Context: context.Background(),
|
|
}
|
|
}
|
|
|
|
// Return a retry strategy with jittered exponential backoff, capped at the
|
|
// given amount of time.
|
|
func driverConnectionRetryStrategy(factor, cap time.Duration) strategy.Strategy {
|
|
backoff := backoff.BinaryExponential(factor)
|
|
|
|
return func(attempt uint) bool {
|
|
if attempt > 0 {
|
|
duration := backoff(attempt)
|
|
if duration > cap {
|
|
duration = cap
|
|
}
|
|
time.Sleep(duration)
|
|
}
|
|
|
|
return true
|
|
}
|
|
}
|
|
|
|
// Open establishes a new connection to a SQLite database on the dqlite server.
|
|
//
|
|
// The given name must be a pure file name without any directory segment,
|
|
// dqlite will connect to a database with that name in its data directory.
|
|
//
|
|
// Query parameters are always valid except for "mode=memory".
|
|
//
|
|
// If this node is not the leader, or the leader is unknown an ErrNotLeader
|
|
// error is returned.
|
|
func (d *Driver) Open(uri string) (driver.Conn, error) {
|
|
ctx, cancel := context.WithTimeout(d.context, d.connectionTimeout)
|
|
defer cancel()
|
|
|
|
// TODO: generate a client ID.
|
|
connector := protocol.NewConnector(0, d.store, d.clientConfig, d.log)
|
|
|
|
conn := &Conn{
|
|
log: d.log,
|
|
contextTimeout: d.contextTimeout,
|
|
}
|
|
|
|
var err error
|
|
conn.protocol, err = connector.Connect(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to create dqlite connection")
|
|
}
|
|
conn.protocol.SetContextTimeout(d.contextTimeout)
|
|
|
|
conn.request.Init(4096)
|
|
conn.response.Init(4096)
|
|
|
|
defer conn.request.Reset()
|
|
defer conn.response.Reset()
|
|
|
|
protocol.EncodeOpen(&conn.request, uri, 0, "volatile")
|
|
|
|
if err := conn.protocol.Call(ctx, &conn.request, &conn.response); err != nil {
|
|
conn.protocol.Close()
|
|
return nil, errors.Wrap(err, "failed to open database")
|
|
}
|
|
|
|
conn.id, err = protocol.DecodeDb(&conn.response)
|
|
if err != nil {
|
|
conn.protocol.Close()
|
|
return nil, errors.Wrap(err, "failed to open database")
|
|
}
|
|
|
|
return conn, nil
|
|
}
|
|
|
|
// SetContextTimeout sets the default client timeout when no context deadline
|
|
// is provided.
|
|
func (d *Driver) SetContextTimeout(timeout time.Duration) {
|
|
d.contextTimeout = timeout
|
|
}
|
|
|
|
// ErrNoAvailableLeader is returned as root cause of Open() if there's no
|
|
// leader available in the cluster.
|
|
var ErrNoAvailableLeader = protocol.ErrNoAvailableLeader
|
|
|
|
// Conn implements the sql.Conn interface.
|
|
type Conn struct {
|
|
log client.LogFunc
|
|
protocol *protocol.Protocol
|
|
request protocol.Message
|
|
response protocol.Message
|
|
id uint32 // Database ID.
|
|
contextTimeout time.Duration
|
|
}
|
|
|
|
// PrepareContext returns a prepared statement, bound to this connection.
|
|
// context is for the preparation of the statement, it must not store the
|
|
// context within the statement itself.
|
|
func (c *Conn) PrepareContext(ctx context.Context, query string) (driver.Stmt, error) {
|
|
defer c.request.Reset()
|
|
defer c.response.Reset()
|
|
|
|
stmt := &Stmt{
|
|
protocol: c.protocol,
|
|
request: &c.request,
|
|
response: &c.response,
|
|
}
|
|
|
|
protocol.EncodePrepare(&c.request, uint64(c.id), query)
|
|
|
|
if err := c.protocol.Call(ctx, &c.request, &c.response); err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
var err error
|
|
stmt.db, stmt.id, stmt.params, err = protocol.DecodeStmt(&c.response)
|
|
if err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
return stmt, nil
|
|
}
|
|
|
|
// Prepare returns a prepared statement, bound to this connection.
|
|
func (c *Conn) Prepare(query string) (driver.Stmt, error) {
|
|
return c.PrepareContext(context.Background(), query)
|
|
}
|
|
|
|
// ExecContext is an optional interface that may be implemented by a Conn.
|
|
func (c *Conn) ExecContext(ctx context.Context, query string, args []driver.NamedValue) (driver.Result, error) {
|
|
defer c.request.Reset()
|
|
defer c.response.Reset()
|
|
|
|
protocol.EncodeExecSQL(&c.request, uint64(c.id), query, args)
|
|
|
|
if err := c.protocol.Call(ctx, &c.request, &c.response); err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
result, err := protocol.DecodeResult(&c.response)
|
|
if err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
return &Result{result: result}, nil
|
|
}
|
|
|
|
// Query is an optional interface that may be implemented by a Conn.
|
|
func (c *Conn) Query(query string, args []driver.Value) (driver.Rows, error) {
|
|
return c.QueryContext(context.Background(), query, valuesToNamedValues(args))
|
|
}
|
|
|
|
// QueryContext is an optional interface that may be implemented by a Conn.
|
|
func (c *Conn) QueryContext(ctx context.Context, query string, args []driver.NamedValue) (driver.Rows, error) {
|
|
defer c.request.Reset()
|
|
|
|
protocol.EncodeQuerySQL(&c.request, uint64(c.id), query, args)
|
|
|
|
if err := c.protocol.Call(ctx, &c.request, &c.response); err != nil {
|
|
c.response.Reset()
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
rows, err := protocol.DecodeRows(&c.response)
|
|
if err != nil {
|
|
c.response.Reset()
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
return &Rows{ctx: ctx, request: &c.request, response: &c.response, protocol: c.protocol, rows: rows}, nil
|
|
}
|
|
|
|
// Exec is an optional interface that may be implemented by a Conn.
|
|
func (c *Conn) Exec(query string, args []driver.Value) (driver.Result, error) {
|
|
return c.ExecContext(context.Background(), query, valuesToNamedValues(args))
|
|
}
|
|
|
|
// Close invalidates and potentially stops any current prepared statements and
|
|
// transactions, marking this connection as no longer in use.
|
|
//
|
|
// Because the sql package maintains a free pool of connections and only calls
|
|
// Close when there's a surplus of idle connections, it shouldn't be necessary
|
|
// for drivers to do their own connection caching.
|
|
func (c *Conn) Close() error {
|
|
return c.protocol.Close()
|
|
}
|
|
|
|
// BeginTx starts and returns a new transaction. If the context is canceled by
|
|
// the user the sql package will call Tx.Rollback before discarding and closing
|
|
// the connection.
|
|
//
|
|
// This must check opts.Isolation to determine if there is a set isolation
|
|
// level. If the driver does not support a non-default level and one is set or
|
|
// if there is a non-default isolation level that is not supported, an error
|
|
// must be returned.
|
|
//
|
|
// This must also check opts.ReadOnly to determine if the read-only value is
|
|
// true to either set the read-only transaction property if supported or return
|
|
// an error if it is not supported.
|
|
func (c *Conn) BeginTx(ctx context.Context, opts driver.TxOptions) (driver.Tx, error) {
|
|
if _, err := c.ExecContext(ctx, "BEGIN", nil); err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
tx := &Tx{
|
|
conn: c,
|
|
}
|
|
|
|
return tx, nil
|
|
}
|
|
|
|
// Begin starts and returns a new transaction.
|
|
//
|
|
// Deprecated: Drivers should implement ConnBeginTx instead (or additionally).
|
|
func (c *Conn) Begin() (driver.Tx, error) {
|
|
return c.BeginTx(context.Background(), driver.TxOptions{})
|
|
}
|
|
|
|
// Tx is a transaction.
|
|
type Tx struct {
|
|
conn *Conn
|
|
}
|
|
|
|
// Commit the transaction.
|
|
func (tx *Tx) Commit() error {
|
|
ctx, cancel := context.WithTimeout(context.Background(), tx.conn.contextTimeout)
|
|
defer cancel()
|
|
|
|
if _, err := tx.conn.ExecContext(ctx, "COMMIT", nil); err != nil {
|
|
return driverError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Rollback the transaction.
|
|
func (tx *Tx) Rollback() error {
|
|
ctx, cancel := context.WithTimeout(context.Background(), tx.conn.contextTimeout)
|
|
defer cancel()
|
|
|
|
if _, err := tx.conn.ExecContext(ctx, "ROLLBACK", nil); err != nil {
|
|
return driverError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Stmt is a prepared statement. It is bound to a Conn and not
|
|
// used by multiple goroutines concurrently.
|
|
type Stmt struct {
|
|
protocol *protocol.Protocol
|
|
request *protocol.Message
|
|
response *protocol.Message
|
|
db uint32
|
|
id uint32
|
|
params uint64
|
|
}
|
|
|
|
// Close closes the statement.
|
|
func (s *Stmt) Close() error {
|
|
defer s.request.Reset()
|
|
defer s.response.Reset()
|
|
|
|
protocol.EncodeFinalize(s.request, s.db, s.id)
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := s.protocol.Call(ctx, s.request, s.response); err != nil {
|
|
return driverError(err)
|
|
}
|
|
|
|
if err := protocol.DecodeEmpty(s.response); err != nil {
|
|
return driverError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// NumInput returns the number of placeholder parameters.
|
|
func (s *Stmt) NumInput() int {
|
|
return int(s.params)
|
|
}
|
|
|
|
// ExecContext executes a query that doesn't return rows, such
|
|
// as an INSERT or UPDATE.
|
|
//
|
|
// ExecContext must honor the context timeout and return when it is canceled.
|
|
func (s *Stmt) ExecContext(ctx context.Context, args []driver.NamedValue) (driver.Result, error) {
|
|
defer s.request.Reset()
|
|
defer s.response.Reset()
|
|
|
|
protocol.EncodeExec(s.request, s.db, s.id, args)
|
|
|
|
if err := s.protocol.Call(ctx, s.request, s.response); err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
result, err := protocol.DecodeResult(s.response)
|
|
if err != nil {
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
return &Result{result: result}, nil
|
|
}
|
|
|
|
// Exec executes a query that doesn't return rows, such
|
|
func (s *Stmt) Exec(args []driver.Value) (driver.Result, error) {
|
|
return s.ExecContext(context.Background(), valuesToNamedValues(args))
|
|
}
|
|
|
|
// QueryContext executes a query that may return rows, such as a
|
|
// SELECT.
|
|
//
|
|
// QueryContext must honor the context timeout and return when it is canceled.
|
|
func (s *Stmt) QueryContext(ctx context.Context, args []driver.NamedValue) (driver.Rows, error) {
|
|
defer s.request.Reset()
|
|
|
|
// FIXME: this shouldn't be needed but we have hit a few panics
|
|
// probably due to the response object not being fully reset.
|
|
s.response.Reset()
|
|
|
|
protocol.EncodeQuery(s.request, s.db, s.id, args)
|
|
|
|
if err := s.protocol.Call(ctx, s.request, s.response); err != nil {
|
|
s.response.Reset()
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
rows, err := protocol.DecodeRows(s.response)
|
|
if err != nil {
|
|
s.response.Reset()
|
|
return nil, driverError(err)
|
|
}
|
|
|
|
return &Rows{ctx: ctx, request: s.request, response: s.response, protocol: s.protocol, rows: rows}, nil
|
|
}
|
|
|
|
// Query executes a query that may return rows, such as a
|
|
func (s *Stmt) Query(args []driver.Value) (driver.Rows, error) {
|
|
return s.QueryContext(context.Background(), valuesToNamedValues(args))
|
|
}
|
|
|
|
// Result is the result of a query execution.
|
|
type Result struct {
|
|
result protocol.Result
|
|
}
|
|
|
|
// LastInsertId returns the database's auto-generated ID
|
|
// after, for example, an INSERT into a table with primary
|
|
// key.
|
|
func (r *Result) LastInsertId() (int64, error) {
|
|
return int64(r.result.LastInsertID), nil
|
|
}
|
|
|
|
// RowsAffected returns the number of rows affected by the
|
|
// query.
|
|
func (r *Result) RowsAffected() (int64, error) {
|
|
return int64(r.result.RowsAffected), nil
|
|
}
|
|
|
|
// Rows is an iterator over an executed query's results.
|
|
type Rows struct {
|
|
ctx context.Context
|
|
protocol *protocol.Protocol
|
|
request *protocol.Message
|
|
response *protocol.Message
|
|
rows protocol.Rows
|
|
consumed bool
|
|
types []string
|
|
}
|
|
|
|
// Columns returns the names of the columns. The number of
|
|
// columns of the result is inferred from the length of the
|
|
// slice. If a particular column name isn't known, an empty
|
|
// string should be returned for that entry.
|
|
func (r *Rows) Columns() []string {
|
|
return r.rows.Columns
|
|
}
|
|
|
|
// Close closes the rows iterator.
|
|
func (r *Rows) Close() error {
|
|
err := r.rows.Close()
|
|
|
|
// If we consumed the whole result set, there's nothing to do as
|
|
// there's no pending response from the server.
|
|
if r.consumed {
|
|
return nil
|
|
}
|
|
|
|
// If there is was a single-response result set, we're done.
|
|
if err == io.EOF {
|
|
return nil
|
|
}
|
|
|
|
// Let's issue an interrupt request and wait until we get an empty
|
|
// response, signalling that the query was interrupted.
|
|
if err := r.protocol.Interrupt(r.ctx, r.request, r.response); err != nil {
|
|
return driverError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Next is called to populate the next row of data into
|
|
// the provided slice. The provided slice will be the same
|
|
// size as the Columns() are wide.
|
|
//
|
|
// Next should return io.EOF when there are no more rows.
|
|
func (r *Rows) Next(dest []driver.Value) error {
|
|
err := r.rows.Next(dest)
|
|
|
|
if err == protocol.ErrRowsPart {
|
|
r.rows.Close()
|
|
if err := r.protocol.More(r.ctx, r.response); err != nil {
|
|
return driverError(err)
|
|
}
|
|
rows, err := protocol.DecodeRows(r.response)
|
|
if err != nil {
|
|
return driverError(err)
|
|
}
|
|
r.rows = rows
|
|
return r.rows.Next(dest)
|
|
}
|
|
|
|
if err == io.EOF {
|
|
r.consumed = true
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
// ColumnTypeScanType implements RowsColumnTypeScanType.
|
|
func (r *Rows) ColumnTypeScanType(i int) reflect.Type {
|
|
// column := sql.NewColumn(r.rows, i)
|
|
|
|
// typ, err := r.protocol.ColumnTypeScanType(context.Background(), column)
|
|
// if err != nil {
|
|
// return nil
|
|
// }
|
|
|
|
// return typ.DriverType()
|
|
return nil
|
|
}
|
|
|
|
// ColumnTypeDatabaseTypeName implements RowsColumnTypeDatabaseTypeName.
|
|
// warning: not thread safe
|
|
func (r *Rows) ColumnTypeDatabaseTypeName(i int) string {
|
|
if r.types == nil {
|
|
var err error
|
|
r.types, err = r.rows.ColumnTypes()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
return r.types[i]
|
|
}
|
|
|
|
// Convert a driver.Value slice into a driver.NamedValue slice.
|
|
func valuesToNamedValues(args []driver.Value) []driver.NamedValue {
|
|
namedValues := make([]driver.NamedValue, len(args))
|
|
for i, value := range args {
|
|
namedValues[i] = driver.NamedValue{
|
|
Ordinal: i + 1,
|
|
Value: value,
|
|
}
|
|
}
|
|
return namedValues
|
|
}
|
|
|
|
func driverError(err error) error {
|
|
switch err := errors.Cause(err).(type) {
|
|
case *net.OpError:
|
|
return driver.ErrBadConn
|
|
case protocol.ErrRequest:
|
|
switch err.Code {
|
|
case errIoErrNotLeader:
|
|
fallthrough
|
|
case errIoErrLeadershipLost:
|
|
return driver.ErrBadConn
|
|
default:
|
|
return Error{
|
|
Code: int(err.Code),
|
|
Message: err.Description,
|
|
}
|
|
}
|
|
}
|
|
return err
|
|
}
|