Skip to content

Commit

Permalink
add user feedback
Browse files Browse the repository at this point in the history
  • Loading branch information
tustanivsky committed Mar 14, 2024
1 parent 2ae5de6 commit 6a1e382
Show file tree
Hide file tree
Showing 10 changed files with 226 additions and 0 deletions.
10 changes: 10 additions & 0 deletions examples/example.c
Original file line number Diff line number Diff line change
Expand Up @@ -365,6 +365,16 @@ main(int argc, char **argv)

sentry_capture_event(event);
}
if (has_arg(argc, argv, "capture-user-feedback")) {
sentry_value_t event = sentry_value_new_message_event(
SENTRY_LEVEL_INFO, "my-logger", "Hello World!");
sentry_uuid_t event_id = sentry_capture_event(event);

sentry_value_t user_feedback = sentry_value_new_user_feedback(&event_id,
"some-name", "some-email", "some-comment");

sentry_capture_user_feedback(user_feedback);
}

if (has_arg(argc, argv, "capture-transaction")) {
sentry_transaction_context_t *tx_ctx
Expand Down
21 changes: 21 additions & 0 deletions include/sentry.h
Original file line number Diff line number Diff line change
Expand Up @@ -1888,6 +1888,27 @@ SENTRY_EXPERIMENTAL_API void sentry_transaction_set_name(
SENTRY_EXPERIMENTAL_API void sentry_transaction_set_name_n(
sentry_transaction_t *transaction, const char *name, size_t name_len);

/**
* Creates a new User Feedback with a specific name, email and comments.
*
* See https://develop.sentry.dev/sdk/envelopes/#user-feedback
*
* User Feedback has to be associated with a specific event that has been
* sent to Sentry earlier.
*/
SENTRY_API sentry_value_t sentry_value_new_user_feedback(
const sentry_uuid_t *uuid,
const char *name, const char *email, const char *comments);
SENTRY_API sentry_value_t sentry_value_new_user_feedback_n(
const sentry_uuid_t *uuid,
const char *name, size_t name_len, const char *email, size_t email_len,
const char *comments, size_t comments_len);

/**
* Captures a manually created User Feedback and sends it to Sentry.
*/
SENTRY_API void sentry_capture_user_feedback(sentry_value_t user_feedback);

/**
* The status of a Span or Transaction.
*
Expand Down
35 changes: 35 additions & 0 deletions src/sentry_core.c
Original file line number Diff line number Diff line change
Expand Up @@ -550,6 +550,24 @@ sentry__prepare_transaction(const sentry_options_t *options,
return NULL;
}

sentry_envelope_t *
sentry__prepare_user_feedback(sentry_value_t user_feedback)
{
sentry_envelope_t *envelope = NULL;

envelope = sentry__envelope_new();
if (!envelope || !sentry__envelope_add_user_feedback(envelope, user_feedback)) {
goto fail;
}

return envelope;

fail:
SENTRY_WARN("dropping user feedback");
sentry_envelope_free(envelope);
return NULL;
}

void
sentry_handle_exception(const sentry_ucontext_t *uctx)
{
Expand Down Expand Up @@ -1120,6 +1138,23 @@ sentry_span_finish(sentry_span_t *opaque_span)
sentry__span_decref(opaque_span);
}

void sentry_capture_user_feedback(sentry_value_t user_feedback)
{
sentry_envelope_t *envelope = NULL;

bool was_captured = false;
SENTRY_WITH_OPTIONS (options) {
was_captured = true;
envelope = sentry__prepare_user_feedback(user_feedback);
if (envelope) {
sentry__capture_envelope(options->transport, envelope);
}
}
if (!was_captured) {
sentry_value_decref(user_feedback);
}
}

int
sentry_get_crashed_last_run(void)
{
Expand Down
30 changes: 30 additions & 0 deletions src/sentry_envelope.c
Original file line number Diff line number Diff line change
Expand Up @@ -295,6 +295,36 @@ sentry__envelope_add_transaction(
return item;
}

sentry_envelope_item_t *
sentry__envelope_add_user_feedback(
sentry_envelope_t *envelope, sentry_value_t user_feedback)
{
sentry_envelope_item_t *item = envelope_add_item(envelope);
if (!item) {
return NULL;
}

sentry_jsonwriter_t *jw = sentry__jsonwriter_new(NULL);
if (!jw) {
return NULL;
}

sentry_value_t event_id = sentry__ensure_event_id(user_feedback, NULL);

sentry__jsonwriter_write_value(jw, user_feedback);
item->payload = sentry__jsonwriter_into_string(jw, &item->payload_len);

sentry__envelope_item_set_header(
item, "type", sentry_value_new_string("user_report"));
sentry_value_t length = sentry_value_new_int32((int32_t)item->payload_len);
sentry__envelope_item_set_header(item, "length", length);

sentry_value_incref(event_id);
sentry__envelope_set_header(envelope, "event_id", event_id);

return item;
}

sentry_envelope_item_t *
sentry__envelope_add_session(
sentry_envelope_t *envelope, const sentry_session_t *session)
Expand Down
6 changes: 6 additions & 0 deletions src/sentry_envelope.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,12 @@ sentry_envelope_item_t *sentry__envelope_add_event(
sentry_envelope_item_t *sentry__envelope_add_transaction(
sentry_envelope_t *envelope, sentry_value_t transaction);

/**
* Add a user feedback to this envelope.
*/
sentry_envelope_item_t *sentry__envelope_add_user_feedback(
sentry_envelope_t *envelope, sentry_value_t user_feedback);

/**
* Add a session to this envelope.
*/
Expand Down
33 changes: 33 additions & 0 deletions src/sentry_value.c
Original file line number Diff line number Diff line change
Expand Up @@ -1265,6 +1265,39 @@ sentry_value_new_stacktrace(void **ips, size_t len)
return stacktrace;
}

sentry_value_t sentry_value_new_user_feedback(const sentry_uuid_t *uuid,
const char *name, const char *email, const char *comments)
{
size_t name_len = name ? strlen(name) : 0;
size_t email_len = email ? strlen(email) : 0;
size_t comments_len = email ? strlen(comments) : 0;
return sentry_value_new_user_feedback_n(uuid, name, name_len,
email, email_len, comments, comments_len);
}
sentry_value_t sentry_value_new_user_feedback_n(const sentry_uuid_t *uuid,
const char *name, size_t name_len, const char *email, size_t email_len,
const char *comments, size_t comments_len)
{
sentry_value_t rv = sentry_value_new_object();

sentry_value_set_by_key(rv, "event_id", sentry__value_new_uuid(uuid));

if (name) {
sentry_value_set_by_key(rv, "name",
sentry_value_new_string_n(name, name_len));
}
if (email) {
sentry_value_set_by_key(rv, "email",
sentry_value_new_string_n(email, email_len));
}
if (comments) {
sentry_value_set_by_key(rv, "comments",
sentry_value_new_string_n(comments, comments_len));
}

return rv;
}

static sentry_value_t
sentry__get_or_insert_values_list(sentry_value_t parent, const char *key)
{
Expand Down
13 changes: 13 additions & 0 deletions tests/assertions.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,19 @@ def assert_session(envelope, extra_assertion=None):
if extra_assertion:
assert_matches(session, extra_assertion)

def assert_user_feedback(envelope, extra_assertion=None):
user_feedback = None
for item in envelope:
if item.headers.get("type") == "user_report" and item.payload.json is not None:
user_feedback = item.payload.json

assert user_feedback is not None
assert user_feedback["name"] == "some-name"
assert user_feedback["email"] == "some-email"
assert user_feedback["comments"] == "some-comment"
if extra_assertion:
assert_matches(user_feedback, extra_assertion)


def assert_meta(
envelope,
Expand Down
29 changes: 29 additions & 0 deletions tests/test_integration_http.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
assert_exception,
assert_inproc_crash,
assert_session,
assert_user_feedback,
assert_minidump,
assert_breakpad_crash,
)
Expand Down Expand Up @@ -116,6 +117,34 @@ def test_capture_and_session_http(cmake, httpserver):
envelope = Envelope.deserialize(output)
assert_session(envelope, {"status": "exited", "errors": 0})

def test_user_feedback_http(cmake, httpserver):
tmp_path = cmake(["sentry_example"], {"SENTRY_BACKEND": "none"})

httpserver.expect_request(
"/api/123456/envelope/",
headers={"x-sentry-auth": auth_header},
).respond_with_data("OK")
env = dict(os.environ, SENTRY_DSN=make_dsn(httpserver))

run(
tmp_path,
"sentry_example",
["log", "capture-user-feedback"],
check=True,
env=env,
)

assert len(httpserver.log) == 2
output = httpserver.log[0][0].get_data()
envelope = Envelope.deserialize(output)

assert_event(envelope)

output = httpserver.log[1][0].get_data()
envelope = Envelope.deserialize(output)

assert_user_feedback(envelope)


def test_exception_and_session_http(cmake, httpserver):
tmp_path = cmake(["sentry_example"], {"SENTRY_BACKEND": "none"})
Expand Down
28 changes: 28 additions & 0 deletions tests/unit/test_envelopes.c
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,34 @@ SENTRY_TEST(basic_http_request_preparation_for_transaction)
sentry__dsn_decref(dsn);
}

SENTRY_TEST(basic_http_request_preparation_for_user_feedback)
{
sentry_dsn_t *dsn = sentry__dsn_new("https://[email protected]/42");

sentry_uuid_t event_id
= sentry_uuid_from_string("c993afb6-b4ac-48a6-b61b-2558e601d65d");
sentry_envelope_t *envelope = sentry__envelope_new();
sentry_value_t user_feedback = sentry_value_new_user_feedback(
&event_id, "some-name", "some-email", "some-comment");
sentry__envelope_add_user_feedback(envelope, user_feedback);

sentry_prepared_http_request_t *req
= sentry__prepare_http_request(envelope, dsn, NULL, NULL);
TEST_CHECK_STRING_EQUAL(req->method, "POST");
TEST_CHECK_STRING_EQUAL(
req->url, "https://sentry.invalid:443/api/42/envelope/");
TEST_CHECK_STRING_EQUAL(req->body,
"{\"event_id\":\"c993afb6-b4ac-48a6-b61b-2558e601d65d\"}\n"
"{\"type\":\"user_report\",\"length\":117}\n"
"{\"event_id\":\"c993afb6-b4ac-48a6-b61b-2558e601d65d\",\"email\":"
"\"some-email\",\"comments\":\"some-comment\",\"name\":"
"\"some-name\"}");
sentry__prepared_http_request_free(req);
sentry_envelope_free(envelope);

sentry__dsn_decref(dsn);
}

SENTRY_TEST(basic_http_request_preparation_for_event_with_attachment)
{
sentry_dsn_t *dsn = sentry__dsn_new("https://[email protected]/42");
Expand Down
21 changes: 21 additions & 0 deletions tests/unit/test_value.c
Original file line number Diff line number Diff line change
Expand Up @@ -772,3 +772,24 @@ SENTRY_TEST(thread_without_name_still_valid)
test_name);
sentry_value_decref(thread);
}

SENTRY_TEST(user_feedback_is_valid)
{
sentry_uuid_t event_id
= sentry_uuid_from_string("c993afb6-b4ac-48a6-b61b-2558e601d65d");
sentry_value_t user_feedback = sentry_value_new_user_feedback(
&event_id, "some-name", "some-email", "some-comment");

TEST_CHECK(!sentry_value_is_null(user_feedback));
TEST_CHECK_STRING_EQUAL(sentry_value_as_string(sentry_value_get_by_key(
user_feedback, "name")),
"some-name");
TEST_CHECK_STRING_EQUAL(sentry_value_as_string(sentry_value_get_by_key(
user_feedback, "email")),
"some-email");
TEST_CHECK_STRING_EQUAL(sentry_value_as_string(sentry_value_get_by_key(
user_feedback, "comments")),
"some-comment");

sentry_value_decref(user_feedback);
}

0 comments on commit 6a1e382

Please sign in to comment.