54: Pick a test that you know is complicated and start there

A big piece of news recently was that the Swift language account, social media account,

is now on Mastodon and BlueSky. There was an announcement on the Swift forums. This is a

follow-up to a discussion that was also on the Swift forums where someone asked what the state

is of moving and apparently took quite a bit of effort to get this going. And it's great to see

that this is finally happening because I think I know there have been a lot of people, myself

included, who moved off of Twitter quite a while ago and were sort of cut off from any official

Swift language announcements. I mean, it wasn't hard to see or hear about stuff that was happening,

but it is kind of nice to have the official accounts in the places where people are, right?

And that has changed very significantly. To my knowledge, that doesn't mean that the Twitter

account has been abandoned. This is something I personally wish would happen too. Twitter has been

a platform that runs counter to the values Apple usually promote in and around the company. And

ever since the Musk takeover, I think it's very clear that this is going in a very different

direction. And I think it dilutes their message quite significantly. I don't think you can have

it both ways here. Promote these values and then be on Twitter. That's my personal opinion. I just

feel like I'm wanting to say that actually. And this is also something that Matt Massicotte

addressed in his blog post, Leverage, that he posted a few weeks ago. And I think you also

mentioned in Iowa's Step Weekly at the time. I did. Yeah. Yeah. And he wrote that in response

specifically to Apple recommencing their advertising on Twitter, which I think goes

even beyond just being present on the platform. It's actually giving money to the platform. I

think that's even a step beyond that. And I feel it's important that we make our voices heard here.

It's something I want to do on this occasion. I don't think it's a good look for Apple to be

present on Twitter in general, but particularly to give the platform money, as I said. And I

recall a time when Parler, for instance, was removed from the App Store because it violated

the policy. And I don't think Twitter is significantly different of a platform these

days. If anything, it's worse if you look at what's being posted there and by whom in particular.

I know these things aren't easy. There's probably a lot of pressure on the company to keep doing

certain things. But leadership is doing the hard bits, right? If stuff was easy,

there was no need for leadership. Personally, and I've also expressed this on Masternod,

I quite struggle with the fact that Apple is sponsoring Swift Package Index, which allows

me and us to work on this project that I love. But that same Apple is giving money to Musk. And

I think that's a bit of a bitter pill to swallow. I can rationalize this to a degree because I know

Apple is not a monolith. We're dealing with various different parts of Apple. And I suspect

there's lots of people, in fact, I know there are people at Apple who have different opinions

because I see them post them. And it's a huge company and there's always going to be different

ideas around how to do things. And as I said, I know there are certain pressures, but I think

it's important to not lose sight of what you actually want to be seen as standing up for.

And I think the association here is bad. And I wish all of Apple took a stronger stand in this

regard. So yeah, that's my little soapbox moment. - I mean, I agree on the Twitter thing, of course.

I wrote some stuff in iOS Dev Weekly, as you mentioned. I don't really have anything to add

to what you just said then. Well, what I will say is that it is great to see Blue Sky and

Macedon support for the Swiftlang account, to see them active there. And actually, the other

thing that's worth mentioning is that before the adoption of the two new social networks,

I know that the Swift team have been doing more promotion on social media. So they've actually

been using some of the package recommendations through the package page on the Swift website

to promote some of those packages through social media. And of course, now those are going to be

on all the platforms as well as just Twitter. So it is good to see that not only there is an

expansion of the platforms, but also that there's an expansion of the usage of those accounts as

well. - Yeah. And just to reiterate, I think it's great to see that adoption of Macedon and Blue

Sky and my little rant wasn't intended to sort of dilute that message. I think that's great that

it's happening. And I guess small steps, maybe there's a discussion to be had about taking it

further, but this is great to see. And I also have seen that the Swiftlang account has picked up some

of the links. I don't know about the change because I haven't seen what happened or is happening on

Twitter, but it's just great in general to see them show up in these places now. - Well, that's

the irony of it. I heard about this happening. I didn't see it happening because I also don't use

Twitter anymore. So I didn't see it through seeing the posts. I saw it through people telling me

that there were posts. - Right. Okay. - Which is a far more pleasant way to use Twitter in my

experience. - Twitter by osmosis. - Exactly. Yeah. By proxy. Yep. All right. Let's move on to

