Creating and Using a Span

A Span is the name of the object that we use to capture data about an error and its surrounding context. It is designed to be similar to, but not exactly like, the Span from the OpenTelemetry standard specification.

Creating a new Span

A Span can be created by calling appsignal.createSpan(), which initializes a new Span object with any default options that you passed when the Appsignal object was initialized.

const span = appsignal.createSpan();

The createSpan() method can also take a function as a parameter, allowing you to define the Span's data as the same time as it is created.

const span = appsignal.createSpan((span) => {
  return span.setTags({
    tag: "value",
  });
});

Spans cannot be nested, nor can multiple Spans be passed to appsignal.send() at once. We recommend using Promise.all for concurrent send operations.

const span1 = appsignal.createSpan();
const span2 = appsignal.createSpan();
 
Promise.all([appsignal.send(span1), appsignal.send(span2)]);

Updating a Span

After a Span is created, you can begin adding data to it using methods on the Span object:

const span = appsignal.createSpan();
span.setTags({ tag: "value" });
 
console.log(span.serialize().tags); // { tag: "value" }

Each method that modifies the Span returns this, allowing you to chain methods together:

const span = appsignal.createSpan();
 
span.setTags({ tag: "value" }).setError(new Error("test error"));
 
console.log(span.serialize().tags); // { tag: "value" }
console.log(span.serialize().error); // { name: "Error", message: "test error", backtrace: [...] }

span.setAction(name: string)

Sets the action of the current Span. The action must never be an empty string - it can be either undefined or a non-empty string. An action name is used to identify the location of a certain sample error and performance issues.

span.setNamespace(name: string)

Sets the namespace of the current Span. Namespaces are a way to group error incidents, performance incidents, and host metrics in your app.

span.setError(error: Error | object)

Sets the error of the current Span. The Error object has a name, message and backtrace property. Make sure to only use something describing the error's type as the name. AppSignal groups the errors based on this name. You can put anything you like in the message.

When an Error object is passed to the setError method, the stack property is normalised and transformed into an array of strings, with each string representing a line in the stacktrace. For consistency with our other integrations, stack is renamed to backtrace.

span.setTags(tags: object)

Adds tags to the current Span. The current tags will be merged with the tags object passed as a parameter.

span.setParams(params: object)

Adds params to the current Span. The current params will be merged with the params object passed as a parameter. The params object must not contain any nested objects.

span.serialize()

Returns the contents of a Span as an object.

Sending a Span to AppSignal

When you're finished adding data to the Span, it can then be passed to appsignal.send() to be pushed to our API.

const span = appsignal.createSpan((span) => {
  return span.setError(new Error("test error"));
});
 
appsignal.send(span);

The send() method is different to the sendError() method as it allows a Span to be passed as a parameter, which is either pushed immediately to the API, or in the case of a network error, added to the queue to be retried later.

Once the Span is passed to appsignal.send(), any Hooks are applied to the Span in the following order:

  • Decorators
  • Optional tags or namespace arguments to appsignal.send()
  • Overrides

About the Retry Queue

If, for any reason, pushing an error to the API fails (e.g. if the network connection is not working), the Span object that it belongs to is placed in the retry queue. By default, requests are retried 5 times with exponential backoff. If the request succeeds, the corresponding Span is removed from the queue. Once the retry limit has been reached, any Spans left in the queue are discarded.

No caching is currently implemented for the retry queue, meaning that if a Span is in the queue when the user navigates away from your aplication, that Span will also be discarded.

Want to help us improve this documentation page?

Create a pull request

Need more help?

Contact us and speak directly with the engineers working on AppSignal. They will help you get set up, tweak your code and make sure you get the most out of using AppSignal.

Contact us

Start a trial - 30 days free

AppSignal is a great way to monitor your Ruby, Elixir & Node.js applications. Works great with Rails, Phoenix, Express and other frameworks, with support for background jobs too. Let's improve your apps together.

Start a trial