You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
142 lines
3.0 KiB
Go
142 lines
3.0 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
|
|
"github.com/mkideal/cli"
|
|
"github.com/mkideal/pkg/textutil"
|
|
cl "github.com/rqlite/rqlite/cmd/rqlite/http"
|
|
)
|
|
|
|
// Rows represents query result
|
|
type Rows struct {
|
|
Columns []string `json:"columns"`
|
|
Types []string `json:"types"`
|
|
Values [][]interface{} `json:"values"`
|
|
Time float64 `json:"time"`
|
|
Error string `json:"error,omitempty"`
|
|
}
|
|
|
|
// RowCount implements textutil.Table interface
|
|
func (r *Rows) RowCount() int {
|
|
return len(r.Values) + 1
|
|
}
|
|
|
|
// ColCount implements textutil.Table interface
|
|
func (r *Rows) ColCount() int {
|
|
return len(r.Columns)
|
|
}
|
|
|
|
// Get implements textutil.Table interface
|
|
func (r *Rows) Get(i, j int) string {
|
|
if i == 0 {
|
|
if j >= len(r.Columns) {
|
|
return ""
|
|
}
|
|
return r.Columns[j]
|
|
}
|
|
|
|
if r.Values == nil {
|
|
return "NULL"
|
|
}
|
|
|
|
if i-1 >= len(r.Values) {
|
|
return "NULL"
|
|
}
|
|
if j >= len(r.Values[i-1]) {
|
|
return "NULL"
|
|
}
|
|
return fmt.Sprintf("%v", r.Values[i-1][j])
|
|
}
|
|
|
|
func (r *Rows) validate() error {
|
|
if r.Error != "" {
|
|
return fmt.Errorf(r.Error)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// headerRenderStyle render the header of result
|
|
type headerRenderStyle struct {
|
|
textutil.DefaultStyle
|
|
}
|
|
|
|
func (render headerRenderStyle) CellRender(row, col int, cell string, cw *textutil.ColorWriter) {
|
|
if row != 0 {
|
|
fmt.Fprint(cw, cell)
|
|
} else {
|
|
fmt.Fprint(cw, cw.Color.Cyan(cell))
|
|
}
|
|
}
|
|
|
|
var headerRender = &headerRenderStyle{}
|
|
|
|
type queryResponse struct {
|
|
Results []*Rows `json:"results"`
|
|
Error string `json:"error,omitempty"`
|
|
Time float64 `json:"time"`
|
|
}
|
|
|
|
func queryWithClient(ctx *cli.Context, client *cl.Client, timer bool, consistency, query string) error {
|
|
queryStr := url.Values{}
|
|
queryStr.Set("level", consistency)
|
|
queryStr.Set("q", query)
|
|
if timer {
|
|
queryStr.Set("timings", "")
|
|
}
|
|
u := url.URL{
|
|
Path: fmt.Sprintf("%sdb/query", client.Prefix),
|
|
RawQuery: queryStr.Encode(),
|
|
}
|
|
|
|
resp, err := client.Query(u)
|
|
|
|
var hcr error
|
|
if err != nil {
|
|
// If the error is HostChangedError, it should be propagated back to the caller to handle
|
|
// accordingly (change prompt display), but we should still assume that the request succeeded on some
|
|
// host and not treat it as an error.
|
|
err, ok := err.(*cl.HostChangedError)
|
|
if !ok {
|
|
return err
|
|
}
|
|
hcr = err
|
|
}
|
|
|
|
response, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
return fmt.Errorf("server responded with %s: %s", resp.Status, response)
|
|
}
|
|
|
|
// Parse response and write results
|
|
ret := &queryResponse{}
|
|
if err := parseResponse(&response, &ret); err != nil {
|
|
return err
|
|
}
|
|
if ret.Error != "" {
|
|
return fmt.Errorf(ret.Error)
|
|
}
|
|
if len(ret.Results) != 1 {
|
|
return fmt.Errorf("unexpected results length: %d", len(ret.Results))
|
|
}
|
|
|
|
result := ret.Results[0]
|
|
if err := result.validate(); err != nil {
|
|
return err
|
|
}
|
|
textutil.WriteTable(ctx, result, headerRender)
|
|
|
|
if timer {
|
|
fmt.Printf("Run Time: %f seconds\n", result.Time)
|
|
}
|
|
return hcr
|
|
}
|