something else. One thing that I wanted to talk about this week was we've had a fairly long

standing issue with our ready for Swift 6 page that we did last year, which is the collection

and reporting of the number of data race safety issues that the Swift compiler is reporting

if it were in Swift, or that it would report if it were in Swift 6 language mode. And we display

that on every package page as an indicator of whether a package has data race safety errors.

And we also have this, we still have the ready for Swift 6 page on the site, which shows those

numbers over time, basically, as the Xcode releases happened over last summer. We have the kind of

historic view of the number of data race safety errors across the entire package ecosystem.

But of course, at this point now, the information on the package page is the most important thing.

And what we saw as we kind of looked into this in some detail, is that there are certain situations

where package authors are dealing with and potentially suppressing in completely legitimate

ways, data race safety errors. And the numbers that were coming out of the stats system in the

Swift compiler were not representing those suppressions correctly. I think that's pretty

much a reasonable summary of it. Yeah, maybe suppression is like, you mean like fixing or

addressing? Yeah, suppression is the wrong word. Yeah, suppression is the wrong word. I knew it,

as I was saying, I was saying the wrong word. I think specifically, like if you add

pre-concurrency imports, where you effectively indicate, well, this library isn't Swift 6 ready,

but I want to use it, I have to use it. And I know it's a legitimate use of that. And I think

the false positives there were that these certain errors were still being counted as errors,

even though they weren't actually showing up as errors when you run the build with Xcode and Swift

build. That's it. That's exactly it. So anyway, the good news is that we reported the error to the

Swift GitHub repository. And the great news is that that error has now been fixed. It's not entirely

clear whether that merge, so the fix has been merged, but whether it has been merged into 6.1,

which is now in beta, or whether we may have to wait for 6.2 for that to get through into an actual

tool chain that ships with Xcode and, you know, is in the Linux release as well. But the good news

is it's fixed. So we should, there is light at the end of the tunnel for being able to report

truly accurate beta race safety numbers, as you would see if you open the project in Xcode.

Yeah, yeah, I've made a note. So by next time, we'll know. And depending on the outcome,

well, maybe in general, we will run a new run with 6.1 is out, we sort of stopped doing the

every two week runs, because with the numbers, sort of uncertain, it didn't really make that

much sense to stress the system every two weeks with something where we knew a lot of packages,

where we're seeing issues. And yeah, it felt like, you know, not worth the effort,

without that error actually being fixed. So we know we get good numbers.

But it's going to be really great to see those numbers finally be something that people can rely

on. Where at the moment, I think we actually have a slightly softened wording on, I'm trying to find

a package with data race errors, and I can't find one. Isn't that great? I clicked on four packages

there and not a single one had data race safety errors. In fact, I'm clicking on more and none

of them have data race safety errors. That's just amazing. Let's keep rolling while you find one.

You should play some suspense music when you do the editing.

Exactly. This is remarkable. I'm still clicking and I can't find any. Okay, but I'm sure we

softened the wording slightly to say there may be data race safety errors or something like that.

I think GRDB is one because that's actually a false positive. So I think it should list an error.

If you want to check that one. Here we go. That's it. I'm loading it up right now.

Yeah, so it actually, maybe we didn't soften it because it does say has data race safety errors.

And that will be fixed. So it's great news. Excellent. All right. The other bit of news

regarding the Swift package index was that we have just finished the transition of moving all

of our tests from XCTest to Swift testing. So Swift testing is the new testing framework that

was announced at last WWDC and shipped with Swift 6. So that was the first version that had it.

Swift 6.1 has an additional feature in Swift testing, which we will want to be using. But

the initial transition is actually just needs Swift 6. So that was sort of one of the things

that we were waiting for. But also, you know, the obvious first question is, well, why would you even

want to do that? Because our tests were working. We have a lot of them. So why actually try and do

that? Ultimately, the reason is we want to try and run our tests in parallel. We have a lot of them.

They're using actual live database connections that run in the Docker container. And the idea

is to speed that up. So they are slow because of that. We have around 800 tests. And they take,

I think in CI, they take 10 minutes. It's off the order of minutes.

No, it's like, yeah, 10 to 15 minutes, I would say.

