parent
f525a6642a
commit
27cda64d90
@ -0,0 +1,5 @@
|
|||||||
|
# Copyright 2023 William Perron. All rights reserved. MIT License.
|
||||||
|
dist:
|
||||||
|
name: otelcol-dev
|
||||||
|
description: Basic Otel Collector for development
|
||||||
|
output_path: ./bin/otelcol-dev
|
@ -0,0 +1,4 @@
|
|||||||
|
-- Copyright 2023 William Perron. All rights reserved. MIT License.
|
||||||
|
DROP TABLE links;
|
||||||
|
DROP TABLE events;
|
||||||
|
DROP TABLE spans;
|
@ -0,0 +1,38 @@
|
|||||||
|
-- Copyright 2023 William Perron. All rights reserved. MIT License.
|
||||||
|
CREATE TABLE IF NOT EXISTS spans(
|
||||||
|
"span_id" BLOB,
|
||||||
|
"trace_id" BLOB,
|
||||||
|
"parent_span_id" BLOB,
|
||||||
|
"tracestate" TEXT,
|
||||||
|
"__service_name" TEXT,
|
||||||
|
"__duration" INTEGER,
|
||||||
|
"name" TEXT,
|
||||||
|
"kind" TEXT,
|
||||||
|
"start_time" INTEGER,
|
||||||
|
"end_time" INTEGER,
|
||||||
|
"status_code" INTEGER,
|
||||||
|
"status_description" TEXT,
|
||||||
|
"attributes" TEXT,
|
||||||
|
"dropped_attributes_count" INTEGER,
|
||||||
|
"resource_attributes" TEXT,
|
||||||
|
"resource_dropped_attributes_count" INTEGER,
|
||||||
|
PRIMARY KEY ("span_id", "trace_id")
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS events(
|
||||||
|
"span_id" BLOB,
|
||||||
|
"timestamp" INTEGER,
|
||||||
|
"name" TEXT,
|
||||||
|
"attributes" TEXT,
|
||||||
|
"dropped_attributes_count" INTEGER,
|
||||||
|
FOREIGN KEY ("span_id") REFERENCES spans("span_id")
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS links(
|
||||||
|
"parent_span_id" BLOB,
|
||||||
|
"span_id" BLOB,
|
||||||
|
"trace_id" BLOB,
|
||||||
|
"tracestate" TEXT,
|
||||||
|
"attributes" TEXT,
|
||||||
|
"dropped_attributes_count" INTEGER
|
||||||
|
);
|
@ -0,0 +1,257 @@
|
|||||||
|
// Copyright 2023 William Perron. All rights reserved. MIT License.
|
||||||
|
package sqliteexporter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"database/sql"
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
_ "github.com/mattn/go-sqlite3"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"go.opentelemetry.io/collector/pdata/pcommon"
|
||||||
|
"go.opentelemetry.io/collector/pdata/ptrace"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Test_ExporterExportSpan(t *testing.T) {
|
||||||
|
ctx := context.Background()
|
||||||
|
now := time.Now()
|
||||||
|
|
||||||
|
// manually build the exporter so we can inspect the database
|
||||||
|
db, err := sql.Open("sqlite3", ":memory:")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
tx, _ := db.Begin()
|
||||||
|
|
||||||
|
_, err = tx.Exec(initScript)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = tx.Commit()
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
ex := sqliteExporter{db: db}
|
||||||
|
|
||||||
|
// build the trace
|
||||||
|
testTrace := ptrace.NewTraces()
|
||||||
|
rs := testTrace.ResourceSpans().AppendEmpty()
|
||||||
|
r := rs.Resource()
|
||||||
|
r.Attributes().PutStr("service.name", "test-service")
|
||||||
|
ss := rs.ScopeSpans().AppendEmpty()
|
||||||
|
|
||||||
|
// define a first server span
|
||||||
|
span1 := ss.Spans().AppendEmpty()
|
||||||
|
span1.SetTraceID(pcommon.TraceID{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01})
|
||||||
|
span1.SetSpanID(pcommon.SpanID{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x01})
|
||||||
|
ts := span1.TraceState()
|
||||||
|
ts.FromRaw("rojo=00f067aa0ba902b7")
|
||||||
|
span1.SetName("span1")
|
||||||
|
span1.SetKind(ptrace.SpanKindServer)
|
||||||
|
span1.SetStartTimestamp(pcommon.NewTimestampFromTime(now.Add(-5 * time.Millisecond)))
|
||||||
|
span1.SetEndTimestamp(pcommon.NewTimestampFromTime(now))
|
||||||
|
span1.Status().SetCode(ptrace.StatusCodeOk)
|
||||||
|
span1.Status().SetMessage(ptrace.StatusCodeOk.String())
|
||||||
|
span1.Attributes().PutStr("http.method", "GET")
|
||||||
|
span1.Attributes().PutInt("http.status_code", 200)
|
||||||
|
span1event1 := span1.Events().AppendEmpty()
|
||||||
|
span1event1.SetTimestamp(pcommon.NewTimestampFromTime(now.Add(-3 * time.Millisecond)))
|
||||||
|
span1event1.SetName("this_happened")
|
||||||
|
span1event1.Attributes().PutStr("value", "example")
|
||||||
|
span1event2 := span1.Events().AppendEmpty()
|
||||||
|
span1event2.SetTimestamp(pcommon.NewTimestampFromTime(now.Add(-4 * time.Millisecond)))
|
||||||
|
span1event2.SetName("that_happened")
|
||||||
|
span1event2.Attributes().PutStr("value", "example")
|
||||||
|
|
||||||
|
// define an internal child span for span1
|
||||||
|
span2 := ss.Spans().AppendEmpty()
|
||||||
|
span2.SetTraceID(pcommon.TraceID{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01})
|
||||||
|
span2.SetSpanID(pcommon.SpanID{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x02})
|
||||||
|
span2.SetParentSpanID(pcommon.SpanID{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x01})
|
||||||
|
span2.SetName("span2")
|
||||||
|
span2.SetKind(ptrace.SpanKindInternal)
|
||||||
|
span2.SetStartTimestamp(pcommon.NewTimestampFromTime(now.Add(-4 * time.Millisecond)))
|
||||||
|
span2.SetEndTimestamp(pcommon.NewTimestampFromTime(now.Add(-1 * time.Millisecond)))
|
||||||
|
span2.Status().SetCode(ptrace.StatusCodeOk)
|
||||||
|
span2.Status().SetMessage(ptrace.StatusCodeOk.String())
|
||||||
|
span2.Attributes().PutStr("custom.key", "custom-value")
|
||||||
|
span2event1 := span2.Events().AppendEmpty()
|
||||||
|
span2event1.SetTimestamp(pcommon.NewTimestampFromTime(now.Add(-3 * time.Millisecond)))
|
||||||
|
span2event1.SetName("this_happened")
|
||||||
|
span2event1.Attributes().PutStr("value", "example")
|
||||||
|
span2event2 := span2.Events().AppendEmpty()
|
||||||
|
span2event2.SetTimestamp(pcommon.NewTimestampFromTime(now.Add(-4 * time.Millisecond)))
|
||||||
|
span2event2.SetName("that_happened")
|
||||||
|
span2event2.Attributes().PutStr("value", "example")
|
||||||
|
span2link1 := span2.Links().AppendEmpty()
|
||||||
|
span2link1.SetSpanID(span1.SpanID())
|
||||||
|
span2link1.SetTraceID(span1.TraceID())
|
||||||
|
span2link1.Attributes().PutStr("relation", "follows_from")
|
||||||
|
|
||||||
|
err = ex.ConsumeTraces(ctx, testTrace)
|
||||||
|
require.Nil(t, err)
|
||||||
|
|
||||||
|
row := db.QueryRow("select count(1) from spans;")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
var total int
|
||||||
|
err = row.Scan(&total)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 2, total)
|
||||||
|
|
||||||
|
rows, err := db.Query(`select
|
||||||
|
span_id,
|
||||||
|
trace_id,
|
||||||
|
parent_span_id,
|
||||||
|
tracestate,
|
||||||
|
__service_name,
|
||||||
|
__duration,
|
||||||
|
name,
|
||||||
|
kind,
|
||||||
|
start_time,
|
||||||
|
end_time,
|
||||||
|
status_code,
|
||||||
|
status_description,
|
||||||
|
attributes,
|
||||||
|
dropped_attributes_count,
|
||||||
|
resource_attributes,
|
||||||
|
resource_dropped_attributes_count
|
||||||
|
from spans order by name asc;`)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
ok := rows.Next()
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
var rspanId, rtraceId, rparentSpanId sql.RawBytes
|
||||||
|
var tracestate, serviceName, name, kind, statusDescription, attributes, resourceAttributes string
|
||||||
|
var duration, startTime, endTime, statusCode, droppedAttributesCount, resourceDroppedAttributesCount int
|
||||||
|
err = rows.Scan(
|
||||||
|
&rspanId,
|
||||||
|
&rtraceId,
|
||||||
|
&rparentSpanId,
|
||||||
|
&tracestate,
|
||||||
|
&serviceName,
|
||||||
|
&duration,
|
||||||
|
&name,
|
||||||
|
&kind,
|
||||||
|
&startTime,
|
||||||
|
&endTime,
|
||||||
|
&statusCode,
|
||||||
|
&statusDescription,
|
||||||
|
&attributes,
|
||||||
|
&droppedAttributesCount,
|
||||||
|
&resourceAttributes,
|
||||||
|
&resourceDroppedAttributesCount,
|
||||||
|
)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, sql.RawBytes([]byte{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x01}), rspanId)
|
||||||
|
assert.Equal(t, sql.RawBytes([]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}), rtraceId)
|
||||||
|
assert.Nil(t, rparentSpanId)
|
||||||
|
assert.Equal(t, "rojo=00f067aa0ba902b7", tracestate)
|
||||||
|
assert.Equal(t, "test-service", serviceName)
|
||||||
|
assert.Equal(t, 5000, duration)
|
||||||
|
assert.Equal(t, "span1", name)
|
||||||
|
assert.Equal(t, ptrace.SpanKindServer.String(), kind)
|
||||||
|
assert.Equal(t, now.Add(-5*time.Millisecond).Unix(), int64(startTime))
|
||||||
|
assert.Equal(t, now.Unix(), int64(endTime))
|
||||||
|
assert.Equal(t, int(ptrace.StatusCodeOk), statusCode)
|
||||||
|
assert.Equal(t, ptrace.StatusCodeOk.String(), statusDescription)
|
||||||
|
assert.Equal(t, "{\"http.method\":\"GET\",\"http.status_code\":200}", attributes)
|
||||||
|
assert.Equal(t, 0, droppedAttributesCount)
|
||||||
|
assert.Equal(t, "{\"service.name\":\"test-service\"}", resourceAttributes)
|
||||||
|
assert.Equal(t, 0, resourceDroppedAttributesCount)
|
||||||
|
|
||||||
|
ok = rows.Next()
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
err = rows.Scan(
|
||||||
|
&rspanId,
|
||||||
|
&rtraceId,
|
||||||
|
&rparentSpanId,
|
||||||
|
&tracestate,
|
||||||
|
&serviceName,
|
||||||
|
&duration,
|
||||||
|
&name,
|
||||||
|
&kind,
|
||||||
|
&startTime,
|
||||||
|
&endTime,
|
||||||
|
&statusCode,
|
||||||
|
&statusDescription,
|
||||||
|
&attributes,
|
||||||
|
&droppedAttributesCount,
|
||||||
|
&resourceAttributes,
|
||||||
|
&resourceDroppedAttributesCount,
|
||||||
|
)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, sql.RawBytes([]byte{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x02}), rspanId)
|
||||||
|
assert.Equal(t, sql.RawBytes([]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}), rtraceId)
|
||||||
|
assert.Equal(t, sql.RawBytes([]byte{0xee, 0xbc, 0x00, 0x00, 0x00, 0x00, 0xab, 0x01}), rparentSpanId)
|
||||||
|
assert.Equal(t, "", tracestate)
|
||||||
|
assert.Equal(t, "test-service", serviceName)
|
||||||
|
assert.Equal(t, 3000, duration)
|
||||||
|
assert.Equal(t, "span2", name)
|
||||||
|
assert.Equal(t, ptrace.SpanKindInternal.String(), kind)
|
||||||
|
assert.Equal(t, now.Add(-4*time.Millisecond).Unix(), int64(startTime))
|
||||||
|
assert.Equal(t, now.Add(-1*time.Millisecond).Unix(), int64(endTime))
|
||||||
|
assert.Equal(t, int(ptrace.StatusCodeOk), statusCode)
|
||||||
|
assert.Equal(t, ptrace.StatusCodeOk.String(), statusDescription)
|
||||||
|
assert.Equal(t, "{\"custom.key\":\"custom-value\"}", attributes)
|
||||||
|
assert.Equal(t, 0, droppedAttributesCount)
|
||||||
|
assert.Equal(t, "{\"service.name\":\"test-service\"}", resourceAttributes)
|
||||||
|
assert.Equal(t, 0, resourceDroppedAttributesCount)
|
||||||
|
|
||||||
|
assert.False(t, rows.Next())
|
||||||
|
|
||||||
|
row = db.QueryRow("select count(1) from events;")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = row.Scan(&total)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 4, total)
|
||||||
|
|
||||||
|
row = db.QueryRow("select count(1) from links;")
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = row.Scan(&total)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 1, total)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_pcommonMapAsJSON(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
args pcommon.Map
|
||||||
|
want []byte
|
||||||
|
wantErr bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty",
|
||||||
|
args: pcommon.NewMap(),
|
||||||
|
want: []byte("{}"),
|
||||||
|
wantErr: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "simple",
|
||||||
|
args: func() pcommon.Map {
|
||||||
|
m := pcommon.NewMap()
|
||||||
|
m.PutStr("http.method", "GET")
|
||||||
|
m.PutInt("http.status_code", 200)
|
||||||
|
return m
|
||||||
|
}(),
|
||||||
|
want: []byte("{\"http.method\":\"GET\",\"http.status_code\":200}"),
|
||||||
|
wantErr: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
got, err := pcommonMapAsJSON(tt.args)
|
||||||
|
if (err != nil) != tt.wantErr {
|
||||||
|
t.Errorf("pcommonMapAsJSON() error = %v, wantErr %v", err, tt.wantErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !reflect.DeepEqual(got, tt.want) {
|
||||||
|
t.Errorf("pcommonMapAsJSON() = %v, want %v", string(got), string(tt.want))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue