Hub Based API

Motivation

Sentry has a wide range of SDKs that have been developed over the years by different developers and based on different ideas. This has lead to the situation that the feature sets across the SDKs are different, use different concepts and terms which has lead to the situation that it's often not clear how to achieve the same thing on different platforms.

Additionally those SDKs were purely centered around error reporting through explicit clients which meant that certain integrations (such as breadcrumbs) were often not possible.

General Guidelines

  • We want a unified language/wording of all SDK APIs to aid support and documentation as well as making it easier for users to use Sentry in different environments.

  • Design the SDK in a way where we can trivially add new features later that go past pure event reporting (transactions, APM etc.).

  • Design the SDK that with the same client instance we can both work naturally in the runtime environment through dependency injection etc. as well as using an implied context dispatching to already existing clients and scopes to hook into most environments. This is important because it allows events to include data from other integrations in the process.

  • The common tasks need to be easy and obvious.

  • For helping 3rd party libraries the case of “non configured Sentry” needs to be fast (and lazily executed).

  • The common API needs make sense in most languages and must not depend on super special constructs. To make it feel more natural we should consider language specifics and explicitly support them as alternatives (disposables, stack guards etc.).

Simplified Visualization

Unified API Visualized

Terminology

  • minimal: A separate "facade" package that re-exports a subset of the SDK's functionality through interfaces or proxies. That package does not directly depend on the SDK, instead it should make every operation a noop if the SDK is not installed.

    The purpose of such a package is to allow random libraries to record breadcrumbs and set context data while not having a hard dependency on the SDK.

  • hub: An object that manages the state. An implied global thread local or similar hub exists that can be used by default. Hubs can be created manually.

  • scope: A scope holds data that should implicitly be sent with Sentry events. It can hold context data, extra parameters, level overrides, fingerprints etc.

  • client: A client is an object that is configured once and can be bound to the hub. The user can then auto discover the client and dispatch calls to it. Users typically do not need to work with the client directly. They either do it via the hub or static convenience functions. The client is mostly responsible for building Sentry events and dispatching those to the transport.

  • client options: Are parameters that are language and runtime specific and used to configure the client. This can be release and environment but also things like which integrations to configure, how in-app works etc.

  • context: Contexts give extra data to Sentry. There are the special contexts (user and similar) and the generic ones (runtime, os, device), etc. Check out Contexts for some predefined keys - users can also add arbitrary context keys. Note: In older SDKs, you might encounter an unrelated concept of context, which is now deprecated by scopes

  • tags: Tags can be arbitrary string→string pairs by which events can be searched. Contexts are converted into tags.

  • extra: Truly arbitrary data attached by client users. Extra should be avoided where possible, and instead structured context or tags should be used - as these can be queried and visualized in a better way.

  • transport: The transport is an internal construct of the client that abstracts away the event sending. Typically the transport runs in a separate thread and gets events to send via a queue. The transport is responsible for sending, retrying and handling rate limits. The transport might also persist unsent events across restarts if needed.

  • integration: Code that provides middlewares, bindings or hooks into certain frameworks or environments, along with code that inserts those bindings and activates them. Usage for integrations does not follow a common interface.

  • event processors: Callbacks that run for every event.
    They can either modify and return the event, or null. Returning null will discard the event and not process further.

    See Event Pipeline for more information.

  • disabled SDK: Most of the SDK functionality depends on a configured and active client. Sentry considers the client active when it has a transport. Otherwise, the client is inactive, and the SDK is considered "disabled". In this case, certain callbacks, such as configure_scope or event processors, may not be invoked. As a result, breadcrumbs are not recorded.

"Static API"

The static API functions is the most common user facing API. A user just imports these functions and can start emitting events to Sentry or configuring scopes. These shortcut functions should be exported in the top-level namespace of your package. Behind the scenes they use hubs and scopes (see Concurrency for more information) if available on that platform. Note that all listed functions below are mostly aliases for Hub::get_current().function.

  • init(options): This is the entry point for every SDK.

This typically creates / reinitializes the global hub which is propagated to all new threads/execution contexts, or a hub is created per thread/execution context.

Takes options (dsn etc.), configures a client and binds it to the current hub or initializes it. Should return a stand-in that can be used to drain events (a disposable).

This might return a handle or guard for disposing. How this is implemented is entirely up to the SDK. This might even be a client if that’s something that makes sense for the SDK. In Rust it’s a ClientInitGuard, in JavaScript it could be a helper object with a close method that is awaitable.

You should be able to call this multiple times where calling it a second time either tears down the previous client or decrements a refcount for the previous client etc.

Calling this multiple times should be used for testing only.
It’s undefined what happens if you call init on anything but application startup.

A user has to call init once but it’s permissible to call this with a disabled DSN of sorts. Might for instance be no parameter passed etc.

