Finally learning some Rust - hello photo-backlog-exporter!
Posted on March 9, 2024 with tags opensource, rust. See the previous or next posts.
Wow that's a long title š
After 4? 5? or so years of wanting to learn Rust, over the past 4 or so months I finally bit the bullet and found the motivation to write some Rust. And the subject.
And I was, and still am, thoroughly surprised. Itās like someone took Haskell, simplified it to some extents, and wrote a systems language out of it. Writing Rust after Haskell seems easy, and pleasant, and you:
- donāt have to care about unintended laziness which causes memory āleaksā (stuck memory, more like).
- donāt have to care about GC eating too much of your multi-threaded RTS.
- can be happy that thereās lots of activity and buzz around the language.
- can be happy for generating very small, efficient binaries that feel right at home on Raspberry Pi, especially not the 5.
- are very happy that error handling is done right (Option and Result, not like Goā¦)
On the other hand:
- there are no actual monads; the ?operator kind-of-looks-like being indoblocks, but only and only for Option and Result, sadly.
- thereās no Stackage, itās like having only Hackage available, and you can hope all packages work together well.
- most packaging is designed to work only against upstream/online crates.io, so offline packaging is doable but not ānativeā (from what Iāve seen).
However, overall, one can clearly see thereās more movement in Rust, and the quality of some parts of the toolchain is better (looking at you, rust-analyzer, compared to HLS).
So, with that, Iāve just tagged photo-backlog-exporter v0.1.0. Itās a port of a Python script that was run as a textfile collector, which meant updates every ~15 minutes, since it was a bit slow to start, which I then rewrote in Go (but I donāt like Go the language, plus the GC - if I have to deal with a GC, Iād rather write Haskell), then finally rewrote in Rust.
What does this do? It exports metrics for Prometheus based on the count, age and distribution of files in a directory. These files being, for me, the pictures I still have to sort, cull and process, because I never have enough free time to clear out the backlog. The script is kind of designed to work together with Corydalis, but since it doesnāt care about file content, it can also double (easily) as simple āfile count/age exporterā.
And to my surprise, writing in Rust is soo pleasant, that the feature list is greater than the original Python script, and - compared to that untested script - Iāve rather easily achieved a very high coverage ratio. Rust has multiple types of tests, and the combination allows getting pretty down to details on testing:
- region coverage: >80%
- function coverage: >89% (so close here!)
- line coverage: >95%
I had to combine a (large) number of testing crates to get it
expressive enough, but it was worth the effort. The last find from
yesterday, assert_cmd, is
excellent to describe testing/assertion in Rust itself, rather than
via a separate, new DSL, like I was using shelltest for, in Haskell.
To some extent, I feel like I found the missing arrow in the quiver. Haskell is good, quite very good for some type of workloads, but of course not all, and Rust complements that very nicely, with lots of overlap (as expected). Python can fill in any quick-and-dirty scripting needed. And I just need to learn more frontend, specifically Typescript (the language, not referring to any specific libraries/frameworks), and Iāll be ready for AI to take over coding š ā¦
So, for now, Iāll need to split my free time coding between all of the above, and keep exercising my skills. But so glad to have found a good new language!