-
Notifications
You must be signed in to change notification settings - Fork 37
/
Copy pathgorqlite.go
138 lines (122 loc) · 3.59 KB
/
gorqlite.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
// Package gorqlite provieds a database/sql-like driver for rqlite,
// the distributed consistent sqlite.
//
// Copyright (c)2016 andrew fabbro (andrew@fabbro.org)
//
// See LICENSE.md for license. tl;dr: MIT. Conveniently, the same license as rqlite.
//
// Project home page: https://github.com/raindo308/gorqlite
//
// Learn more about rqlite at: https://github.com/rqlite/rqlite
package gorqlite
// this file contains package-level stuff:
// consts
// init()
// Open, TraceOn(), TraceOff()
import (
"crypto/rand"
"fmt"
"io"
"strings"
)
type apiOperation int
const (
api_QUERY apiOperation = iota
api_STATUS
api_WRITE
api_NODES
api_REQUEST
)
func init() {
traceOut = io.Discard
}
// Open creates and returns a "connection" to rqlite.
//
// Since rqlite is stateless, there is no actual connection.
// Open() creates and initializes a gorqlite Connection type,
// which represents various config information.
//
// The URL should be in a form like this:
//
// http://localhost:4001
//
// http:// default, no auth, localhost:4001
// https:// default, no auth, localhost:4001, using https
//
// http://localhost:1234
// http://mary:secret2@localhost:1234
//
// https://mary:secret2@somewhere.example.com:1234
// https://mary:secret2@somewhere.example.com // will use 4001
func Open(connURL string) (*Connection, error) {
var conn = &Connection{}
// generate our uuid for trace
b := make([]byte, 16)
_, err := rand.Read(b)
if err != nil {
return conn, err
}
conn.ID = fmt.Sprintf("%X-%X-%X-%X-%X", b[0:4], b[4:6], b[6:8], b[8:10], b[10:])
trace("%s: Open() called for url: %s", conn.ID, connURL)
// set defaults
conn.hasBeenClosed = false
// parse the URL given
err = conn.initConnection(connURL)
if err != nil {
return conn, err
}
if !conn.disableClusterDiscovery {
// call updateClusterInfo() to re-populate the cluster and discover peers
// also tests the user's default
if err := conn.updateClusterInfo(); err != nil {
return conn, err
}
}
return conn, nil
}
// trace adds a message to the trace output
//
// not a public function. we (inside) can add - outside they can
// only see.
//
// Call trace as: Sprintf pattern , args...
//
// This is done so that the more expensive Sprintf() stuff is
// done only if truly needed. When tracing is off, calls to
// trace() just hit a bool check and return. If tracing is on,
// then the Sprintf-ing is done at a leisurely pace because, well,
// we're tracing.
//
// Premature optimization is the root of all evil, so this is
// probably sinful behavior.
//
// Don't put a \n in your Sprintf pattern becuase trace() adds one
func trace(pattern string, args ...interface{}) {
// don't do the probably expensive Sprintf() if not needed
if !wantsTrace {
return
}
// this could all be made into one long statement but we have
// compilers to do such things for us. let's sip a mint julep
// and spell this out in glorious exposition.
// make sure there is one and only one newline
nlPattern := strings.TrimSpace(pattern) + "\n"
msg := fmt.Sprintf(nlPattern, args...)
traceOut.Write([]byte(msg))
}
// TraceOn turns on tracing output to the io.Writer of your choice.
//
// Trace output is very detailed and verbose, as you might expect.
//
// Normally, you should run with tracing off, as it makes absolutely
// no concession to performance and is intended for debugging/dev use.
func TraceOn(w io.Writer) {
traceOut = w
wantsTrace = true
}
// TraceOff turns off tracing output. Once you call TraceOff(), no further
// info is sent to the io.Writer, unless it is TraceOn'd again.
func TraceOff() {
wantsTrace = false
traceOut = io.Discard
}