Additionally it also sets up all default integrations.

  • capture_event(event): Takes an already assembled event and dispatches it to the currently active hub. The event object can be a plain dictionary or a typed object whatever makes more sense in the SDK. It should follow the native protocol as close as possible ignoring platform specific renames (case styles etc.).

  • capture_exception(error): Report an error or exception object. Depending on the platform different parameters are possible. The most obvious version accepts just an error object but also variations are possible where no error is passed and the current exception is used.

  • capture_message(message, level): Reports a message. The level can be optional in language with default parameters in which case it should default to info.

  • add_breadcrumb(crumb): Adds a new breadcrumb to the scope. If the total number of breadcrumbs exceeds the max_breadcrumbs setting, the SDK should remove the oldest breadcrumb. This works like the Hub API with regards to what crumb can be. If the SDK is disabled, it should ignore the breadcrumb.

  • configure_scope(callback): Calls a callback with a scope object that can be reconfigured. This is used to attach contextual data for future events in the same scope.

  • last_event_id(): Should return the last event ID emitted by the current scope. This is for instance used to implement user feedback dialogs.

  • start_session(): Stores a session on the current scope and starts tracking it. This normally attaches a brand new session to the scope, and implicitly ends any already existing session.

  • end_session(): Ends the session, setting an appropriate status and duration, and enqueues it for sending to Sentry.

Concurrency

All SDKs should have the concept of concurrency safe context storage. What this means depends on the language. The basic idea is that a user of the SDK can call a method to safely provide additional context information for all events that are about to be recorded.

This is implemented as a thread local stack in most languages, but in some (such as JavaScript) it might be global under the assumption that this is something that makes sense in the environment.

Here are some common concurrency patterns:

  • Thread bound hub: In that pattern each thread gets its own "hub" which internally manages a stack of scopes. If that pattern is followed one thread (the one that calls init()) becomes the "main" hub which is used as the base for newly spawned threads which will get a hub that is based on the main hub (but otherwise independent).

  • Internally scoped hub: On some platforms such as .NET ambient data is available in which case the Hub can internally manage the scopes.

  • Dummy hub: On some platforms concurrency just doesn't inherently exist. In that case the hub might be entirely absent or just be a singleton without concurrency management.

Hub

Under normal circumstances the hub consists of a stack of clients and scopes.

