Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/heckj/distributedtracer
A wrapped swift-otel tracer you can use with iOS and macOS tests
https://github.com/heckj/distributedtracer
distributed-tracing opentelemetry swift swift-otel
Last synced: 3 months ago
JSON representation
A wrapped swift-otel tracer you can use with iOS and macOS tests
- Host: GitHub
- URL: https://github.com/heckj/distributedtracer
- Owner: heckj
- License: mit
- Created: 2024-04-01T01:52:24.000Z (9 months ago)
- Default Branch: main
- Last Pushed: 2024-04-03T16:40:22.000Z (9 months ago)
- Last Synced: 2024-10-09T19:23:28.751Z (3 months ago)
- Topics: distributed-tracing, opentelemetry, swift, swift-otel
- Language: Swift
- Homepage:
- Size: 14.6 KB
- Stars: 3
- Watchers: 2
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# DistributedTracer
My "Stupid tricks with Distributed Tracing" - generating traces from XCTest cases.
I shoved this together, wrapping the amazing work of [Moritz Lang](https://github.com/slashmo)'s [swift-otel](https://github.com/slashmo/swift-otel) package, so that I could
instrument a library using [Swift Distributed Tracing](https://github.com/apple/swift-distributed-tracing) - and see the traces from running tests locally.It's admittedly a hack-job, needed because some of the libraries around tracing infrastructure can only be initialized once per process.
To accommodate that constraint, I've set this up as a global actor that you access through a `shared` instance,
and you have to call the async method `bootstrap()` on it, passing in a name for the "service" you're bootstrapping.
You can call that `bootstrap()` method as many times as you like, but it only really works the first time.
Since tests can be run in any order (and you generally want that), I call `bootstrap` in an async setup function.For example, using `XCTest`, the general structure looks like:
```swift
final class SomeTestSuite: XCTestCase {
override func setUp() async throws {
await TestTracer.shared.bootstrap(serviceName: "TestService")
}override func tearDown() async throws {
if let tracer = await TestTracer.shared.tracer {
tracer.forceFlush()
// Testing does NOT have a polite shutdown waiting for a flush to complete, so
// we explicitly give it some extra time here to flush out any spans remaining.
try await Task.sleep(for: .milliseconds(100))// This would be a whole lot better at the tail end of the entire suite, or at least
// at the class tearDown, but `override class func tearDown` doesn't appear to have
// an async callback where we can do cleanup/flush work like this, so it's in
// each test execution. Sorry about that.
}
}func testClosureInvokerTest() async throws {
...
}
}
```## Seeing it
To spin up a Jaeger trace collector:
```
docker-compose up -d
```Run [the tests from this package](https://github.com/heckj/DistributedTracer/blob/main/Tests/DistributedTracerTests/DistributedTracerTests.swift) to generate a few tests:
```
swift test
```Open up a browser to see any traces generated to [http://localhost:16686/](http://localhost:16686/), which accesses the Jaeger running in Docker.
## Possible Future Work
I have some hare-brained ideas about using this further, and definitely trying it out with [swift-testing](https://github.com/apple/swift-testing) once that gets a little easier.
In general, I would really like to be able to both export spans from tests, and - in the case of an integration test specifically - be able to retrieve them after the actions of the integration test(s) are complete in order to assert/validation against them.The other possibility that seems super-interesting to me is enabling tracing from Apple native-platform apps (iOS, macOS, etc) that works in concert with back-end tracing.
There are significantly more challenges there, among which are:
- making sure there's some representation of privacy if you use such a library - privacy manifests, etc
- coordinating capturing traces with backend infrastructure (in large environments, there are too many traces to rationally keep, so you sample them - but ideally you want the samples to be matched between these disparate tracing collection sites. The idea is talked about in distributed tracing circles as 'tail sampling')
- tuning the amount of data captured, and enabling a transport to get them to your back-end infrastructure (basically, don't bloat memory and die, and try to reliably transmit this stuff so you don't loose "the good stuff")
- coming up with an way to ingest into your back-end services in a way that doesn't open up a monstrous security hole that would allow someone to denial-of-service your diagnostics/observability infrastructure.If any of this sounds super-interesting to you, let me know - or jump in and contribute here if you like.
And if you want something like this for your company... let me know.
I'm available freelance, and we can work out a means to add the feature or capabilities to this (or other) open-source projects to support the things we like to use.