kicad/thirdparty/sentry-native/tests/unit/test_tracing.c

812 lines
29 KiB
C

#include "sentry_testsupport.h"
#include "sentry_scope.h"
#include "sentry_string.h"
#include "sentry_tracing.h"
#include "sentry_uuid.h"
#define IS_NULL(Src, Field) \
sentry_value_is_null(sentry_value_get_by_key(Src, Field))
#define CHECK_STRING_PROPERTY(Src, Field, Expected) \
TEST_CHECK_STRING_EQUAL( \
sentry_value_as_string(sentry_value_get_by_key(Src, Field)), Expected)
SENTRY_TEST(basic_tracing_context)
{
sentry_transaction_t *opaque_tx
= sentry__transaction_new(sentry_value_new_null());
TEST_CHECK(!opaque_tx);
sentry_value_t tx = sentry_value_new_object();
opaque_tx = sentry__transaction_new(sentry__value_clone(tx));
sentry_value_set_by_key(tx, "op", sentry_value_new_string("honk.beep"));
TEST_CHECK(sentry_value_is_null(
sentry__value_get_trace_context(opaque_tx->inner)));
sentry_uuid_t trace_id = sentry_uuid_new_v4();
sentry_value_set_by_key(
tx, "trace_id", sentry__value_new_internal_uuid(&trace_id));
sentry__transaction_decref(opaque_tx);
opaque_tx = sentry__transaction_new(sentry__value_clone(tx));
TEST_CHECK(sentry_value_is_null(
sentry__value_get_trace_context(opaque_tx->inner)));
sentry_uuid_t span_id = sentry_uuid_new_v4();
sentry_value_set_by_key(
tx, "span_id", sentry__value_new_span_uuid(&span_id));
sentry__transaction_decref(opaque_tx);
opaque_tx = sentry__transaction_new(sentry__value_clone(tx));
sentry_value_t trace_context
= sentry__value_get_trace_context(opaque_tx->inner);
TEST_CHECK(!sentry_value_is_null(trace_context));
TEST_CHECK(!IS_NULL(trace_context, "trace_id"));
TEST_CHECK(!IS_NULL(trace_context, "span_id"));
CHECK_STRING_PROPERTY(trace_context, "op", "honk.beep");
sentry_value_decref(trace_context);
sentry_value_decref(tx);
sentry__transaction_decref(opaque_tx);
}
SENTRY_TEST(basic_transaction)
{
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new(NULL, NULL);
sentry_value_t tx_cxt;
if (opaque_tx_cxt != NULL) {
tx_cxt = opaque_tx_cxt->inner;
TEST_CHECK(!sentry_value_is_null(tx_cxt));
CHECK_STRING_PROPERTY(tx_cxt, "transaction", "");
CHECK_STRING_PROPERTY(tx_cxt, "op", "");
TEST_CHECK(!IS_NULL(tx_cxt, "trace_id"));
TEST_CHECK(!IS_NULL(tx_cxt, "span_id"));
} else {
TEST_CHECK(opaque_tx_cxt != NULL);
}
sentry__transaction_context_free(opaque_tx_cxt);
opaque_tx_cxt = sentry_transaction_context_new("", "");
if (opaque_tx_cxt != NULL) {
tx_cxt = opaque_tx_cxt->inner;
TEST_CHECK(!sentry_value_is_null(tx_cxt));
CHECK_STRING_PROPERTY(tx_cxt, "transaction", "");
CHECK_STRING_PROPERTY(tx_cxt, "op", "");
TEST_CHECK(!IS_NULL(tx_cxt, "trace_id"));
TEST_CHECK(!IS_NULL(tx_cxt, "span_id"));
} else {
TEST_CHECK(opaque_tx_cxt != NULL);
}
sentry__transaction_context_free(opaque_tx_cxt);
opaque_tx_cxt = sentry_transaction_context_new("honk.beep", "beepbeep");
if (opaque_tx_cxt != NULL) {
tx_cxt = opaque_tx_cxt->inner;
TEST_CHECK(!sentry_value_is_null(tx_cxt));
CHECK_STRING_PROPERTY(tx_cxt, "transaction", "honk.beep");
CHECK_STRING_PROPERTY(tx_cxt, "op", "beepbeep");
TEST_CHECK(!IS_NULL(tx_cxt, "trace_id"));
TEST_CHECK(!IS_NULL(tx_cxt, "span_id"));
sentry_transaction_context_set_name(opaque_tx_cxt, "");
CHECK_STRING_PROPERTY(tx_cxt, "transaction", "");
sentry_transaction_context_set_operation(opaque_tx_cxt, "");
CHECK_STRING_PROPERTY(tx_cxt, "op", "");
sentry_transaction_context_set_sampled(opaque_tx_cxt, 1);
TEST_CHECK(
sentry_value_is_true(sentry_value_get_by_key(tx_cxt, "sampled"))
== 1);
} else {
TEST_CHECK(opaque_tx_cxt != NULL);
}
sentry__transaction_context_free(opaque_tx_cxt);
}
static void
check_backfilled_name(sentry_envelope_t *envelope, void *data)
{
uint64_t *called = data;
*called += 1;
sentry_value_t tx = sentry_envelope_get_transaction(envelope);
TEST_CHECK(!sentry_value_is_null(tx));
CHECK_STRING_PROPERTY(tx, "transaction", "<unlabeled transaction>");
sentry_envelope_free(envelope);
}
SENTRY_TEST(transaction_name_backfill_on_finish)
{
uint64_t called = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport = sentry_transport_new(check_backfilled_name);
sentry_transport_set_state(transport, &called);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_init(options);
sentry_transaction_context_t *tx_cxt
= sentry_transaction_context_new(NULL, NULL);
sentry_transaction_t *tx
= sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_uuid_t event_id = sentry_transaction_finish(tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
tx_cxt = sentry_transaction_context_new("", "");
tx = sentry_transaction_start(tx_cxt, sentry_value_new_null());
event_id = sentry_transaction_finish(tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_close();
TEST_CHECK_INT_EQUAL(called, 2);
}
static void
send_transaction_envelope_test_basic(sentry_envelope_t *envelope, void *data)
{
uint64_t *called = data;
*called += 1;
sentry_value_t tx = sentry_envelope_get_transaction(envelope);
TEST_CHECK(!sentry_value_is_null(tx));
CHECK_STRING_PROPERTY(
tx, "event_id", "4c035723-8638-4c3a-923f-2ab9d08b4018");
if (*called != 1) {
CHECK_STRING_PROPERTY(tx, "type", "transaction");
CHECK_STRING_PROPERTY(tx, "transaction", "honk");
}
sentry_envelope_free(envelope);
}
SENTRY_TEST(basic_function_transport_transaction)
{
uint64_t called = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport
= sentry_transport_new(send_transaction_envelope_test_basic);
sentry_transport_set_state(transport, &called);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_require_user_consent(options, true);
sentry_init(options);
sentry_transaction_context_t *tx_cxt = sentry_transaction_context_new(
"How could you", "Don't capture this.");
sentry_transaction_t *tx
= sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_uuid_t event_id = sentry_transaction_finish(tx);
// TODO: `sentry_capture_event` acts as if the event was sent if user
// consent was not given
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_user_consent_give();
tx_cxt = sentry_transaction_context_new("honk", "beep");
tx = sentry_transaction_start(tx_cxt, sentry_value_new_null());
event_id = sentry_transaction_finish(tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_user_consent_revoke();
tx_cxt = sentry_transaction_context_new(
"How could you again", "Don't capture this either.");
tx = sentry_transaction_start(tx_cxt, sentry_value_new_null());
event_id = sentry_transaction_finish(tx);
// TODO: `sentry_capture_event` acts as if the event was sent if user
// consent was not given
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_close();
TEST_CHECK_INT_EQUAL(called, 1);
}
SENTRY_TEST(transport_sampling_transactions)
{
uint64_t called_transport = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport
= sentry_transport_new(send_transaction_envelope_test_basic);
sentry_transport_set_state(transport, &called_transport);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 0.75);
sentry_init(options);
uint64_t sent_transactions = 0;
for (int i = 0; i < 100; i++) {
sentry_transaction_context_t *tx_cxt
= sentry_transaction_context_new("honk", "beep");
sentry_transaction_t *tx
= sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_uuid_t event_id = sentry_transaction_finish(tx);
if (!sentry_uuid_is_nil(&event_id)) {
sent_transactions += 1;
}
}
sentry_close();
// exact value is nondeterministic because of rng
TEST_CHECK(called_transport > 50 && called_transport < 100);
TEST_CHECK(called_transport == sent_transactions);
}
static sentry_value_t
before_send(sentry_value_t event, void *UNUSED(hint), void *data)
{
uint64_t *called = data;
*called += 1;
sentry_value_decref(event);
return sentry_value_new_null();
}
SENTRY_TEST(transactions_skip_before_send)
{
uint64_t called_beforesend = 0;
uint64_t called_transport = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport
= sentry_transport_new(send_transaction_envelope_test_basic);
sentry_transport_set_state(transport, &called_transport);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_before_send(options, before_send, &called_beforesend);
sentry_init(options);
sentry_transaction_context_t *tx_cxt
= sentry_transaction_context_new("honk", "beep");
sentry_transaction_t *tx
= sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_uuid_t event_id = sentry_transaction_finish(tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_close();
TEST_CHECK_INT_EQUAL(called_transport, 1);
TEST_CHECK_INT_EQUAL(called_beforesend, 0);
}
static void
before_transport(sentry_envelope_t *envelope, void *data)
{
uint64_t *called = data;
*called += 1;
sentry_envelope_free(envelope);
}
SENTRY_TEST(multiple_transactions)
{
uint64_t called_transport = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport = sentry_transport_new(before_transport);
sentry_transport_set_state(transport, &called_transport);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_init(options);
sentry_transaction_context_t *tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *tx
= sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_set_transaction_object(tx);
sentry_value_t scope_tx = sentry__scope_get_span_or_transaction();
CHECK_STRING_PROPERTY(scope_tx, "transaction", "wow!");
sentry_uuid_t event_id = sentry_transaction_finish(tx);
scope_tx = sentry__scope_get_span_or_transaction();
TEST_CHECK(sentry_value_is_null(scope_tx));
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
// Set transaction on scope twice, back-to-back without finishing the first
// one
tx_cxt = sentry_transaction_context_new("whoa!", NULL);
tx = sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_set_transaction_object(tx);
sentry__transaction_decref(tx);
tx_cxt = sentry_transaction_context_new("wowee!", NULL);
tx = sentry_transaction_start(tx_cxt, sentry_value_new_null());
sentry_set_transaction_object(tx);
scope_tx = sentry__scope_get_span_or_transaction();
CHECK_STRING_PROPERTY(scope_tx, "transaction", "wowee!");
event_id = sentry_transaction_finish(tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
sentry_close();
TEST_CHECK_INT_EQUAL(called_transport, 2);
}
SENTRY_TEST(basic_spans)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_init(options);
// Starting a child with no active transaction should fail
sentry_span_t *parentless_child
= sentry_transaction_start_child(NULL, NULL, NULL);
TEST_CHECK(!parentless_child);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_value_t tx = opaque_tx->inner;
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
// Peek into the transaction's span list and make sure everything is
// good
const char *trace_id
= sentry_value_as_string(sentry_value_get_by_key(tx, "trace_id"));
const char *parent_span_id
= sentry_value_as_string(sentry_value_get_by_key(tx, "span_id"));
// Don't track the span yet
TEST_CHECK(IS_NULL(tx, "spans"));
// Sanity check that child isn't finished yet
TEST_CHECK(IS_NULL(child, "timestamp"));
// Now finishing
sentry_span_finish(opaque_child);
TEST_CHECK(!IS_NULL(tx, "spans"));
sentry_value_t spans = sentry_value_get_by_key(tx, "spans");
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry_value_t stored_child = sentry_value_get_by_index(spans, 0);
// Make sure the span inherited everything correctly
CHECK_STRING_PROPERTY(stored_child, "trace_id", trace_id);
CHECK_STRING_PROPERTY(stored_child, "parent_span_id", parent_span_id);
CHECK_STRING_PROPERTY(stored_child, "op", "honk");
CHECK_STRING_PROPERTY(stored_child, "description", "goose");
// Should be finished
TEST_CHECK(!IS_NULL(stored_child, "timestamp"));
sentry__transaction_decref(opaque_tx);
sentry_close();
}
SENTRY_TEST(spans_on_scope)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_init(options);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_set_transaction_object(opaque_tx);
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
// Peek into the transaction's span list and make sure everything is
// good
sentry_value_t scope_tx = sentry__scope_get_span_or_transaction();
const char *trace_id
= sentry_value_as_string(sentry_value_get_by_key(scope_tx, "trace_id"));
const char *parent_span_id
= sentry_value_as_string(sentry_value_get_by_key(scope_tx, "span_id"));
// Don't track the span yet
TEST_CHECK(IS_NULL(scope_tx, "spans"));
// Sanity check that child isn't finished yet
TEST_CHECK(IS_NULL(child, "timestamp"));
sentry_span_finish(opaque_child);
scope_tx = sentry__scope_get_span_or_transaction();
TEST_CHECK(!IS_NULL(scope_tx, "spans"));
sentry_value_t spans = sentry_value_get_by_key(scope_tx, "spans");
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry_value_t stored_child = sentry_value_get_by_index(spans, 0);
// Make sure the span inherited everything correctly
CHECK_STRING_PROPERTY(stored_child, "trace_id", trace_id);
CHECK_STRING_PROPERTY(stored_child, "parent_span_id", parent_span_id);
CHECK_STRING_PROPERTY(stored_child, "op", "honk");
CHECK_STRING_PROPERTY(stored_child, "description", "goose");
// Should be finished
TEST_CHECK(!IS_NULL(stored_child, "timestamp"));
sentry__transaction_decref(opaque_tx);
sentry_close();
}
SENTRY_TEST(child_spans)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_max_spans(options, 3);
sentry_init(options);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_value_t tx = opaque_tx->inner;
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
// Shouldn't be added to spans yet
TEST_CHECK(IS_NULL(tx, "spans"));
sentry_span_t *opaque_grandchild
= sentry_span_start_child(opaque_child, "beep", "car");
sentry_value_t grandchild = opaque_grandchild->inner;
TEST_CHECK(!sentry_value_is_null(grandchild));
// Shouldn't be added to spans yet
TEST_CHECK(IS_NULL(tx, "spans"));
sentry_span_finish(opaque_grandchild);
// Make sure everything on the transaction looks good, check grandchild
const char *trace_id
= sentry_value_as_string(sentry_value_get_by_key(tx, "trace_id"));
const char *parent_span_id
= sentry_value_as_string(sentry_value_get_by_key(child, "span_id"));
TEST_CHECK(!IS_NULL(tx, "spans"));
sentry_value_t spans = sentry_value_get_by_key(tx, "spans");
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry_value_t stored_grandchild = sentry_value_get_by_index(spans, 0);
CHECK_STRING_PROPERTY(stored_grandchild, "trace_id", trace_id);
CHECK_STRING_PROPERTY(stored_grandchild, "parent_span_id", parent_span_id);
CHECK_STRING_PROPERTY(stored_grandchild, "op", "beep");
CHECK_STRING_PROPERTY(stored_grandchild, "description", "car");
// Should be finished
TEST_CHECK(!IS_NULL(stored_grandchild, "timestamp"));
sentry_span_finish(opaque_child);
spans = sentry_value_get_by_key(tx, "spans");
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 2);
sentry__transaction_decref(opaque_tx);
sentry_close();
}
SENTRY_TEST(overflow_spans)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_max_spans(options, 1);
sentry_init(options);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_value_t tx = opaque_tx->inner;
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
const char *child_span_id
= sentry_value_as_string(sentry_value_get_by_key(child, "span_id"));
// Shouldn't be added to spans yet
TEST_CHECK(IS_NULL(tx, "spans"));
sentry_span_t *opaque_drop_on_finish_child
= sentry_span_start_child(opaque_child, "beep", "car");
sentry_value_t drop_on_finish_child = opaque_drop_on_finish_child->inner;
TEST_CHECK(!sentry_value_is_null(drop_on_finish_child));
// Shouldn't be added to spans yet
TEST_CHECK(IS_NULL(tx, "spans"));
sentry_span_finish(opaque_child);
TEST_CHECK(!IS_NULL(tx, "spans"));
sentry_value_t spans = sentry_value_get_by_key(tx, "spans");
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry_value_t stored_child = sentry_value_get_by_index(spans, 0);
CHECK_STRING_PROPERTY(stored_child, "span_id", child_span_id);
sentry_span_finish(opaque_drop_on_finish_child);
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry_span_t *opaque_drop_on_start_child
= sentry_transaction_start_child(opaque_tx, "ring", "bicycle");
TEST_CHECK(!opaque_drop_on_start_child);
TEST_CHECK_INT_EQUAL(sentry_value_get_length(spans), 1);
sentry__transaction_decref(opaque_tx);
sentry_close();
}
SENTRY_TEST(unsampled_spans)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_init(options);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("noisemakers", NULL);
sentry_transaction_context_set_sampled(opaque_tx_cxt, 0);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_value_t tx = opaque_tx->inner;
TEST_CHECK(!sentry_value_is_true(sentry_value_get_by_key(tx, "sampled")));
// check that children and grandchildren inherit the sampling decision,
// i.e. it cascades 1+ levels down
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
TEST_CHECK(
!sentry_value_is_true(sentry_value_get_by_key(child, "sampled")));
sentry_span_t *opaque_grandchild
= sentry_span_start_child(opaque_child, "beep", "car");
sentry_value_t grandchild = opaque_grandchild->inner;
TEST_CHECK(!sentry_value_is_null(grandchild));
TEST_CHECK(
!sentry_value_is_true(sentry_value_get_by_key(grandchild, "sampled")));
// finishing does not add (grand)children to the spans list
sentry_span_finish(opaque_grandchild);
TEST_CHECK(
0 == sentry_value_get_length(sentry_value_get_by_key(tx, "spans")));
sentry_span_finish(opaque_child);
TEST_CHECK(
0 == sentry_value_get_length(sentry_value_get_by_key(tx, "spans")));
// perform the same checks, but with the transaction on the scope
sentry_set_transaction_object(opaque_tx);
opaque_child = sentry_transaction_start_child(opaque_tx, "toot", "boat");
child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
TEST_CHECK(
!sentry_value_is_true(sentry_value_get_by_key(child, "sampled")));
opaque_grandchild
= sentry_span_start_child(opaque_child, "vroom", "sportscar");
grandchild = opaque_grandchild->inner;
TEST_CHECK(!sentry_value_is_null(grandchild));
TEST_CHECK(
!sentry_value_is_true(sentry_value_get_by_key(grandchild, "sampled")));
sentry_span_finish(opaque_grandchild);
TEST_CHECK(
0 == sentry_value_get_length(sentry_value_get_by_key(tx, "spans")));
sentry_span_finish(opaque_child);
TEST_CHECK(
0 == sentry_value_get_length(sentry_value_get_by_key(tx, "spans")));
sentry_transaction_finish(opaque_tx);
sentry_close();
}
static void
check_spans(sentry_envelope_t *envelope, void *data)
{
uint64_t *called = data;
*called += 1;
sentry_value_t transaction = sentry_envelope_get_transaction(envelope);
TEST_CHECK(!sentry_value_is_null(transaction));
size_t span_count = sentry_value_get_length(
sentry_value_get_by_key(transaction, "spans"));
TEST_CHECK_INT_EQUAL(span_count, 1);
sentry_envelope_free(envelope);
}
SENTRY_TEST(drop_unfinished_spans)
{
uint64_t called_transport = 0;
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_transport_t *transport = sentry_transport_new(check_spans);
sentry_transport_set_state(transport, &called_transport);
sentry_options_set_transport(options, transport);
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_max_spans(options, 2);
sentry_init(options);
sentry_transaction_context_t *opaque_tx_cxt
= sentry_transaction_context_new("wow!", NULL);
sentry_transaction_t *opaque_tx
= sentry_transaction_start(opaque_tx_cxt, sentry_value_new_null());
sentry_value_t tx = opaque_tx->inner;
sentry_span_t *opaque_child
= sentry_transaction_start_child(opaque_tx, "honk", "goose");
sentry_value_t child = opaque_child->inner;
TEST_CHECK(!sentry_value_is_null(child));
sentry_span_t *opaque_grandchild
= sentry_span_start_child(opaque_child, "beep", "car");
sentry_value_t grandchild = opaque_grandchild->inner;
TEST_CHECK(!sentry_value_is_null(grandchild));
sentry_span_finish(opaque_grandchild);
// spans are only added to transactions upon completion
TEST_CHECK_INT_EQUAL(
sentry_value_get_length(sentry_value_get_by_key(tx, "spans")), 1);
sentry_uuid_t event_id = sentry_transaction_finish(opaque_tx);
TEST_CHECK(!sentry_uuid_is_nil(&event_id));
// check that nothing explodes if you do finish the lingering child
sentry_span_finish(opaque_child);
sentry_close();
TEST_CHECK_INT_EQUAL(called_transport, 1);
}
static void
forward_headers_to(const char *key, const char *value, void *userdata)
{
sentry_transaction_context_t *tx_ctx
= (sentry_transaction_context_t *)userdata;
sentry_transaction_context_update_from_header(tx_ctx, key, value);
}
SENTRY_TEST(distributed_headers)
{
sentry_options_t *options = sentry_options_new();
sentry_options_set_dsn(options, "https://foo@sentry.invalid/42");
sentry_options_set_traces_sample_rate(options, 1.0);
sentry_options_set_max_spans(options, 2);
sentry_init(options);
const char *trace_header
= "2674eb52d5874b13b560236d6c79ce8a-a0f9fdf04f1a63df-1";
const char *not_expected_header
= "00000000000000000000000000000000-0000000000000000-1";
const char *expected_trace_id = "2674eb52d5874b13b560236d6c79ce8a";
sentry_transaction_context_t *tx_ctx
= sentry_transaction_context_new("wow!", NULL);
// check case insensitive headers, and bogus header names
sentry_transaction_context_update_from_header(
tx_ctx, "SeNtry-TrAcE", trace_header);
sentry_transaction_context_update_from_header(
tx_ctx, "nop", not_expected_header);
sentry_transaction_context_update_from_header(
tx_ctx, "sentry-trace-but-a-lot-longer", not_expected_header);
sentry_transaction_t *tx
= sentry_transaction_start(tx_ctx, sentry_value_new_null());
const char *trace_id = sentry_value_as_string(
sentry_value_get_by_key(tx->inner, "trace_id"));
TEST_CHECK_STRING_EQUAL(trace_id, expected_trace_id);
const char *span_id
= sentry_value_as_string(sentry_value_get_by_key(tx->inner, "span_id"));
TEST_CHECK(!sentry__string_eq(span_id, ""));
// check transaction
tx_ctx = sentry_transaction_context_new("distributed!", NULL);
sentry_transaction_iter_headers(tx, forward_headers_to, (void *)tx_ctx);
sentry_transaction_t *dist_tx
= sentry_transaction_start(tx_ctx, sentry_value_new_null());
const char *dist_trace_id = sentry_value_as_string(
sentry_value_get_by_key(dist_tx->inner, "trace_id"));
TEST_CHECK_STRING_EQUAL(dist_trace_id, trace_id);
const char *parent_span_id = sentry_value_as_string(
sentry_value_get_by_key(dist_tx->inner, "parent_span_id"));
TEST_CHECK_STRING_EQUAL(parent_span_id, span_id);
sentry__transaction_decref(dist_tx);
// check span
sentry_span_t *child = sentry_transaction_start_child(tx, "honk", "goose");
span_id = sentry_value_as_string(
sentry_value_get_by_key(child->inner, "span_id"));
TEST_CHECK(!sentry__string_eq(span_id, ""));
tx_ctx = sentry_transaction_context_new("distributed!", NULL);
sentry_span_iter_headers(child, forward_headers_to, (void *)tx_ctx);
dist_tx = sentry_transaction_start(tx_ctx, sentry_value_new_null());
dist_trace_id = sentry_value_as_string(
sentry_value_get_by_key(dist_tx->inner, "trace_id"));
TEST_CHECK_STRING_EQUAL(dist_trace_id, trace_id);
parent_span_id = sentry_value_as_string(
sentry_value_get_by_key(dist_tx->inner, "parent_span_id"));
TEST_CHECK_STRING_EQUAL(parent_span_id, span_id);
TEST_CHECK(sentry_value_is_true(
sentry_value_get_by_key(dist_tx->inner, "sampled")));
sentry__transaction_decref(dist_tx);
sentry__span_decref(child);
sentry__transaction_decref(tx);
// check sampled flag
tx_ctx = sentry_transaction_context_new("wow!", NULL);
sentry_transaction_context_set_sampled(tx_ctx, 0);
tx = sentry_transaction_start(tx_ctx, sentry_value_new_null());
tx_ctx = sentry_transaction_context_new("distributed!", NULL);
sentry_transaction_iter_headers(tx, forward_headers_to, (void *)tx_ctx);
dist_tx = sentry_transaction_start(tx_ctx, sentry_value_new_null());
TEST_CHECK(!sentry_value_is_true(
sentry_value_get_by_key(dist_tx->inner, "sampled")));
child = sentry_transaction_start_child(tx, "honk", "goose");
TEST_CHECK(!sentry_value_is_true(
sentry_value_get_by_key(child->inner, "sampled")));
tx_ctx = sentry_transaction_context_new("distributed from a child!", NULL);
sentry_span_iter_headers(child, forward_headers_to, (void *)tx_ctx);
sentry__transaction_decref(dist_tx);
dist_tx = sentry_transaction_start(tx_ctx, sentry_value_new_null());
TEST_CHECK(!sentry_value_is_true(
sentry_value_get_by_key(dist_tx->inner, "sampled")));
sentry__transaction_decref(dist_tx);
sentry__span_decref(child);
sentry__transaction_decref(tx);
sentry_close();
}
#undef IS_NULL
#undef CHECK_STRING_PROPERTY