Well, a lot of it is the build. So that's why I hesitate. The whole thing, the whole CI runs 20

minutes. But a lot of that is actually the build. Because in the CI system, that's quite slow. It

might be 10 minutes build, 10 minutes test run. But even locally, where I think the most important

is the full test suite right now on my machine runs in two and a half to three minutes. And that

is quite significant. When you've done something, you know, you kind of want to have a quick check.

That's not a quick check anymore. And the idea here is to just run the tests in parallel and

cut down that time significantly. Now, in principle, that would be possible with XCTest. But

there are a couple of shortcomings with XCTest that make that harder. And I did a little exploration

before I embarked on this whole thing. And I concluded, it's not actually worth the time

trying to fix that in XCTest. It's actually better to just transition and then do it properly in

Swift testing. Because obviously, that's the thing that's going to carry us forward much further than

XCTest ever could. And it also has obviously a lot of other nice things about it, you know, like

better asserts, expect macros, give a nicer printout. And it's faster, actually, the test

discovery is faster. Also, like Linux and macros tests actually run in the same way now. Lots of

little things and bigger things that come together that make this worthwhile. But also, there is one

piece of sort of a tool that really makes that feasible at all. And that is something I mentioned

before on the podcast, I think I even had as a package pick. And that is Swift Testing Evolutionary.

That's a package that was driving this transition. And I'll get into it a bit more. But I just want

to describe the sort of where we started off. So we had our test suite is, as I said, quite

significant, we have 105 tests, like test classes, which turn into test suites and Swift testing,

with overall 798 tests. So that's the individual test methods on XCTest, which get turned into

these at test attributed functions. And inside of those we have 2300 asserts. That's what gets

turned into these expect and require macros after conversion. And you can imagine, I mean, you can

you can potentially manually transform 105 test classes into suites, that's sort of not something

crazy. You could conceivably convert the tests, you know, because that's maybe doable with

replace, you know, some some replace magic. But all the asserts is just that would be madness.

And this is where this tool Swift Testing Revolutionary comes in, because it does that

for you, it actually uses Swift syntax to look at the test structure and rewrites all the code,

it rewrites the test suites, the test functions into the attribute test functions, and it rewrites

the asserts. And that's a critical piece. And it does a really, really good job. There are a couple

of cases where it doesn't quite work. It's mainly around where you have where you try to test

exceptions or not. Yeah, exceptions and have throwables that you want to assert. So there's

a bit of syntax weirdness that you have to deal with manually. But the vast majority of these,

I'd say like 95% of these assert conversions, it does automatically. And that is just huge,

because that really makes that feasible at all. Because otherwise, if this tool didn't exist,

I don't think I would have done that transition, certainly not within the time it took in the end.

So I started this in on February 5, and I finished on March 5. And this was not a full time project.

This was, you know, as I went along, I did, every day, I did a couple of test cases that I converted

and tweaked. Obviously, I picked a difficult one up front to see, you know, which bits are

potentially hard to convert. And there was a bit of upfront work I had to invest because

there are a couple of structural differences between XC testing and Swift testing. They're

mainly around sort of I pattern, which is resource acquisition is initialization. By that I mean,

when you use a database connection, for instance, in a test, you typically set that up in a test

setup function. And then you tear down in a test tear down, you know, because we have a Docker

container that we run against, and we don't want to have tests, cross talk where one test would

leave around stuff that the next test then sees and, you know, it would affect the results.

So what we do is we do a schema setup, and then a schema wipe at the end of the test or at the

start of the next test. And that is something that works a bit differently between the two

mechanisms. And that was a piece that really required a bit of redoing. But once I'd done

that, it was really quite easy and mechanical to convert the tests over. So if you're planning to

do this yourself, I'd really recommend pick a test that you know is sort of complicated deals with

some external dependencies, stuff like that maybe is is testing something very complicated,

convert that. And when when that's working, when you know it convert work on that one, then do the

rest because after that, it'll be really, really easy and quite mechanically especially if you use

Swift testing revolutionary to do the the automated bits for you.

Yeah, that's great. I mean, I, I did look at some of the the the results of the conversion. And I

think it was a curious choice to just replace every assertion with expect true to equals true.

The good news is the test path pass really fast now.

