2016-10-18 12:02:36 +00:00
|
|
|
package opentsdb
|
2015-06-08 16:20:54 +00:00
|
|
|
|
|
|
|
import (
|
2016-10-18 12:02:36 +00:00
|
|
|
"errors"
|
2016-10-13 18:16:34 +00:00
|
|
|
"fmt"
|
2015-06-08 16:20:54 +00:00
|
|
|
"net"
|
|
|
|
"net/http"
|
2016-12-01 18:26:23 +00:00
|
|
|
"os"
|
2015-06-08 16:20:54 +00:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
"sync/atomic"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
2016-10-13 18:16:34 +00:00
|
|
|
"github.com/influxdata/influxdb/internal"
|
2016-02-10 17:26:18 +00:00
|
|
|
"github.com/influxdata/influxdb/models"
|
|
|
|
"github.com/influxdata/influxdb/services/meta"
|
2017-02-17 23:17:22 +00:00
|
|
|
"github.com/uber-go/zap"
|
2015-06-08 16:20:54 +00:00
|
|
|
)
|
|
|
|
|
2016-10-13 18:16:34 +00:00
|
|
|
func Test_Service_OpenClose(t *testing.T) {
|
2016-12-14 16:17:52 +00:00
|
|
|
// Let the OS assign a random port since we are only opening and closing the service,
|
|
|
|
// not actually connecting to it.
|
|
|
|
service := NewTestService("db0", "127.0.0.1:0")
|
2016-10-13 18:16:34 +00:00
|
|
|
|
|
|
|
// Closing a closed service is fine.
|
|
|
|
if err := service.Service.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Closing a closed service again is fine.
|
|
|
|
if err := service.Service.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := service.Service.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Opening an already open service is fine.
|
|
|
|
if err := service.Service.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reopening a previously opened service is fine.
|
|
|
|
if err := service.Service.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := service.Service.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tidy up.
|
|
|
|
if err := service.Service.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
// Ensure a point can be written via the telnet protocol.
|
|
|
|
func TestService_CreatesDatabase(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
database := "db0"
|
|
|
|
s := NewTestService(database, "127.0.0.1:0")
|
|
|
|
s.WritePointsFn = func(string, string, models.ConsistencyLevel, []models.Point) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
called := make(chan struct{})
|
|
|
|
s.MetaClient.CreateDatabaseFn = func(name string) (*meta.DatabaseInfo, error) {
|
|
|
|
if name != database {
|
|
|
|
t.Errorf("\n\texp = %s\n\tgot = %s\n", database, name)
|
|
|
|
}
|
2016-10-18 12:45:58 +00:00
|
|
|
// Allow some time for the caller to return and the ready status to
|
|
|
|
// be set.
|
|
|
|
time.AfterFunc(10*time.Millisecond, func() { called <- struct{}{} })
|
2016-10-18 12:02:36 +00:00
|
|
|
return nil, errors.New("an error")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.Service.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
points, err := models.ParsePointsString(`cpu value=1`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Service.batcher.In() <- points[0] // Send a point.
|
|
|
|
s.Service.batcher.Flush()
|
|
|
|
select {
|
|
|
|
case <-called:
|
|
|
|
// OK
|
2016-10-18 12:45:58 +00:00
|
|
|
case <-time.NewTimer(5 * time.Second).C:
|
2016-10-18 12:02:36 +00:00
|
|
|
t.Fatal("Service should have attempted to create database")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ready status should not have been switched due to meta client error.
|
|
|
|
s.Service.mu.RLock()
|
|
|
|
ready := s.Service.ready
|
|
|
|
s.Service.mu.RUnlock()
|
|
|
|
|
|
|
|
if got, exp := ready, false; got != exp {
|
|
|
|
t.Fatalf("got %v, expected %v", got, exp)
|
|
|
|
}
|
|
|
|
|
|
|
|
// This time MC won't cause an error.
|
|
|
|
s.MetaClient.CreateDatabaseFn = func(name string) (*meta.DatabaseInfo, error) {
|
2016-10-18 12:45:58 +00:00
|
|
|
// Allow some time for the caller to return and the ready status to
|
|
|
|
// be set.
|
|
|
|
time.AfterFunc(10*time.Millisecond, func() { called <- struct{}{} })
|
2016-10-18 12:02:36 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Service.batcher.In() <- points[0] // Send a point.
|
|
|
|
s.Service.batcher.Flush()
|
|
|
|
select {
|
|
|
|
case <-called:
|
|
|
|
// OK
|
2016-10-18 12:45:58 +00:00
|
|
|
case <-time.NewTimer(5 * time.Second).C:
|
2016-10-18 12:02:36 +00:00
|
|
|
t.Fatal("Service should have attempted to create database")
|
|
|
|
}
|
|
|
|
|
|
|
|
// ready status should not have been switched due to meta client error.
|
|
|
|
s.Service.mu.RLock()
|
|
|
|
ready = s.Service.ready
|
|
|
|
s.Service.mu.RUnlock()
|
|
|
|
|
|
|
|
if got, exp := ready, true; got != exp {
|
|
|
|
t.Fatalf("got %v, expected %v", got, exp)
|
|
|
|
}
|
|
|
|
|
2016-10-18 12:45:58 +00:00
|
|
|
s.Service.Close()
|
2016-10-18 12:02:36 +00:00
|
|
|
}
|
|
|
|
|
2015-06-08 16:20:54 +00:00
|
|
|
// Ensure a point can be written via the telnet protocol.
|
|
|
|
func TestService_Telnet(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
s := NewTestService("db0", "127.0.0.1:0")
|
2016-10-13 18:16:34 +00:00
|
|
|
if err := s.Service.Open(); err != nil {
|
2015-06-08 16:20:54 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-10-13 18:16:34 +00:00
|
|
|
defer s.Service.Close()
|
2015-06-08 16:20:54 +00:00
|
|
|
|
|
|
|
// Mock points writer.
|
|
|
|
var called int32
|
2016-10-13 18:16:34 +00:00
|
|
|
s.WritePointsFn = func(database, retentionPolicy string, consistencyLevel models.ConsistencyLevel, points []models.Point) error {
|
2015-06-08 16:20:54 +00:00
|
|
|
atomic.StoreInt32(&called, 1)
|
|
|
|
|
2016-03-10 23:22:22 +00:00
|
|
|
if database != "db0" {
|
|
|
|
t.Fatalf("unexpected database: %s", database)
|
|
|
|
} else if retentionPolicy != "" {
|
|
|
|
t.Fatalf("unexpected retention policy: %s", retentionPolicy)
|
|
|
|
} else if !reflect.DeepEqual(points, []models.Point{
|
2015-10-27 16:21:54 +00:00
|
|
|
models.MustNewPoint(
|
2015-06-08 16:20:54 +00:00
|
|
|
"sys.cpu.user",
|
2016-06-30 16:49:53 +00:00
|
|
|
models.NewTags(map[string]string{"host": "webserver01", "cpu": "0"}),
|
2015-06-08 16:20:54 +00:00
|
|
|
map[string]interface{}{"value": 42.5},
|
|
|
|
time.Unix(1356998400, 0),
|
|
|
|
),
|
|
|
|
}) {
|
2016-03-10 23:22:22 +00:00
|
|
|
t.Fatalf("unexpected points: %#v", points)
|
2015-06-08 16:20:54 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open connection to the service.
|
2016-10-13 18:16:34 +00:00
|
|
|
conn, err := net.Dial("tcp", s.Service.Addr().String())
|
2015-06-08 16:20:54 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
// Write telnet data and close.
|
|
|
|
if _, err := conn.Write([]byte("put sys.cpu.user 1356998400 42.5 host=webserver01 cpu=0")); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := conn.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2016-02-10 15:46:17 +00:00
|
|
|
tick := time.Tick(10 * time.Millisecond)
|
|
|
|
timeout := time.After(10 * time.Second)
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-tick:
|
|
|
|
// Verify that the writer was called.
|
|
|
|
if atomic.LoadInt32(&called) > 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case <-timeout:
|
|
|
|
t.Fatal("points writer not called")
|
|
|
|
}
|
2015-06-08 16:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure a point can be written via the HTTP protocol.
|
|
|
|
func TestService_HTTP(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
s := NewTestService("db0", "127.0.0.1:0")
|
2016-10-13 18:16:34 +00:00
|
|
|
if err := s.Service.Open(); err != nil {
|
2015-06-08 16:20:54 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-10-13 18:16:34 +00:00
|
|
|
defer s.Service.Close()
|
2015-06-08 16:20:54 +00:00
|
|
|
|
|
|
|
// Mock points writer.
|
|
|
|
var called bool
|
2016-10-13 18:16:34 +00:00
|
|
|
s.WritePointsFn = func(database, retentionPolicy string, consistencyLevel models.ConsistencyLevel, points []models.Point) error {
|
2015-06-08 16:20:54 +00:00
|
|
|
called = true
|
2016-03-10 23:22:22 +00:00
|
|
|
if database != "db0" {
|
|
|
|
t.Fatalf("unexpected database: %s", database)
|
|
|
|
} else if retentionPolicy != "" {
|
|
|
|
t.Fatalf("unexpected retention policy: %s", retentionPolicy)
|
|
|
|
} else if !reflect.DeepEqual(points, []models.Point{
|
2015-10-27 16:21:54 +00:00
|
|
|
models.MustNewPoint(
|
2015-06-08 16:20:54 +00:00
|
|
|
"sys.cpu.nice",
|
2016-06-30 16:49:53 +00:00
|
|
|
models.NewTags(map[string]string{"dc": "lga", "host": "web01"}),
|
2015-06-08 16:20:54 +00:00
|
|
|
map[string]interface{}{"value": 18.0},
|
|
|
|
time.Unix(1346846400, 0),
|
|
|
|
),
|
|
|
|
}) {
|
2016-03-10 23:22:22 +00:00
|
|
|
spew.Dump(points)
|
|
|
|
t.Fatalf("unexpected points: %#v", points)
|
2015-06-08 16:20:54 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write HTTP request to server.
|
2016-10-13 18:16:34 +00:00
|
|
|
resp, err := http.Post("http://"+s.Service.Addr().String()+"/api/put", "application/json", strings.NewReader(`{"metric":"sys.cpu.nice", "timestamp":1346846400, "value":18, "tags":{"host":"web01", "dc":"lga"}}`))
|
2015-06-08 16:20:54 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
// Verify status and body.
|
|
|
|
if resp.StatusCode != http.StatusNoContent {
|
|
|
|
t.Fatalf("unexpected status code: %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify that the writer was called.
|
|
|
|
if !called {
|
|
|
|
t.Fatal("points writer not called")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
type TestService struct {
|
|
|
|
Service *Service
|
2016-10-13 18:16:34 +00:00
|
|
|
MetaClient *internal.MetaClientMock
|
|
|
|
WritePointsFn func(database, retentionPolicy string, consistencyLevel models.ConsistencyLevel, points []models.Point) error
|
2015-06-08 16:20:54 +00:00
|
|
|
}
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
// NewTestService returns a new instance of Service.
|
|
|
|
func NewTestService(database string, bind string) *TestService {
|
|
|
|
s, err := NewService(Config{
|
2016-10-13 18:16:34 +00:00
|
|
|
BindAddress: bind,
|
2015-06-09 22:10:32 +00:00
|
|
|
Database: database,
|
|
|
|
ConsistencyLevel: "one",
|
|
|
|
})
|
2015-06-08 16:20:54 +00:00
|
|
|
|
2016-10-13 18:16:34 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
2015-06-08 16:20:54 +00:00
|
|
|
}
|
|
|
|
|
2016-10-18 12:02:36 +00:00
|
|
|
service := &TestService{
|
2016-10-13 18:16:34 +00:00
|
|
|
Service: s,
|
|
|
|
MetaClient: &internal.MetaClientMock{},
|
|
|
|
}
|
2015-06-08 16:20:54 +00:00
|
|
|
|
2016-10-13 18:16:34 +00:00
|
|
|
service.MetaClient.CreateDatabaseFn = func(db string) (*meta.DatabaseInfo, error) {
|
|
|
|
if got, exp := db, database; got != exp {
|
|
|
|
return nil, fmt.Errorf("got %v, expected %v", got, exp)
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
2015-06-08 16:20:54 +00:00
|
|
|
|
2016-12-01 18:26:23 +00:00
|
|
|
if testing.Verbose() {
|
|
|
|
service.Service.WithLogger(zap.New(
|
|
|
|
zap.NewTextEncoder(),
|
|
|
|
zap.Output(os.Stderr),
|
|
|
|
))
|
2016-10-13 18:16:34 +00:00
|
|
|
}
|
2015-06-11 01:29:13 +00:00
|
|
|
|
2016-10-13 18:16:34 +00:00
|
|
|
service.Service.MetaClient = service.MetaClient
|
|
|
|
service.Service.PointsWriter = service
|
|
|
|
return service
|
2015-06-11 01:29:13 +00:00
|
|
|
}
|
|
|
|
|
2017-05-05 17:20:00 +00:00
|
|
|
func (s *TestService) WritePointsPrivileged(database, retentionPolicy string, consistencyLevel models.ConsistencyLevel, points []models.Point) error {
|
2016-10-13 18:16:34 +00:00
|
|
|
return s.WritePointsFn(database, retentionPolicy, consistencyLevel, points)
|
2015-06-11 01:29:13 +00:00
|
|
|
}
|