mirror of https://github.com/grafana/grafana.git
315 lines
8.9 KiB
Go
315 lines
8.9 KiB
Go
package loki
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestSample_MarshalJSON(t *testing.T) {
|
|
sample := Sample{
|
|
T: time.Unix(0, 1234567890000000000), // 1234567890 seconds in nanoseconds
|
|
V: "test log line",
|
|
}
|
|
|
|
data, err := json.Marshal(sample)
|
|
require.NoError(t, err)
|
|
|
|
expected := `["1234567890000000000","test log line"]`
|
|
assert.JSONEq(t, expected, string(data))
|
|
}
|
|
|
|
func TestSample_UnmarshalJSON(t *testing.T) {
|
|
t.Run("valid sample", func(t *testing.T) {
|
|
data := `["1234567890000000000","test log line"]`
|
|
var sample Sample
|
|
|
|
err := json.Unmarshal([]byte(data), &sample)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, time.Unix(0, 1234567890000000000), sample.T)
|
|
assert.Equal(t, "test log line", sample.V)
|
|
})
|
|
|
|
t.Run("invalid format", func(t *testing.T) {
|
|
data := `"invalid"`
|
|
var sample Sample
|
|
|
|
err := json.Unmarshal([]byte(data), &sample)
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "failed to deserialize sample")
|
|
})
|
|
|
|
t.Run("invalid timestamp", func(t *testing.T) {
|
|
data := `["not-a-number","test log line"]`
|
|
var sample Sample
|
|
|
|
err := json.Unmarshal([]byte(data), &sample)
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "timestamp in Loki sample not convertible")
|
|
})
|
|
}
|
|
|
|
func TestClient_Push(t *testing.T) {
|
|
t.Run("successful push", func(t *testing.T) {
|
|
var receivedBody PushRequest
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
assert.Equal(t, "/loki/api/v1/push", r.URL.Path)
|
|
assert.Equal(t, http.MethodPost, r.Method)
|
|
assert.Equal(t, "application/json", r.Header.Get("Content-Type"))
|
|
|
|
err := json.NewDecoder(r.Body).Decode(&receivedBody)
|
|
require.NoError(t, err)
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
streams := []Stream{
|
|
{
|
|
Stream: map[string]string{"job": "test"},
|
|
Values: []Sample{
|
|
{T: time.Unix(0, 1234567890000000000), V: "log line 1"},
|
|
{T: time.Unix(0, 1234567891000000000), V: "log line 2"},
|
|
},
|
|
},
|
|
}
|
|
|
|
err := client.Push(context.Background(), streams)
|
|
assert.NoError(t, err)
|
|
|
|
// Verify the request body
|
|
assert.Len(t, receivedBody.Streams, 1)
|
|
assert.Equal(t, "test", receivedBody.Streams[0].Stream["job"])
|
|
assert.Len(t, receivedBody.Streams[0].Values, 2)
|
|
})
|
|
|
|
t.Run("push failure", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
_, _ = w.Write([]byte("Bad request"))
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
streams := []Stream{{Stream: map[string]string{"job": "test"}}}
|
|
err := client.Push(context.Background(), streams)
|
|
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "non-200 response")
|
|
})
|
|
}
|
|
|
|
func TestClient_RangeQuery(t *testing.T) {
|
|
t.Run("successful query", func(t *testing.T) {
|
|
expectedResponse := QueryRes{
|
|
Data: QueryData{
|
|
Result: []Stream{
|
|
{
|
|
Stream: map[string]string{"job": "test"},
|
|
Values: []Sample{
|
|
{T: time.Unix(0, 1234567890000000000), V: "log line 1"},
|
|
{T: time.Unix(0, 1234567891000000000), V: "log line 2"},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
assert.Equal(t, "/loki/api/v1/query_range", r.URL.Path)
|
|
assert.Equal(t, http.MethodGet, r.Method)
|
|
|
|
// Check query parameters
|
|
params := r.URL.Query()
|
|
assert.Equal(t, `{job="test"}`, params.Get("query"))
|
|
assert.Equal(t, "1000000000", params.Get("start"))
|
|
assert.Equal(t, "2000000000", params.Get("end"))
|
|
assert.Equal(t, "100", params.Get("limit"))
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_ = json.NewEncoder(w).Encode(expectedResponse)
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
result, err := client.RangeQuery(
|
|
context.Background(),
|
|
`{job="test"}`,
|
|
1000000000, // start
|
|
2000000000, // end
|
|
100, // limit
|
|
)
|
|
|
|
assert.NoError(t, err)
|
|
assert.Len(t, result.Data.Result, 1)
|
|
assert.Equal(t, "test", result.Data.Result[0].Stream["job"])
|
|
assert.Len(t, result.Data.Result[0].Values, 2)
|
|
})
|
|
|
|
t.Run("query without limit", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
params := r.URL.Query()
|
|
assert.Equal(t, "", params.Get("limit")) // Should not be set
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_ = json.NewEncoder(w).Encode(QueryRes{})
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
_, err := client.RangeQuery(context.Background(), `{job="test"}`, 1000000000, 2000000000, 0)
|
|
assert.NoError(t, err)
|
|
})
|
|
|
|
t.Run("query failure", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
_, _ = w.Write([]byte("Bad query"))
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
_, err := client.RangeQuery(context.Background(), `{job="test"}`, 1000000000, 2000000000, 100)
|
|
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "non-200 response")
|
|
})
|
|
|
|
t.Run("invalid JSON response", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_, _ = w.Write([]byte("invalid json"))
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
_, err := client.RangeQuery(context.Background(), `{job="test"}`, 1000000000, 2000000000, 100)
|
|
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "error unmarshaling loki response")
|
|
})
|
|
}
|
|
|
|
func TestClient_setAuthAndTenantHeaders(t *testing.T) {
|
|
t.Run("with basic auth and tenant", func(t *testing.T) {
|
|
cfg := createTestConfig(t, "http://localhost", "http://localhost")
|
|
cfg.BasicAuthUser = "testuser"
|
|
cfg.BasicAuthPassword = "testpass"
|
|
cfg.TenantID = "test-tenant"
|
|
|
|
client := NewClient(cfg)
|
|
|
|
req, _ := http.NewRequest(http.MethodGet, "http://localhost", nil)
|
|
client.setAuthAndTenantHeaders(req)
|
|
|
|
username, password, ok := req.BasicAuth()
|
|
assert.True(t, ok)
|
|
assert.Equal(t, "testuser", username)
|
|
assert.Equal(t, "testpass", password)
|
|
assert.Equal(t, "test-tenant", req.Header.Get("X-Scope-OrgID"))
|
|
})
|
|
|
|
t.Run("without auth", func(t *testing.T) {
|
|
cfg := createTestConfig(t, "http://localhost", "http://localhost")
|
|
client := NewClient(cfg)
|
|
|
|
req, _ := http.NewRequest(http.MethodGet, "http://localhost", nil)
|
|
client.setAuthAndTenantHeaders(req)
|
|
|
|
_, _, ok := req.BasicAuth()
|
|
assert.False(t, ok)
|
|
assert.Equal(t, "", req.Header.Get("X-Scope-OrgID"))
|
|
})
|
|
}
|
|
|
|
func TestStream_JSONRoundtrip(t *testing.T) {
|
|
original := Stream{
|
|
Stream: map[string]string{
|
|
"job": "test-job",
|
|
"instance": "test-instance",
|
|
"namespace": "test-ns",
|
|
},
|
|
Values: []Sample{
|
|
{T: time.Unix(0, 1234567890000000000), V: "log line 1"},
|
|
{T: time.Unix(0, 1234567891000000000), V: "log line 2"},
|
|
{T: time.Unix(0, 1234567892000000000), V: "log line 3"},
|
|
},
|
|
}
|
|
|
|
// Marshal to JSON
|
|
data, err := json.Marshal(original)
|
|
require.NoError(t, err)
|
|
|
|
// Unmarshal back
|
|
var restored Stream
|
|
err = json.Unmarshal(data, &restored)
|
|
require.NoError(t, err)
|
|
|
|
// Verify all fields match
|
|
assert.Equal(t, original.Stream, restored.Stream)
|
|
assert.Len(t, restored.Values, len(original.Values))
|
|
|
|
for i, sample := range original.Values {
|
|
assert.True(t, sample.T.Equal(restored.Values[i].T),
|
|
fmt.Sprintf("Timestamp mismatch at index %d: expected %v, got %v", i, sample.T, restored.Values[i].T))
|
|
assert.Equal(t, sample.V, restored.Values[i].V)
|
|
}
|
|
}
|
|
|
|
// Helper functions
|
|
|
|
func createTestClient(t *testing.T, readURL, writeURL string) *Client {
|
|
cfg := createTestConfig(t, readURL, writeURL)
|
|
return NewClient(cfg)
|
|
}
|
|
|
|
func createTestConfig(t *testing.T, readURL, writeURL string) Config {
|
|
readParsed, err := url.Parse(readURL)
|
|
require.NoError(t, err)
|
|
|
|
writeParsed, err := url.Parse(writeURL)
|
|
require.NoError(t, err)
|
|
|
|
return Config{
|
|
ReadPathURL: readParsed,
|
|
WritePathURL: writeParsed,
|
|
ExternalLabels: map[string]string{"source": "test"},
|
|
MaxQuerySize: 1000,
|
|
}
|
|
}
|
|
|
|
func TestClient_ContextCancellation(t *testing.T) {
|
|
t.Run("push with cancelled context", func(t *testing.T) {
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
t.Error("Handler should not be called with cancelled context")
|
|
}))
|
|
defer server.Close()
|
|
|
|
client := createTestClient(t, server.URL, server.URL)
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel()
|
|
|
|
streams := []Stream{{Stream: map[string]string{"job": "test"}}}
|
|
err := client.Push(ctx, streams)
|
|
assert.Error(t, err)
|
|
assert.Contains(t, err.Error(), "context canceled")
|
|
})
|
|
}
|