Yeah. Yeah. And not only that, it also gives you more.

I think you missed my joke. Every expectation is replaced with true equals true.

But everything is super fast.

I end up with Sorry, I missed this joke. It was

well, you can you can add a laugh track. I was so focused on the next thing I was gonna say.

Apologies. The other bit that's really nice about a it makes it really easy to to actually write

expectations because it's you just need to expect and then you write it as you normally would you

don't have to think about well, is it? Yeah, like exit he assert equals is it equals accuracy? Well,

actually, is actually bad example, because that's a bit of a gap in in XC, in Swift testing right now.

But what it does is, for many types, it will actually give you really detailed printout if

something is not equal. And that's that's a problem often when you have complex types where you,

they might be equatable. And you the exit he assert equals macro works, the old one, it will show you

there's a you know, it fails properly. But then the output is, is effectively, it doesn't tell

you anything because you just effectively just see what this fails and the the printout is is useless.

And this is where expect macros are, are better because it can drill into a lot of stuff like

arrays and dictionaries and actually pull out the details and give you more information about the

text failure without you actually having to actively instrument that in more detail yourself.

Right. So yeah, it's that's, it's, it's great. I'm, I'm really looking forward now to tackle

the the actual objective in making our tests, which is the parallel parallel. Yeah. And I've

hit a couple of additional problems now as you do. But I think we can at least move some of them

to run in parallel fairly soon. There's a bit of legwork still to do, but I'm quite hopeful that

this will work out. It has worked in principle for one test, the problem is a bit to to do this

for the whole test suite when you run parts of it in parallel and parts of it really need to

still run sequentially, I'll have to figure out if how to best address that. But I'm hopeful that

this, this will work and then then speed things up. Yeah, that's great. Great to see. And I'm happy to

happy to see that Swift testing has been such a positive enhancement. I don't think there's any

kind of steps backwards with Swift testing. I think it's looking great. Yeah. Yeah. And if you

do the transition and you have any questions, there's a great Slack channel on the swift.org,

the official Slack Swift dash testing, where Jonathan Greenspan, for instance, is very responsive

to questions. And, you know, just pop in that channel if you encounter any any difficulties.

There are a couple of rough edges still that are being ironed out right now. But overall, it's

really pleasant to work with. And if you run into any issues, people there are really super helpful

and will get you will get you going. All right. I think there's just one very short other piece

of news, right? There is indeed. Yes. And that was a blog post that was announced on the Swift

forums, I think just earlier this week or in the last week. Swift on Raspberry Pi building

natively and cross compiling by Nathan Volnick, assisted by Jesse Zamora. And that's sort of our

our ongoing topic of Swift in unusual places. Maybe not that unusual. I think Raspberry Pi

has sort of been a supported platform for quite a while. But this is really great post that gets you

started. You know, like if you have a Raspberry Pi and nothing else, you it'll it'll walk you

through how to set the thing up even not just about Swift, but just Raspberry Pi in general,

what images to flash onto the thing and then how to get Swift onto it. Get a program running,

compile stuff there directly, but also do the cross compilation, which is super useful. And

because the Raspberry Pi can be slow to compile Swift stuff. I mean, obviously, it's it's not a

machine comparable to a Mac compiling stuff. And that's, that's great. Now, I think it was Swift 6

that introduced proper cross compilation was possible before, but now it's a properly supported

setup, which means on your Mac, you can you download and prepare a specific tool chain for

cross compilation, and then you pass a flag to the compilation command and you get a binary that is

that you just SSH or SCP over to the Raspberry Pi and run there. And that's that's that. So you

can cut down on compile times quite a lot, especially if you have a beefy Mac, that should

help a lot to get you going on your Raspberry Pi projects. So yeah, give that give that blog post

a look if you're into that sort of thing. That's great. Okay, should we do some package

recommendations? Let's do it. Yeah, I shall kick us off this week. And I'm going to start with

a double a double recommendation. They're actually quite different packages, but they're both

brand new and both by the same author. So the author is Robb Böhnke. I'm sorry for butchering

that pronunciation. I think I might help you here because I think I think I was gonna ask

for your assistance because you're German. Okay, sorry, Rob. They're from Robb Böhnke.

