Integration Setup
The Embrace SDK can be configured to work with other components in the OpenTelemetry ecosystem. This includes any Exporters and instrumentation libraries that can run on the supported mobile platforms, so long as the underlying API is implemented by Embrace.
Telemetry captured by the Embrace SDK can be sent directly from the mobile app to any configured OTel Collector. As well, additional OTel signals recorded by instrumentation libraries external to Embrace will be included in the Embrace session, as if they were recorded by the SDK itself. Any OTel APIs implemented by Embrace can also be used directly by SDK users if they so choose.
Export to OpenTelemetry Collectors
Telemetry can be sent to any OpenTelemetry Collector directly from an app using the SpanExporter and LogRecordExporter. Once they are configured, any telemetry will be sent to these exporters as soon as it is recorded. More than one exporter can be configured for each signal, but be aware that there can be a performance impact when sending too many network requests.
Currently, traces and logs can be exported using the Embrace Android SDK and Embrace Apple SDK.
Android
- Kotlin
- Java
Embrace.getInstance().addSpanExporter(mySpanExporter)
Embrace.getInstance().addLogRecordExporter(myLogExporter)
Embrace.getInstance().addSpanExporter(mySpanExporter);
Embrace.getInstance().addLogRecordExporter(myLogExporter);
Please note that exporters must be configured before the Embrace Android SDK is started. Exporters added after the SDK has already been started will not be used.
Local Testing
To see this working locally, LoggingSpanExporter and SystemOutLogRecordExporter can be used to output to logcat.
2024-03-05 14:15:15.342 29672-29756 LoggingSpanExporter io.embrace.mysampleapp I 'emb-startup-moment' : d38b4ac26baf1a862ed4a028af7d08ac e3e82dd0f86c0eed INTERNAL [tracer: io.embrace.android.embracesdk:=7.0.0] AttributesMap{data={emb.sequence_id=4, emb.type=PERFORMANCE, emb.key=true}, capacity=128, totalAddedValues=3}
Sending Telemetry Off the Device
You can send your data to any generic OpenTelemetry Collector by using any Android-compatible exporter. Note that not all Java SpanExporter or LogRecordExporter can be used on Android.
- Kotlin
- Java
//gRPC through an OTel Collector in a local docker image
val customDockerExporter = OtlpGrpcSpanExporter.builder()
.setEndpoint("https://otel-collector.mydomain.com:4317")
.build()
Embrace.getInstance().addSpanExporter(customDockerExporter)
//gRPC through an OTel Collector in a local docker image
OtlpGrpcSpanExporter customDockerExporter = OtlpGrpcSpanExporter.builder()
.setEndpoint("https://otel-collector.mydomain.com:4317")
.build();
Embrace.getInstance().addSpanExporter(customDockerExporter);
Network requests to OpenTelemetry Collectors should not be logged
To prevent an infinite loop of network requests spans, any requests used to export telemetry to OpenTelemetry Collectors should be excluded from being recorded by the Embrace Android SDK using the disabled_url_patterns
setting in the Embrace Configuration file. See this page for details.
Sending Telemetry to Grafana Cloud
To send telemetry to Grafana Cloud, set up the collector and add an authorization token as a header.
- Kotlin
- Java
//HTTPS to an OTEL Collector in Grafana Cloud
val grafanaCloudExporter = OtlpHttpSpanExporter.builder()
.setEndpoint("https://myinstance.grafana.net/otlp/v1/traces")
.addHeader("Authorization", "YourToken")
.build()
Embrace.getInstance().addSpanExporter(grafanaCloudExporter)
//HTTPS to an OTEL Collector in Grafana Cloud
OtlpHttpSpanExporter grafanaCloudExporter = OtlpHttpSpanExporter.builder()
.setEndpoint("https://myinstance.grafana.net/otlp/v1/traces")
.addHeader("Authorization", "YourToken")
.build();
Embrace.getInstance().addSpanExporter(grafanaCloudExporter);
Apple
In the Embrace Apple SDK, exporters are configured at the same time that the SDK itself is configured. Any Swift-language OTel exporter can be attached as an optional value when the SDK is set up in Embrace Options. Multiple span exporters can be attached during configuration by using the otel-swift MultiSpanExporter
.
// logging span exporter outputting to the Xcode console
var loggingSpanExporter = StdoutExporter()
// gRPC span exporter to an OTel Collector in a local docker image
var otelCollectorSpanExporter: OtlpTraceExporter {
let config = ClientConnection.Configuration.default(
target: ConnectionTarget.hostAndPort("localhost", 4317),
eventLoopGroup: MultiThreadedEventLoopGroup(numberOfThreads: 1)
)
let client = ClientConnection(configuration: config)
return OtlpTraceExporter(channel: client)
}
// HTTP span exporter to OTel Collector in Grafana Cloud
var grafanaOtelSpanExporter: OtlpHttpTraceExporter {
let grafanaCloudTokenString = //String generated from your account
let urlConfig = URLSessionConfiguration.default
urlConfig.httpAdditionalHeaders = ["Authorization": "Basic \(grafanaCloudTokenString)"]
return OtlpHttpTraceExporter(
endpoint: URL(string: "https://otlp-gateway-prod-us-east-0.grafana.net/otlp/v1/traces")!,
useSession: session
)
}
// configure Embrace Apple SDK, with all three span exporters attached
try? Embrace
.setup(
options: Embrace.Options(
appId: "AppID",
export: OpenTelemetryExport(
spanExporter: MultiSpanExporter(
spanExporters: [
loggingSpanExporter,
otelCollectorSpanExporter,
grafanaOtelSpanExporter
]
),
logExporter: nil
)
)
)
.start()
Use the OpenTelemetry Tracing API
An Embrace-enhanced implementation of the OTel Tracing API can be obtained through the SDK. The resulting OpenTelemetry
object will provide working implementations of interfaces and methods of said API, with proper attribution included in the resource of the exported spans. All other methods will be no-ops for the time being, as the other APIs have not been implemented.
The enhanced Tracing API is currently available in the Embrace Android SDK.
Limitations
The Embrace implementation of the Tracing API deviates from the official SDK implementations in the following ways:
- SpanLinks are not supported -
addLink()
in theSpan
implementation is a no-op. - Attribute values are persisted as Strings.
- The resource attributes
service.name
andservice.version
refer to the Embrace SDK at this time. This will be changed shortly.
Android
To obtain an implementation of OpenTelemetry
, the method getOpenTelemetry()
can be used. With this, instances of TracerProvider
and Tracer
can be created and retrieved, depending on what the instrumentation library requires.
For instance, the OkHttp Instrumentation can be used manually in the following ways:
- Kotlin
- Java
val openTelemetry = Embrace.getInstance().getOpenTelemetry()
val okHttpTelemetry = OkHttpTelemetry.builder(openTelemetry).build()
val okHttpClient = OkHttpClient.Builder().build()
val request = Request.Builder().url("https://api.mydomain.com").build()
okHttpTelemetry.newCallFactory(okHttpClient).execute(request)
OpenTelemetry openTelemetry = Embrace.getInstance().getOpenTelemetry();
OkHttpTelemetry okHttpTelemetry = OkHttpTelemetry.builder(openTelemetry).build();
OkHttpClient okHttpClient = OkHttpClient.Builder().build();
Request request = Request.Builder().url("https://api.mydomain.com").build();
okHttpTelemetry.newCallFactory(okHttpClient).execute(request);