#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", ""); 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_free(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_free(child); sentry__transaction_decref(tx); sentry_close(); } #undef IS_NULL #undef CHECK_STRING_PROPERTY