And the first one is called AtRandom. And it's a property wrapper that you can use specifically,

I'm sure you can use it for other things. But specifically, it's designed for use within

animation code in SwiftUI. So let's say you wanted to grab a random number to start or affect

some animation by a certain random number. So it's not exactly the same every single time.

You might choose to store that in a state variable because you might want the same random number

every time. But instead, you could use this random property wrapper, and it will create a

persistent random number. So it will be the same random number every time you come into your

into the function. But it will be random on that initial set, I guess it's the property

wrapper version of a C static. So that's really not a not again, not something you're going to

use every day. But I did like the idea of that. And the more you work on SwiftUI code, the more

the state variables can kind of explode, especially with things that are not necessarily

really important to store as state variables like this. So I like that. And the second package from

the same person is called visualized touches. And I've seen a couple of these over the years.

But I really like this one. And it's also compatible with SwiftUI, which is great to see.

So when you're recording a demo of your application, or a tutorial for your application,

or something like that documentation, or whatever it is that you need a screen recording for,

you can, it's useful to be able to show where your fingers are touching on the screen. And Rob

has developed a view modifier for SwiftUI, that you can just add visualize touches on the end of

any SwiftUI view. And it will represent wherever you're touching on your device with two very

beautifully drawn kind of circles to show each fingers kind of placement on that view. Not

something you'd ever want to ship. Because you wouldn't want this happening when people were

actually using your application. But really great for preparing demos and tutorials, like I said.

Yeah, that's such a great thing to have. I always love it when that's actually available.

Sometimes I see people posting screen recordings of their iOS devices. And I really struggle to

see what's actually going on, because I really miss that information. You know, there's no mouse

cursor or anything to guide your eye, to guide you where to look. And I feel like often stuff is

happening and I have no idea what's going on. So I really love that sort of that extension and

videos that actually have that as a tool to guide you. Great packages.

Yeah. And it's, they're just, the touches that get visualized, it seems strange to call them

beautiful because it's just a couple of circles, but they're actually really, they're really nicely,

they're really nicely done. So that's Visualize Touches and AtRandom by Robb Böhnke.

Great. My first pick is a package by, well, who other than pointfree co,

repeat offenders. They keep publishing great packages. I think you also had this

on our Dev Weekly, apologies for mentioning it again. It's SharingGRDB

and this is sort of a double mention also because a while ago,

pointfree co released their package, Swift Sharing, which is perhaps best described as

an observable style macro that allows for a configurable backing store.

What that means is that you can annotate a property in your typically Swift UI project,

let's start there, to be backed by some kind of a store like user defaults, the file system,

whatever, it could even be a server that you connect to. The interesting bit about Swift

Sharing in general is that it's not Swift UI only, and that's how it's different to observable.

You can use this to drive your view models or it's also cross-platform, so you can use it on

Linux for stuff there, for instance. And Sharing GRDB is an extension on top of Swift Sharing,

and it gives you a SQLite-backed store powered by GRDB. And that's a library we talked about

in episode 43. And it's probably the most prominent SQLite library in the Swift ecosystem,

I'd say. So this sort of marries the two packages and gives you the power of Swift Sharing backed by

GRDB as the persistence layer. And all that GRDB gives you on top of this, obviously, available.

And SQLite in general is just a very powerful exchange format too,

meaning you could use this, for instance, also as a file format to ship stuff around.

It's very compelling for that. It's very easy to use and certainly more performant and more

structured than a JSON file or multiple files that you might be moving around here, because

you can use SQLite to then run queries and stuff. You can even run subscriptions on changes. You get

notified if stuff changes in the database and then get notified. And Sharing has a mechanism for that

too. That's what Observable is about, to feed views by updating the database. You can then fan

out and update views automatically. It's quite powerful. Look at it if you are in need for that.

Also worth mentioning is Sharing GRDB back deploys to iOS 13. So that might be a really good

alternative to Swift data, because that is not available to my knowledge that far back. Well,

certainly it isn't, but I don't actually know what the cutoff is for that. So if you have needs for

persistence and Observable style stuff, give this a look. It looks really, really compelling.

That's great. And yes, I did mention that in the newsletter too. It's a great looking package.

My final package for this week is kind of a fun one. I think I am getting a reputation for

