812 lines
29 KiB
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_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
|