How to Speed Up Codecov Analysis for Xcode Projects
💁🏻♂️ Note: A version of this post also appears at the Codecov Blog and The New Stack.
At my job, I focus on developer automation and tooling for mobile apps. One of the newer tools we’ve been using is Codecov. It tracks and reports on our code coverage in total, and on pull requests.
One of my recent focuses has been finding ways to optimize our CI workflows. I noticed that for our iOS app, the Codecov step of our workflow was taking much longer than expected, so I decided to find out why, and if there was anything we could do to improve it.
Xcode collects code coverage data and can display it in the IDE for developers. It can also be exported as an
.xcresult file when using
xcodebuild from the command line, like so:
xcrun xcodebuild test \
-project tconnect.xcodeproj \
-scheme tconnect \
-testPlan tconnect \
-destination "platform=iOS Simulator,name=iPhone 14" \
-derivedDataPath DerivedData \
These xcresult files are great, and can be useful in lots of different ways1 , but like many things Apple, they can be difficult to use outside the Apple ecosystem.
xcresult files are a binary format, and while they can be converted into a JSON representation using the
xccov binary included with Xcode, the resulting JSON is not in one of the standard coverage formats that Codecov can ingest. The format has also been known to change without warning with new Xcode releases.
So in order for Codecov to use the coverage results from Xcode, they have to be converted into another format. Codecov’s official GitHub Action can do the conversion, but the way it handles this conversion is by analyzing the coverage for each file one by one, which can take up to a second for each file. This is a fine enough approach for some projects, but when working with a large codebase like ours, that can take quite some time.
xcresultparser is an open source Swift tool that can parse
.xcresult files, and convert them into various other formats. One of these formats is Cobertura XML, which Codecov natively supports.
The big advantage xcresultparser brings is, because it is a compiled program and not a script, it can utilize multiple threads to do the conversion. This speeds up the conversion process immensely.
After running the
xcodebuild command above to generate the
.xcresult file, we tell xcresultparser to convert it like so:
--output-format cobertura \
And finally, we tell the Codecov GitHub Action to upload that XML file instead of the xcresult file.
So, just how much time savings are we getting?
|Total Build Time Before
|Total Build Time After
|App Unit Tests
|Library Unit Tests
We run these builds in parallel, so the total real-time savings for each build is the delta of the Library Unit Tests build; around 7 minutes! This might not seem like much, but when you factor in that we’re running these builds upwards of 20 times a day, it’s a considerable time (and cost) savings. That’s over 2 hours of total developer time saved per day; almost 12 hours per week!
Bonus: Test Result Summaries
While implementing xcresulparser for our project, I learned that it can also print a summary of test results to the command line. Our Library unit tests are 8 separate test suites that run in serial. If a test fails near the top of the log output, it can be difficult to find.
So, at the end of each test run, we print out a summary like so:
--output-format cli \
This produces output that looks like this:
Test result summary example
xcresultparser has improved the lives of our developers quite a bit. And the fact that it is open source means that we as a developer community can help improve it for the benefit of ourselves and others. Check it out if you’re using Codecov (or another similar tool) to track code coverage on your Xcode projects.
|Jun 29th, 2023
|apple, programming, software, swift