maybe making my last package a fun one each week. Maybe that's going to stick.

But it's called AmplifyUILiveness. And it is a wrapper around a Amazon AWS service that can

determine the liveness of a face. Are you looking at a real live human being or are you looking at

a photograph or a simulation of a human being? And it will determine whether, well, it will try

and determine, I don't know how accurate it is, I haven't tried it, but it will attempt to determine

to determine the liveness of the face that you present to it. And so this is actually from,

the package is from AWS Amplify themselves in their organization. And yeah, I just,

I love that idea. I think it's such a, it's a very 2025 problem.

You know what my immediate first thought is, how easy is that going to be to defeat with a photo

and some AI stuff happening on a photo to make it? Well, I think that's the point of it. I think

that's exactly what it is intended to defeat. Oh, really? Okay. Interesting.

And I'm guessing the kind of use for this would be, I know that my bank, every time I need to

log in to my bank's app, if it's somehow found itself, you know, maybe after a major version

upgrade on iOS or something like that, you have to actually record yourself a little selfie video

and send it. And I believe that what they're doing is a human checks it because it's like 15 to 20

minutes before it kind of gets back to you. So I'm guessing that's a human sitting there and

saying, is this the person that is supposed to be? And I'm guessing that this might enable at

least like a pre-check or something like that. I wouldn't want to use something like this for a

positive signal, but it would be great to use this as a negative signal. So this might screen out all

the obviously fake ones or something like that. Yeah. Interesting. Yeah, that's an interesting

package. I hadn't seen that. There we go. I'm going to find the obscure packages every few weeks.

Well, I have no surprise for you because my second pick is also one, apologies again,

I sort of had this on my list as well. And I didn't feel like chucking it because

you mentioned it on our stuff weekly. It's Spices by Simon Støvring.

Do you get all your package recommendations from iOS stuff weekly?

I shall send you messages when I make my picks and make sure that they happen before you publish.

So what was the package? Sorry.

The package was Spices by Simon Støvring. And that's a, as you know, a lovely library to create

debug views and menus from a spec for your app. It's iOS only, I believe. And essentially what

you do, you annotate a Spice store entity with @Spice attributes, like the properties on it.

And then effectively, that's it. That's all you do. Off you go. And it generates a whole view,

a debug view that you can bring up in one way or another. Typically people do it with a shake or

secret gesture or something like a three finger tap, whatever you do. And then you get a really

nice rendering of all the settings that you've added to this Spice store, like Boolean flags or

pop-ups, which API service you might want to connect to, like a QA environment or a staging

environment, that sort of stuff. Stuff that's typically in dev settings for a device.

And have a look at the readme. It has a great little video showing how easy it is to get up

and running with this. And I think it's also a great use of a third party library in this context.

You know, we often talk about when would you use something or how careful do you have to be with

bringing in dependencies? And I think this is a super low risk thing to do because push comes to

shove. Your product release would not depend on this, right? Because it's a pure testing tool.

Obviously it might be an important testing tool, but it's probably not a show-stopping testing

tool if something were to go wrong with this, or for some reason a new Xcode version had troubles

with it and you wouldn't be able to build with it and you had to take it out. That wouldn't be

breaking your bill. So I always like it when dependencies come in that way and give you this

escape hatch of, well, if everything goes pear-shaped, you can at least not use it and

still have a product that you can release. I think in that sense that's a fantastic library to use

and make your dev environment nicer very easily because it's super simple to use.

That sounds great.

Yeah. So there you go. Spices by Simon Støvring.

Fantastic. And we'll call it there, I think. So we will be back in a couple of weeks, well,

a few weeks, with more package recommendations and more news from the world of Swift and Swift

packages. But until then, I think we will see you next time.

See you next time. Bye-bye.

All right. Bye-bye.

Creators and Guests

Dave Verwer
Host
Dave Verwer
Independent iOS developer, technical writer, author of @iOSDevWeekly, and creator of @SwiftPackages. He/him.
Sven A. Schmidt
Host
Sven A. Schmidt
Physicist & techie. CERN alumnus. Co-creator @SwiftPackages. Hummingbird app: https://t.co/2S9Y4ln53I
54: Pick a test that you know is complicated and start there
Broadcast by