hs-opentelemetry-api-0.0.2.0: OpenTelemetry API for use by libraries for direct instrumentation or wrapper packages.
Copyright(c) Ian Duncan 2021
LicenseBSD-3
MaintainerIan Duncan
Stabilityexperimental
Portabilitynon-portable (GHC extensions)
Safe HaskellNone
LanguageHaskell2010

OpenTelemetry.Context.ThreadLocal

Description

Thread-local contexts may be attached as implicit state at a per-Haskell-thread level.

This module uses a fair amount of GHC internals to enable performing lookups of context for any threads that are alive. Caution should be taken for consumers of this module to not retain ThreadId references indefinitely, as that could delay cleanup of thread-local state.

Thread-local contexts have the following semantics:

  • A value attached to a ThreadId will remain alive at least as long as the ThreadId.
  • A value may be detached from a ThreadId via detach by the library consumer without detriment.
  • No guarantees are made about when a value will be garbage-collected once all references to ThreadId have been dropped. However, this simply means in practice that any unused contexts will cleaned up upon the next garbage collection and may not be actively freed when the program exits.

Note that this implementation of context sharing is mildly expensive for the garbage collector, hard to reason about without deep knowledge of the code you are instrumenting, and has limited guarantees of behavior across GHC versions due to internals usage.

Why use this implementation, then? Depending on the structure of libraries that you are attempting to instrument, this may be the only way to smuggle a Context in without significant breaking changes to the existing library interface.

The rule of thumb: - Where possible, use OpenTelemetry.Context in a reader-esque monad, or pass it around directly. - When all else fails, use this instead.

Synopsis

Thread-local context

getContext :: MonadIO m => m Context Source #

Retrieve a stored Context for the current thread, or an empty context if none exists.

Warning: this can easily cause disconnected traces if libraries don't explicitly set the context on forked threads.

lookupContext :: MonadIO m => m (Maybe Context) Source #

Retrieve a stored Context for the current thread, if it exists.

attachContext :: MonadIO m => Context -> m (Maybe Context) Source #

Store a given Context for the current thread, returning any context previously stored.

detachContext :: MonadIO m => m (Maybe Context) Source #

Remove a stored Context for the current thread, returning any context previously stored.

adjustContext :: MonadIO m => (Context -> Context) -> m () Source #

Alter the context on the current thread using the provided function

Generalized thread-local context functions

lookupContextOnThread :: MonadIO m => ThreadId -> m (Maybe Context) Source #

Retrieve a stored Context for the provided ThreadId, if it exists.

attachContextOnThread :: MonadIO m => ThreadId -> Context -> m (Maybe Context) Source #

Store a given Context for the provided ThreadId, returning any context previously stored.

detachContextFromThread :: MonadIO m => ThreadId -> m (Maybe Context) Source #

Remove a stored Context for the provided ThreadId, returning any context previously stored.

adjustContextOnThread :: MonadIO m => ThreadId -> (Context -> Context) -> m () Source #

Alter the context