The SDK maintains two variables: The main hub (a global variable) and the current hub (a variable local to the current thread or execution context, also sometimes known as async local or context local)

  • Hub::new(client, scope): Creates a new hub with the given client and scope. The client can be reused between hubs. The scope should be owned by the hub (make a clone if necessary)

  • Hub::new_from_top(hub) / alternatively native constructor overloads: Creates a new hub by cloning the top stack of another hub.

  • get_current_hub() / Hub::current() / Hub::get_current(): Global function or static function to return the current (thread's) hub

  • get_main_hub() / Hub::main() / Hub::get_main(): In languages where the main thread is special ("Thread bound hub" model) this returns the main thread’s hub instead of the current thread’s hub. This might not exist in all languages.

  • Hub::capture_event / Hub::capture_message / Hub::capture_exception Capture message / exception call into capture event. capture_event merges the event passed with the scope data and dispatches to the client. As an additional argument it also takes a Hint.

    For the hint parameter see hints.

  • Hub::push_scope(): Pushes a new scope layer that inherits the previous data. This should return a disposable or stack guard for languages where it makes sense. When the "internally scoped hub" concurrency model is used calls to this are often necessary as otherwise a scope might be accidentally incorrectly shared.

  • Hub::with_scope(callback) (optional): In Python this could be a context manager, in Ruby a block function. Pushes and pops a scope for integration work.

  • Hub::pop_scope() (optional): Only exists in languages without better resource management. Better to have this function on a return value of push_scope or to use with_scope. This is also sometimes called pop_scope_unsafe to indicate that this method should not be used directly.

  • Hub::configure_scope(callback): Invokes the callback with a mutable reference to the scope for modifications. This can also be a with statement in languages that have it (Python). If no active client is bound to this hub, the SDK should not invoke the callback.

  • Hub::add_breadcrumb(crumb, hint): Adds a breadcrumb to the current scope.

    • The argument supported should be:
      • function that creates a breadcrumb
      • an already created breadcrumb object
      • a list of breadcrumbs (optional)
    • In languages where we do not have a basic form of overloading only a raw breadcrumb object should be accepted.

    The SDK should ignore the breadcrumb if no active client is bound to this hub.

    For the hint parameter see hints.

  • Hub::client() / Hub::get_client() (optional): Accessor or getter that returns the current client or None.

  • Hub::bind_client(new_client): Binds a different client to the hub. If the hub is also the owner of the client that was created by init it needs to keep a reference to it still if the hub is the object responsible for disposing it.

  • Hub::unbind_client() (optional): Optional way to unbind for languages where bind_client does not accept nullables.

  • Hub::last_event_id(): Should return the last event ID emitted by the current scope. This is for instance used to implement user feedback dialogs.

  • Hub::run(hub, callback) hub.run(callback), run_in_hub(hub, callback) (optional): Runs a callback with the hub bound as the current hub.

Scope

A scope holds data that should implicitly be sent with Sentry events. It can hold context data, extra parameters, level overrides, fingerprints etc.

The user can modify the current scope (to set extra, tags, current user) through the global function configure_scope. configure_scope takes a callback function to which it passes the current scope.

The reason for this callback-based API is efficiency. If the SDK is disabled, it should not invoke the callback, thus avoiding unnecessary work.

Copied
Sentry.configureScope(scope =>
  scope.setExtra("character_name", "Mighty Fighter"));
  • scope.set_user(user): Shallow merges user configuration (email, username, …). Removing user data is SDK-defined, either with a remove_user function or by passing nothing as data.

  • scope.set_extra(key, value): Sets the extra key to an arbitrary value, overwriting a potential previous value. Removing a key is SDK-defined, either with a remove_extra function or by passing nothing as data. This is deprecated functionality and users should be encouraged to use contexts instead.

  • scope.set_extras(extras): Sets an object with key/value pairs, convenience function instead of multiple set_extra calls. As with set_extra this is considered deprecated functionality.

  • scope.set_tag(key, value): Sets the tag to a string value, overwriting a potential previous value. Removing a key is SDK-defined, either with a remove_tag function or by passing nothing as data.

  • scope.set_tags(tags): Sets an object with key/value pairs, convenience function instead of multiple set_tag calls.

  • scope.set_context(key, value): Sets the context key to a value, overwriting a potential previous value. Removing a key is SDK-defined, either with a remove_context function or by passing nothing as data. The types are sdk specified.

  • scope.set_level(level): Sets the level of all events sent within this scope.

  • scope.set_transaction(transaction_name): Sets the name of the current transaction.

  • scope.set_fingerprint(fingerprint[]): Sets the fingerprint to group specific events together

  • scope.add_event_processor(processor): Registers an event processor function. It takes an event and returns a new event or None to drop it. This is the basis of many integrations.

  • scope.add_error_processor(processor) (optional): Registers an error processor function. It takes an event and exception object and returns a new event or None to drop it. This can be used to extract additional information out of an exception object that the SDK cannot extract itself.

  • scope.clear(): Resets a scope to default values while keeping all registered event processors. This does not affect either child or parent scopes.

  • scope.add_breadcrumb(breadcrumb): Adds a breadcrumb to the current scope.

  • scope.clear_breadcrumbs(): Deletes current breadcrumbs from the scope.

  • scope.apply_to_event(event[, max_breadcrumbs]): Applies the scope data to the given event object. This also applies the event processors stored in the scope internally. Some implementations might want to set a max breadcrumbs count here.

Client

A Client is the part of the SDK that is responsible for event creation. To give an example, the Client should convert an exception to a Sentry event. The Client should be stateless, it gets the Scope injected and delegates the work of sending the event to the Transport.

  • Client::from_config(config): (alternatively normal constructor) This takes typically an object with options + dsn.

  • Client::capture_event(event, scope): Captures the event by merging it with other data with defaults from the client. In addition, if a scope is passed to this system, the data from the scope passes it to the internal transport.

  • Client::close(timeout): Flushes out the queue for up to timeout seconds. If the client can guarantee delivery of events only up to the current point in time this is preferred. This might block for timeout seconds. The client should be disabled or disposed after close is called

  • Client::flush(timeout): Same as close difference is that the client is NOT disposed after calling flush

Hints

Optionally an additional parameter is supported to event capturing and breadcrumb adding: a hint.

A hint is SDK specific but provides high level information about the origin of the event. For instance if an exception was captured the hint might carry the original exception object. Not all SDKs are required to provide this. The parameter however is reserved for this purpose.

Event Pipeline

An event captured by capture_event is processed in the following order.
Note: The event can be discarded at any of the stages, at which point no further processing happens.

  1. If the SDK is disabled, Sentry discards the event right away.
  2. The client samples events as defined by the configured sample rate. Events may be discarded randomly, according to the sample rate.
  3. The scope is applied, using apply_to_event. The scope’s event processors are invoked in order.
  4. Sentry invokes the before-send hook.
  5. Sentry passes the event to the configured transport. The transport can discard the event if it does not have a valid DSN; its internal queue is full; or due to rate limiting, as requested by the server.

Options

Many options are standardized across SDKs. For a list of these refer to the main options documentation.

You can edit this page on GitHub.