1447 lines
44 KiB
Go
1447 lines
44 KiB
Go
package httpd_test
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"os"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/influxdb/influxdb"
|
|
"github.com/influxdb/influxdb/httpd"
|
|
"github.com/influxdb/influxdb/influxql"
|
|
"github.com/influxdb/influxdb/messaging"
|
|
)
|
|
|
|
func init() {
|
|
influxdb.BcryptCost = 4
|
|
}
|
|
|
|
func TestHandler_Databases(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateDatabase("bar")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "SHOW DATABASES"}, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"rows":[{"columns":["name"],"values":[["bar"],["foo"]]}]}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DatabasesPrettyPrinted(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateDatabase("bar")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "SHOW DATABASES", "pretty": "true"}, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{
|
|
"results": [
|
|
{
|
|
"rows": [
|
|
{
|
|
"columns": [
|
|
"name"
|
|
],
|
|
"values": [
|
|
[
|
|
"bar"
|
|
],
|
|
[
|
|
"foo"
|
|
]
|
|
]
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDatabase(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "CREATE DATABASE foo"}, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDatabase_BadRequest_NoName(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "CREATE DATABASE"}, nil, "")
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDatabase_Conflict(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "CREATE DATABASE foo"}, nil, "")
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"database exists"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteDatabase(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "DROP DATABASE foo"}, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteDatabase_NotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "DROP DATABASE bar"}, nil, "")
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"database not found"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_RetentionPolicies(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "SHOW RETENTION POLICIES foo"}, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"rows":[{"columns":["name","duration","replicaN"],"values":[["bar","168h0m0s",1]]}]}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_RetentionPolicies_DatabaseNotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "SHOW RETENTION POLICIES foo"}, nil, "")
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"database not found"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateRetentionPolicy(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE RETENTION POLICY bar ON foo DURATION 1h REPLICATION 1"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateRetentionPolicyAsDefault(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE RETENTION POLICY bar ON foo DURATION 1h REPLICATION 1 DEFAULT"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
|
|
rp, err := srvr.DefaultRetentionPolicy("foo")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
} else if rp.Name != "bar" {
|
|
t.Fatalf("default retention policy mismatch:\n exp=%s\n got=%s\n", "bar", rp.Name)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateRetentionPolicy_DatabaseNotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE RETENTION POLICY bar ON foo DURATION 1h REPLICATION 1"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateRetentionPolicy_Conflict(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE RETENTION POLICY bar ON foo DURATION 1h REPLICATION 1"}
|
|
MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateRetentionPolicy_BadRequest(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE RETENTION POLICY bar ON foo DURATION ***BAD*** REPLICATION 1"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateRetentionPolicy(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "ALTER RETENTION POLICY bar ON foo REPLICATION 42 DURATION 1m"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
// Verify updated policy.
|
|
p, _ := srvr.RetentionPolicy("foo", "bar")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != "" {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
} else if p.ReplicaN != 42 {
|
|
t.Fatalf("unexpected replication factor: %d", p.ReplicaN)
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateRetentionPolicy_BadRequest(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "ALTER RETENTION POLICY bar ON foo DURATION ***BAD*** REPLICATION 1"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
// Verify response.
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateRetentionPolicy_DatabaseNotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "ALTER RETENTION POLICY bar ON foo DURATION 1h REPLICATION 1"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
// Verify response.
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateRetentionPolicy_NotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "ALTER RETENTION POLICY qux ON foo DURATION 1h REPLICATION 1"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
// Verify response.
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteRetentionPolicy(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "DROP RETENTION POLICY bar ON foo"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteRetentionPolicy_DatabaseNotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "DROP RETENTION POLICY bar ON qux"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"database not found"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteRetentionPolicy_NotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "DROP RETENTION POLICY bar ON foo"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"retention policy not found"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_Ping(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, _ := MustHTTP("GET", s.URL+`/ping`, nil, nil, "")
|
|
|
|
if status != http.StatusNoContent {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_Users_NoUsers(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "SHOW USERS"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"rows":[{"columns":["user","admin"]}]}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_Users_OneUser(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("jdoe", "1337", true)
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "SHOW USERS"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"rows":[{"columns":["user","admin"],"values":[["jdoe",true]]}]}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_Users_MultipleUsers(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("jdoe", "1337", false)
|
|
srvr.CreateUser("mclark", "1337", true)
|
|
srvr.CreateUser("csmith", "1337", false)
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "SHOW USERS"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"rows":[{"columns":["user","admin"],"values":[["csmith",false],["jdoe",false],["mclark",true]]}]}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateUser(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": `CREATE USER testuser WITH PASSWORD '1337'`}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateUser_BadRequest(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE USER 0xBAD WITH PASSWORD pwd1337"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"error":"error parsing query: found 0, expected identifier at line 1, char 13"}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateUser_BadRequest_NoName(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE USER WITH PASSWORD pwd1337"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"error":"error parsing query: found WITH, expected identifier at line 1, char 13"}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateUser_BadRequest_NoPassword(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "CREATE USER jdoe"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"error":"error parsing query: found EOF, expected WITH at line 1, char 18"}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateUser(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("jdoe", "1337", false)
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
// Save original password hash.
|
|
hash := srvr.User("jdoe").Hash
|
|
|
|
// Update user password.
|
|
status, body := MustHTTP("PUT", s.URL+`/users/jdoe`, nil, nil, `{"password": "7331"}`)
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
} else if srvr.User("jdoe").Hash == hash {
|
|
t.Fatalf("expected password hash to change")
|
|
}
|
|
}
|
|
|
|
func TestHandler_UpdateUser_PasswordBadRequest(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("jdoe", "1337", false)
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("PUT", s.URL+`/users/jdoe`, nil, nil, `{"password": 10}`)
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `json: cannot unmarshal number into Go value of type string` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteUser(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("jdoe", "1337", false)
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "DROP USER jdoe"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteUser_UserNotFound(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "DROP USER jdoe"}
|
|
status, body := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"results":[{"error":"user not found"}]}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DataNodes(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenUninitializedServer(NewMessagingClient())
|
|
srvr.CreateDataNode(MustParseURL("http://localhost:1000"))
|
|
srvr.CreateDataNode(MustParseURL("http://localhost:2000"))
|
|
srvr.CreateDataNode(MustParseURL("http://localhost:3000"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("GET", s.URL+`/data_nodes`, nil, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `[{"id":1,"url":"http://localhost:1000"},{"id":2,"url":"http://localhost:2000"},{"id":3,"url":"http://localhost:3000"}]` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDataNode(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenUninitializedServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/data_nodes`, nil, nil, `{"url":"http://localhost:1000"}`)
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `{"id":1,"url":"http://localhost:1000"}` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDataNode_BadRequest(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/data_nodes`, nil, nil, `{"name":`)
|
|
if status != http.StatusBadRequest {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `unexpected EOF` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_CreateDataNode_InternalServerError(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/data_nodes`, nil, nil, `{"url":""}`)
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: %d, %s", status, body)
|
|
} else if body != `data node url required` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteDataNode(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDataNode(MustParseURL("http://localhost:1000"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("DELETE", s.URL+`/data_nodes/1`, nil, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_DeleteUser_DataNodeNotFound(t *testing.T) {
|
|
t.Skip()
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("DELETE", s.URL+`/data_nodes/10000`, nil, nil, "")
|
|
if status != http.StatusNotFound {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
} else if body != `data node not found` {
|
|
t.Fatalf("unexpected body: %s", body)
|
|
}
|
|
}
|
|
|
|
// Perform a subset of endpoint testing, with authentication enabled.
|
|
|
|
func TestHandler_AuthenticatedCreateAdminUser(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
// Attempting to create a non-admin user should fail.
|
|
query := map[string]string{"q": "CREATE USER maeve WITH PASSWORD 'pass'"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusUnauthorized {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
|
|
// Creating the first admin user, without supplying authentication
|
|
// credentials should be OK.
|
|
query = map[string]string{"q": "CREATE USER orla WITH PASSWORD 'pass' WITH ALL PRIVILEGES"}
|
|
status, _ = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
|
|
// Creating a second admin user, without supplying authentication
|
|
// credentials should fail.
|
|
query = map[string]string{"q": "CREATE USER louise WITH PASSWORD 'pass' WITH ALL PRIVILEGES"}
|
|
status, _ = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusUnauthorized {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
|
|
}
|
|
|
|
func TestHandler_AuthenticatedDatabases_Unauthorized(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, map[string]string{"q": "SHOW DATABASES"}, nil, "")
|
|
if status != http.StatusUnauthorized {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_AuthenticatedDatabases_AuthorizedQueryParams(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("lisa", "password", true)
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "SHOW DATABASES", "u": "lisa", "p": "password"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_AuthenticatedDatabases_UnauthorizedQueryParams(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("lisa", "password", true)
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
query := map[string]string{"q": "SHOW DATABASES", "u": "lisa", "p": "wrong"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
if status != http.StatusUnauthorized {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_AuthenticatedDatabases_AuthorizedBasicAuth(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("lisa", "password", true)
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
auth := make(map[string]string)
|
|
auth["Authorization"] = "Basic " + base64.StdEncoding.EncodeToString([]byte("lisa:password"))
|
|
query := map[string]string{"q": "SHOW DATABASES"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, auth, "")
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_AuthenticatedDatabases_UnauthorizedBasicAuth(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateUser("lisa", "password", true)
|
|
s := NewAuthenticatedHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
auth := make(map[string]string)
|
|
auth["Authorization"] = "Basic " + base64.StdEncoding.EncodeToString([]byte("lisa:wrong"))
|
|
query := map[string]string{"q": "SHOW DATABASES"}
|
|
status, _ := MustHTTP("GET", s.URL+`/query`, query, auth, "")
|
|
if status != http.StatusUnauthorized {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveWriteSeries(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, _ := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}]}`)
|
|
|
|
if status != http.StatusOK {
|
|
t.Fatalf("unexpected status: %d", status)
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveWriteSeries_noDatabaseExists(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}]}`)
|
|
|
|
expectedStatus := http.StatusNotFound
|
|
if status != expectedStatus {
|
|
t.Fatalf("unexpected status: expected: %d, actual: %d", expectedStatus, status)
|
|
}
|
|
|
|
response := `{"error":"database not found: \"foo\""}`
|
|
if body != response {
|
|
t.Fatalf("unexpected body: expected %s, actual %s", response, body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveWriteSeries_invalidJSON(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : foo", "retentionPolicy" : "bar", "points": [{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}]}`)
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: expected: %d, actual: %d", http.StatusInternalServerError, status)
|
|
}
|
|
|
|
response := `{"error":"invalid character 'o' in literal false (expecting 'a')"}`
|
|
if body != response {
|
|
t.Fatalf("unexpected body: expected %s, actual %s", response, body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveWriteSeries_noDatabaseSpecified(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{}`)
|
|
|
|
if status != http.StatusInternalServerError {
|
|
t.Fatalf("unexpected status: expected: %d, actual: %d", http.StatusInternalServerError, status)
|
|
}
|
|
|
|
response := `{"error":"database is required"}`
|
|
if body != response {
|
|
t.Fatalf("unexpected body: expected %s, actual %s", response, body)
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveShowSeries(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
srvr.SetDefaultRetentionPolicy("foo", "bar")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [
|
|
{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "uswest"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "gpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}
|
|
]}`)
|
|
|
|
if status != http.StatusOK {
|
|
t.Log(body)
|
|
t.Fatalf("unexpected status after write: %d", status)
|
|
}
|
|
|
|
var tests = []struct {
|
|
q string
|
|
r *influxdb.Results
|
|
err string
|
|
}{
|
|
// SHOW SERIES
|
|
{
|
|
q: `SHOW SERIES`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"host", "region"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01", ""}),
|
|
str2iface([]string{"server01", "uswest"}),
|
|
str2iface([]string{"server01", "useast"}),
|
|
str2iface([]string{"server02", "useast"}),
|
|
},
|
|
},
|
|
&influxql.Row{
|
|
Name: "gpu",
|
|
Columns: []string{"host", "region"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server02", "useast"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW SERIES ... LIMIT
|
|
// {
|
|
// q: `SHOW SERIES LIMIT 1`,
|
|
// r: &influxdb.Results{
|
|
// Results: []*influxdb.Result{
|
|
// &influxdb.Result{
|
|
// Rows: []*influxql.Row{
|
|
// &influxql.Row{
|
|
// Name: "cpu",
|
|
// Columns: []string{"host", "region"},
|
|
// Values: [][]interface{}{
|
|
// str2iface([]string{"server01", ""}),
|
|
// str2iface([]string{"server01", "uswest"}),
|
|
// str2iface([]string{"server01", "useast"}),
|
|
// str2iface([]string{"server02", "useast"}),
|
|
// },
|
|
// },
|
|
// },
|
|
// },
|
|
// },
|
|
// },
|
|
// },
|
|
// SHOW SERIES FROM
|
|
{
|
|
q: `SHOW SERIES FROM cpu`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"host", "region"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01", ""}),
|
|
str2iface([]string{"server01", "uswest"}),
|
|
str2iface([]string{"server01", "useast"}),
|
|
str2iface([]string{"server02", "useast"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW SERIES WHERE
|
|
{
|
|
q: `SHOW SERIES WHERE region = 'uswest'`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"host", "region"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01", "uswest"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW SERIES FROM ... WHERE
|
|
{
|
|
q: `SHOW SERIES FROM cpu WHERE region = 'useast'`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"host", "region"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01", "useast"}),
|
|
str2iface([]string{"server02", "useast"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
query := map[string]string{"db": "foo", "q": tt.q}
|
|
status, body = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Errorf("unexpected status: %d", status)
|
|
}
|
|
|
|
r := &influxdb.Results{}
|
|
if err := json.Unmarshal([]byte(body), r); err != nil {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Error(err)
|
|
}
|
|
|
|
if !reflect.DeepEqual(tt.err, errstring(r.Err)) {
|
|
t.Errorf("%d. %s: error mismatch:\n exp=%s\n got=%s\n\n", i, tt.q, tt.err, r.Err)
|
|
} else if tt.err == "" && !reflect.DeepEqual(tt.r, r) {
|
|
t.Log(body)
|
|
t.Errorf("%d. %s: result mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.q, tt.r, r)
|
|
}
|
|
}
|
|
}
|
|
|
|
// str2iface converts an array of strings to an array of interfaces.
|
|
func str2iface(strs []string) []interface{} {
|
|
a := make([]interface{}, 0, len(strs))
|
|
for _, s := range strs {
|
|
a = append(a, interface{}(s))
|
|
}
|
|
return a
|
|
}
|
|
|
|
func TestHandler_serveShowMeasurements(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [
|
|
{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "uswest"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "gpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}
|
|
]}`)
|
|
|
|
if status != http.StatusOK {
|
|
t.Log(body)
|
|
t.Fatalf("unexpected status after write: %d", status)
|
|
}
|
|
|
|
var tests = []struct {
|
|
q string
|
|
r string
|
|
err string
|
|
}{
|
|
// SHOW SERIES
|
|
{
|
|
q: `SHOW MEASUREMENTS LIMIT 2`,
|
|
r: `{"results":[{"rows":[{"name":"cpu","columns":["host","region"]},{"name":"gpu","columns":["host","region"]}]}]}`,
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
query := map[string]string{"db": "foo", "q": tt.q}
|
|
status, body = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Errorf("unexpected status: %d", status)
|
|
}
|
|
|
|
r := &influxdb.Results{}
|
|
if err := json.Unmarshal([]byte(body), r); err != nil {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Error("error marshaling result: ", err)
|
|
}
|
|
|
|
if body != tt.r {
|
|
t.Errorf("result mismatch\n exp: %s\n got: %s\n", tt.r, body)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveShowTagKeys(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
srvr.SetDefaultRetentionPolicy("foo", "bar")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [
|
|
{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "uswest"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "gpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}
|
|
]}`)
|
|
|
|
if status != http.StatusOK {
|
|
t.Log(body)
|
|
t.Fatalf("unexpected status after write: %d", status)
|
|
}
|
|
|
|
var tests = []struct {
|
|
q string
|
|
r *influxdb.Results
|
|
err string
|
|
}{
|
|
// SHOW TAG KEYS
|
|
{
|
|
q: `SHOW TAG KEYS`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagKey"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"host"}),
|
|
str2iface([]string{"region"}),
|
|
},
|
|
},
|
|
&influxql.Row{
|
|
Name: "gpu",
|
|
Columns: []string{"tagKey"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"host"}),
|
|
str2iface([]string{"region"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW TAG KEYS FROM...
|
|
{
|
|
q: `SHOW TAG KEYS FROM cpu`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagKey"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"host"}),
|
|
str2iface([]string{"region"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for i, tt := range tests {
|
|
query := map[string]string{"db": "foo", "q": tt.q}
|
|
status, body = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Errorf("unexpected status: %d", status)
|
|
}
|
|
|
|
r := &influxdb.Results{}
|
|
if err := json.Unmarshal([]byte(body), r); err != nil {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Error(err)
|
|
}
|
|
|
|
if !reflect.DeepEqual(tt.err, errstring(r.Err)) {
|
|
t.Errorf("%d. %s: error mismatch:\n exp=%s\n got=%s\n\n", i, tt.q, tt.err, r.Err)
|
|
} else if tt.err == "" && !reflect.DeepEqual(tt.r, r) {
|
|
b, _ := json.Marshal(tt.r)
|
|
t.Log(string(b))
|
|
t.Log(body)
|
|
t.Errorf("%d. %s: result mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.q, tt.r, r)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHandler_serveShowTagValues(t *testing.T) {
|
|
srvr := OpenServer(NewMessagingClient())
|
|
srvr.CreateDatabase("foo")
|
|
srvr.CreateRetentionPolicy("foo", influxdb.NewRetentionPolicy("bar"))
|
|
srvr.SetDefaultRetentionPolicy("foo", "bar")
|
|
s := NewHTTPServer(srvr)
|
|
defer s.Close()
|
|
|
|
status, body := MustHTTP("POST", s.URL+`/write`, nil, nil, `{"database" : "foo", "retentionPolicy" : "bar", "points": [
|
|
{"name": "cpu", "tags": {"host": "server01"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "uswest"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server01", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "cpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}},
|
|
{"name": "gpu", "tags": {"host": "server02", "region": "useast"},"timestamp": "2009-11-10T23:00:00Z","values": {"value": 100}}
|
|
]}`)
|
|
|
|
if status != http.StatusOK {
|
|
t.Log(body)
|
|
t.Fatalf("unexpected status after write: %d", status)
|
|
}
|
|
|
|
var tests = []struct {
|
|
q string
|
|
r *influxdb.Results
|
|
err string
|
|
}{
|
|
// SHOW TAG VALUES
|
|
{
|
|
q: `SHOW TAG VALUES WITH KEY = host`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagValue"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01"}),
|
|
str2iface([]string{"server02"}),
|
|
},
|
|
},
|
|
&influxql.Row{
|
|
Name: "gpu",
|
|
Columns: []string{"tagValue"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server02"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW TAG VALUES FROM ...
|
|
{
|
|
q: `SHOW TAG VALUES FROM cpu WITH KEY = host`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagValue"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01"}),
|
|
str2iface([]string{"server02"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW TAG VALUES FROM ... WHERE ...
|
|
{
|
|
q: `SHOW TAG VALUES FROM cpu WITH KEY = host WHERE region = 'uswest'`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagValue"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
// SHOW TAG VALUES FROM ... WITH KEY IN ... WHERE ...
|
|
{
|
|
q: `SHOW TAG VALUES FROM cpu WITH KEY IN (host, region) WHERE region = 'uswest'`,
|
|
r: &influxdb.Results{
|
|
Results: []*influxdb.Result{
|
|
&influxdb.Result{
|
|
Rows: []*influxql.Row{
|
|
&influxql.Row{
|
|
Name: "cpu",
|
|
Columns: []string{"tagValue"},
|
|
Values: [][]interface{}{
|
|
str2iface([]string{"server01"}),
|
|
str2iface([]string{"uswest"}),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for i, tt := range tests {
|
|
query := map[string]string{"db": "foo", "q": tt.q}
|
|
status, body = MustHTTP("GET", s.URL+`/query`, query, nil, "")
|
|
|
|
if status != http.StatusOK {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Errorf("unexpected status: %d", status)
|
|
}
|
|
|
|
r := &influxdb.Results{}
|
|
if err := json.Unmarshal([]byte(body), r); err != nil {
|
|
t.Logf("query #%d: %s", i, tt.q)
|
|
t.Log(body)
|
|
t.Error(err)
|
|
}
|
|
|
|
if !reflect.DeepEqual(tt.err, errstring(r.Err)) {
|
|
t.Errorf("%d. %s: error mismatch:\n exp=%s\n got=%s\n\n", i, tt.q, tt.err, r.Err)
|
|
} else if tt.err == "" && !reflect.DeepEqual(tt.r, r) {
|
|
b, _ := json.Marshal(tt.r)
|
|
t.Log(string(b))
|
|
t.Log(body)
|
|
t.Errorf("%d. %s: result mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.q, tt.r, r)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Utility functions for this test suite.
|
|
|
|
func MustHTTP(verb, path string, params, headers map[string]string, body string) (int, string) {
|
|
req, err := http.NewRequest(verb, path, bytes.NewBuffer([]byte(body)))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if params != nil {
|
|
q := url.Values{}
|
|
for k, v := range params {
|
|
q.Set(k, v)
|
|
}
|
|
req.URL.RawQuery = q.Encode()
|
|
}
|
|
|
|
for k, v := range headers {
|
|
req.Header.Set(k, v)
|
|
}
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
client := &http.Client{}
|
|
resp, err := client.Do(req)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
b, err := ioutil.ReadAll(resp.Body)
|
|
return resp.StatusCode, strings.TrimRight(string(b), "\n")
|
|
}
|
|
|
|
// MustParseURL parses a string into a URL. Panic on error.
|
|
func MustParseURL(s string) *url.URL {
|
|
u, err := url.Parse(s)
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return u
|
|
}
|
|
|
|
// Server is a test HTTP server that wraps a handler
|
|
type HTTPServer struct {
|
|
*httptest.Server
|
|
Handler *httpd.Handler
|
|
}
|
|
|
|
func NewHTTPServer(s *Server) *HTTPServer {
|
|
h := httpd.NewHandler(s.Server, false, "X.X")
|
|
return &HTTPServer{httptest.NewServer(h), h}
|
|
}
|
|
|
|
func NewAuthenticatedHTTPServer(s *Server) *HTTPServer {
|
|
h := httpd.NewHandler(s.Server, true, "X.X")
|
|
return &HTTPServer{httptest.NewServer(h), h}
|
|
}
|
|
|
|
func (s *HTTPServer) Close() {
|
|
s.Server.Close()
|
|
}
|
|
|
|
// Server is a wrapping test struct for influxdb.Server.
|
|
type Server struct {
|
|
*influxdb.Server
|
|
}
|
|
|
|
// NewServer returns a new test server instance.
|
|
func NewServer() *Server {
|
|
return &Server{influxdb.NewServer()}
|
|
}
|
|
|
|
// OpenServer returns a new, open test server instance.
|
|
func OpenServer(client influxdb.MessagingClient) *Server {
|
|
s := OpenUninitializedServer(client)
|
|
if err := s.Initialize(&url.URL{Host: "127.0.0.1:8080"}); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return s
|
|
}
|
|
|
|
// OpenUninitializedServer returns a new, uninitialized, open test server instance.
|
|
func OpenUninitializedServer(client influxdb.MessagingClient) *Server {
|
|
s := NewServer()
|
|
if err := s.Open(tempfile()); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
if err := s.SetClient(client); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return s
|
|
}
|
|
|
|
// TODO corylanou: evaluate how much of this should be in this package
|
|
// vs. how much should be a mocked out interface
|
|
// MessagingClient represents a test client for the messaging broker.
|
|
type MessagingClient struct {
|
|
index uint64
|
|
c chan *messaging.Message
|
|
|
|
PublishFunc func(*messaging.Message) (uint64, error)
|
|
CreateReplicaFunc func(replicaID uint64) error
|
|
DeleteReplicaFunc func(replicaID uint64) error
|
|
SubscribeFunc func(replicaID, topicID uint64) error
|
|
UnsubscribeFunc func(replicaID, topicID uint64) error
|
|
}
|
|
|
|
// NewMessagingClient returns a new instance of MessagingClient.
|
|
func NewMessagingClient() *MessagingClient {
|
|
c := &MessagingClient{c: make(chan *messaging.Message, 1)}
|
|
c.PublishFunc = c.send
|
|
c.CreateReplicaFunc = func(replicaID uint64) error { return nil }
|
|
c.DeleteReplicaFunc = func(replicaID uint64) error { return nil }
|
|
c.SubscribeFunc = func(replicaID, topicID uint64) error { return nil }
|
|
c.UnsubscribeFunc = func(replicaID, topicID uint64) error { return nil }
|
|
return c
|
|
}
|
|
|
|
// Publish attaches an autoincrementing index to the message.
|
|
// This function also execute's the client's PublishFunc mock function.
|
|
func (c *MessagingClient) Publish(m *messaging.Message) (uint64, error) {
|
|
c.index++
|
|
m.Index = c.index
|
|
return c.PublishFunc(m)
|
|
}
|
|
|
|
// send sends the message through to the channel.
|
|
// This is the default value of PublishFunc.
|
|
func (c *MessagingClient) send(m *messaging.Message) (uint64, error) {
|
|
c.c <- m
|
|
return m.Index, nil
|
|
}
|
|
|
|
// Creates a new replica with a given ID on the broker.
|
|
func (c *MessagingClient) CreateReplica(replicaID uint64) error {
|
|
return c.CreateReplicaFunc(replicaID)
|
|
}
|
|
|
|
// Deletes an existing replica with a given ID from the broker.
|
|
func (c *MessagingClient) DeleteReplica(replicaID uint64) error {
|
|
return c.DeleteReplicaFunc(replicaID)
|
|
}
|
|
|
|
// Subscribe adds a subscription to a replica for a topic on the broker.
|
|
func (c *MessagingClient) Subscribe(replicaID, topicID uint64) error {
|
|
return c.SubscribeFunc(replicaID, topicID)
|
|
}
|
|
|
|
// Unsubscribe removes a subscrition from a replica for a topic on the broker.
|
|
func (c *MessagingClient) Unsubscribe(replicaID, topicID uint64) error {
|
|
return c.UnsubscribeFunc(replicaID, topicID)
|
|
}
|
|
|
|
// C returns a channel for streaming message.
|
|
func (c *MessagingClient) C() <-chan *messaging.Message { return c.c }
|
|
|
|
// tempfile returns a temporary path.
|
|
func tempfile() string {
|
|
f, _ := ioutil.TempFile("", "influxdb-")
|
|
path := f.Name()
|
|
f.Close()
|
|
os.Remove(path)
|
|
return path
|
|
}
|
|
|
|
// errstring converts an error to its string representation.
|
|
func errstring(err error) string {
|
|
if err != nil {
|
|
return err.Error()
|
|
}
|
|
return ""
|
|
}
|