Hello,
This is my first post to this Rust community and I am happy to introduce myself and discuss what I love about this language, but first a little about myself.
I’m Alice, Fluffy to most, and she/her are my pronouns. I’m from the UK. I got a little boy who love like crazy. I’m Autistic, suffer from cPTSD and I also 🩷 Rust!!
Rust I feel appeals to Autistic people because of it’s focus on accuracy and correctness. It’s a common feeling people have about the language. But as the type of person I am I love it.
I began learning it in 2023, before this I was using C++. Rust showed me the importance of error’s as values and helped improve the quality of my code.
Currently I’m writing a game-engine as a hobby. The game-engine is a work in progress (WIP) and I have only just begun it’s development. Since the game-engine will natively support various platforms. To ensure consistency I’m writing all the platform specific code manually and building my own custom standard library for my engine, loosely based on the actual Rust standard library.
Right now I have the code in place to create/remove directories and to create, delete, write, read and set file pointer location. Convert UTF8 to UTF16 and output to the console in Unicode (Windows C API uses UTF16) and heap functions to get the process heap and create and delete memory dynamically.
Next will be the ‘config.json’ for which Serde will be used. Then the logger, and so on.
So it is a WIP but it’s fun and given my conditions allows me to do what I love, writing Rust code.
Thanks for reading!!
Alice 🏳️⚧️
Next will be the ‘config.json’ for which Serde will be used.
Have you considered TOML instead? For files that need to be edited and read by humans it tends to be better than JSON due to the easier ability to split it over lines and add comments in between.
I have considered it in the past but JSON feels like the standard. But TOML could be an option. I might try to see which I like better
TOML is a terrible format. It is anything but obvious, especially when you have more than one level of nesting.
It is pretty annoying that there isn’t an obvious format that Serde supports to use though:
- YAML is an awful format (worse than TOML in many ways). serde-yaml is abandoned anyway.
- serde_json is great but dtonlay refuses to support comments/trailing commas
- serde_json5 is an option but JSON5 doesn’t have good IDE support.
I would probably go with either RON or one of the forks of serde_json that adds support for comments. I think there’s serde_jsonc and serde_jsonc2 maybe.
JSON5 is seriously what I feel JSON should’ve been. Comments, trailing commas, hex numbers, etc.
The thing is, for many use-cases, you care about the configuration language being widely understood/supported. And TOML hit kind of a local maximum, where it’s capable enough for most configuration needs, while being basically INI++. Presumably, you could even fork an INI parser for building a TOML parser, which makes it easy to adopt in many ecosystems.
TOML… nesting
If you’re doing a lot of nesting in your config, you should rethink your config. Config should be pretty flat.
If you’re putting data in TOML, you’re doing it wrong. If your config needs to include data, IMO it should just reference the data in a separate file that uses a proper data format (e.g. JSON).
TOML rocks precisely because it nudges you into making simpler configs. Nesting is inherently hard to read (see endless debates over indentation standards), and TOML sidesteps the whole problem elegantly, forcing you to think about whether you actually need it. In most cases you don’t, and when you do, it’s possible and not unreasonable.
Config should be pretty flat.
Why? I don’t see any reason for that.
TOML rocks precisely because it nudges you into making simpler configs.
This is just “you’re holding it wrong”.
In most cases you don’t, and when you do, it’s possible and not unreasonable.
Really. Here’s the first Gitlab CI example I could find:
lintDebug: interruptible: true stage: build script: - ./gradlew -Pci --console=plain :app:lintDebug -PbuildDir=lint artifacts: paths: - app/lint/reports/lint-results-debug.html expose_as: "lint-report" when: always
Let’s see the TOML:
[lintDebug] interruptible = true stage = "build" script = [ "./gradlew -Pci --console=plain :app:lintDebug -PbuildDir=lint" ] [lintDebug.artifacts] paths = [ "app/lint/reports/lint-results-debug.html" ] expose_as = "lint-report" when = "always"
Gross. The tool I used to convert even added extra indentation because otherwise it is unclear.
IMO JSON5 is the best:
{ lintDebug: { interruptible: true, stage: 'build', script: [ './gradlew -Pci --console=plain :app:lintDebug -PbuildDir=lint', ], artifacts: { paths: [ 'app/lint/reports/lint-results-debug.html', ], expose_as: 'lint-report', when: 'always', }, }, }
This is much clearer than TOML and less ambiguous than YAML. It could do without the outer
{}
, but overall it’s still better.Gross
The only gross thing I see is your indentation. I find the TOML to be more readable honestly.
In a config format, you want to group top level fields together, and it enforces that. Your example could be like this instead:
lintDebug: stage: build script: - ./gradlew -Pci --console=plain :app:lintDebug -PbuildDir=lint artifacts: paths: - app/lint/reports/lint-results-debug.html expose_as: "lint-report" when: always interruptible: true
It’s easy to tack stuff on like that, and it’s also easy to miss when quickly scanning, especially if you’re using 2-space indenting.
The TOML would be the same in either case, because it forces related fields to be grouped separately from child groups.
The main thing I don’t like about TOML is yellow table groups:
[[thing]] [[thing]]
I rarely see it in the wild though.
IMO JSON5 is the best:
It’s trying to do two things at once, and ends up doing neither very well. The extra curly braces are there because it’s trying to be a data format, and it’s bad as a data format because it’s not universal. We’ve standardized on JSON as a community for data, and that’s unlikely to change, so we shouldn’t try to “fix” it just for configs. IMO, either use JSON or a config-specific format.
I agree that YAML is plain awful. It shouldn’t exist.
XML is awful for anything outside of markup, and even then I think we can do better. I’m partial to Lua tables:
{x=10, y=45, "one", "two", "three"}
A typical HTML drop down menu {pulled from MDN):
<select name="pets" id="pet-select"> <option value="">--Please choose an option--</option> <option value="hamster">Hamster</option> <option value="parrot">Parrot</option> </select>
Could be like this in a Lua table:
{ node="select", name="pets", id="pet-select", {value="", "--Please choose an option--"}, {value="hamster", "Hamster"}, {value="parrot", "Pattot"}, }
Or maybe something a little different with the node type outside:
select{ name="pets", id="pet-select", option{value="", "--Please choose an option--"}, option{value="hamster", "Hamster"}, option{value="parrot", "Parrot"}, }
I think QT Quick uses something like that, but it’s been years since I played with it.
Personally, I’ve found excessive nesting to be problematic, because each level of nesting is an invariant, which might not fit your application anymore as it evolves.
For example, let’s say you’ve got “application.logging.enabled” as one flag.
Then you decide to introduce more extensive telemetry in the next release. So, if you go with ‘proper’ nesting, you’d have to call the flags for that:- application.telemetry.metrics.enabled
- application.telemetry.tracing.enabled
Theoretically, you’d have to move the logging config also under that nesting level, but that would break existing configs.
Obviously, this example is relatively easy to avoid by not introducing the nesting in that case.
But you can find other examples where you might have called that level one way and then later ended up introducing a bunch of configs under a different name, where it could’ve also been under. That will confuse end users who try to configure your application and also might make it just more difficult to remember or guess config names in general, when you also have to guess the intermediate levels.
glhf
Hello (sadly i relate to the cptsd n autism xd) and good luck with your project!
Rust can be a big PITA but ultimately i love it. Are you using wgpu as the backend? You should look it up, very cool. And as taladar said you should consider using TOML instead.
I have came accress WGPU. But my goal is to write as close to the OS and other API’s as possible. So i will probably go with Vulkan natively but thanks for the idea!!
Good for you!
I will agree that rust is a very satisfying language to write in. Lol.
For me it’s either go full rust aiming for correctness and robustness, or go full python aiming for duck typing and flexibility.
This all being said Gleam and Scala are also quite beautiful to me at least.
At my company, we picked an awful middle-ground: write Python like it’s Java, but don’t use correct type hints.
In our project:
- everything is a static class -
from a.b.c import C; C.thing()
instead offrom a.b.c import thing; thing()
orimport a.b.c; c.thing()
- use declarative dependency injection frameworks instead of duck typing
- force rigid class structures, but don’t do typing properly, and don’t necessarily keep the calling signature of the base class methods (i.e. break SOLID all over the place)
- use decorators to hide magic (i.e. implicit is better than implicit and other Zen of Python violations)
We get the worst of OOP and functional styles with none of the benefits by abusing Python’s flexibility.
But hey, I guess it works…
That’s terrible. I noticed that there is a certain type of programmer. Normally they went to Uni in the late 90s or early 2000s and their entire education was focused around Java.
They went out to get a job writing “Enterprise” Java for 10 years.
This programmer then tries to learn another language, but because all they know is Java everything ends up looking like and structured as Java.
I’ve seen this developer over and over again. It’s possible for them to overcome Java Brain but it’s rare.
I bet one of these types of programmers is or was the technical lead of that project.
I noticed something similar when I worked on a Scala project in the past. Yes I know we’re on the JVM, but we don’t need a data deconstructor factory when flatmap is right there.
Nailed it.
We outsourced the project at the start. My company picked the tech stack (Python + Postgres, pretty sensible) and the consultancy company separately, so the consultant re-skilled to deliver the project. They were Java devs, so the whole project looks like Java.
The code was massive when I joined, and now it’s infeasible for me to fix, even if I can get buy-in.
The worst part is that we decided to go async on new microservices, so they pulled in different Java BS in the name of “quality.”
- everything is a static class -
I’ve been meaning to try Gleam, I hear really good things about it. Maybe I can make a Gleam-like scripting language for my game-engine.
That would be interesting.
Have you considered using LUA or perhaps scheme?
I know helix is likely to use scheme as its future scripting language. It’s quite robust and there is an excellent library to embed it.
I only mention LUA because it’s like the goto embedded scripting language in my book.