The Untouched Goldmine of F#
48 comments
·January 12, 2025jamalaramala
fluoridation
Same-process modularization should solve that problem. Splitting the architecture into separate processes means a crash in one module doesn't propagate to other modules, but the whole system still breaks down if the process is essential.
zbobet2012
What same-process modularization does _not_ solve is independent lifecycle management. In a monolithic system your change rate often becomes tied to your slowest, most bug prone module or team. If you've some integration test that bakes some piece of important code for 48 hours, you can only make a change _everywhere_ else every 48 hours.
Now sometimes folks (the Windows team famously did this) build systems which identify which tests to run based on changes that occur in the codebase, but that's _not_ easy.
jamalaramala
Not all bugs are crashes. It could be a copy text or price calculation.
thfuran
A system can break without crashing. Incorrectly processing financial transactions due to a bug in price calculations would be a good example of a serious breakage that may well be worse than crashing.
sebazzz
> With microservices, each team becomes responsible for one part, and (as long as they keep their SLAs) they can't break each other's code.
Well, they still can and do break someone else's code. It is just breaking someones code with more steps in between. And who is at fault isn't necessarily as clear cut.
lolinder
Yes. And the little-understood reason why microservices became the answer is that the tools for defining clear ownership boundaries in a single service were (and to an extent still are) lacking.
Defining a clear public contract in most existing programming languages that cannot be violated by a lazy programmer is tricky. Java's package-private should be the answer but is too blunt and ends up requiring a team to make things public so they can use them. C# has a bit more control that might make clear contracts possible (I haven't tried), but not everyone wants to use it because Microsoft. And most of the popular languages at the time of the rise of microservices had basically no mechanism for defining public versus private interfaces.
Microservices allow you to do that in any language— your HTTP API is your public interface. Anything inside of that is physically impossible to access without adding an explicit build-time dependency, which would be trivial to catch in code review.
Yes, code review could theoretically solve the same problems, but in practice no organization is that disciplined. You need tooling, and those tools didn't exist yet when microservices came to be. We're seeing them develop now (Rust's visibility modifiers are very powerful, and monorepo tools are starting to develop lints that can impose module boundaries on languages that lack them natively), and it's not a coincidence that around the same time we're seeing a lot of people moving back to monoliths.
breckenedge
Sounds like extra bureaucracy.
jamalaramala
It definitely is; but this bureaucracy can be useful when the company has thousands of developers.
SideburnsOfDoom
Yes and no. Would you rather ask for deployment of a microservice owned by a team of 10 people, or a monolith that has changes from a department of 100 people. Which deployment do you think has "extra bureaucracy"? Which one do you think can be done this week?
There are co-ordination costs to microservices. But there are also independence benefits.
mangodrunk
Having micro services doesn’t protect against affecting another team’s code/service. Think of a service changing some behavior or contract on either side of your service. Are you thinking of a person changing the actual code of another team? That can be solved without micro services.
ciupicri
Yet some companies use monorepos, so everybody has access to everything.
SideburnsOfDoom
> With microservices, each team becomes responsible for one part, and (as long as they keep their SLAs) they can't break each other's code.
When done well, microservices work as a unit of deployment. The team that owns the service code deploys it on their own pace. Sure there is process to follow, but it involves 10 people in that microservice team not 100 people for the monolithic app. This allow for smaller batches and independence of development.
boxed
> It may not have the information on the filename or the line of code like the ordinary stack traces, but these are useless information anyway.
That is the weirdest and most crazy thing I've read in years.
twic
The whole post feels like it came from a parallel universe. People adopted microservices because stack traces are too long! Exposing the implementation details of a function in its type signature is good! The most meaningful way to specify a function is by how it can fail (this one is fun, though)!
antigeox
In soviet F# the monads... uh....
johnnyjeans
the endofunctors operate on the category of you!
arwhatever
No doubt it’s harder to write structured code and assertions against a string stacktrace, but as a human reading it, the information is immensely valuable.
stonemetal12
Depends on how your app is structured, but it very well can be.
badvalue = dobadthing(); dostuffwith(badvalue);
Depending on the indirection between dobadthing() and the place that chokes on badvalue the stack trace can be completely worthless. It is one of the reasons people hated the original Spring. Claimed it decoupled stuff but all it really did was obfuscate.
I have seen similar claims made about Clojure. Bad data coming from somewhere, but no way to trace it through the system. Sure you can filter bad data in to the garbage can, but that just masks the issue not fix it.
cies
Yeah, that's also my main problem with the article. Otherwise great to prmote TSTs.
I think the TSTs should also contain the stacktrace. Not or X or Y, but both X and Y.
jitl
So… if I ever decide I want to change the call stack of a lower level function… I’m going to break types that all my callers and callers’ callers and callers’ callers’ callers’ are depending on? Like, my call stack is enshrined in a type so that to change it means a refactor all the way up to main()?
agentultra
Is TST essentially the validation monad[0]?
Glad to see more folks finding and appreciating FP languages like F#. It's good stuff. Scott Wlachin has a great site to discover more F# goodness[1].
[0]: https://hackage.haskell.org/package/monad-validate-1.3.0.0/d...
WorldMaker
It seems a step on the path to learning both the Either Monad and the Validation Monad, especially for someone coming from golang which desperately needs a proper Either Monad and essentially has baked in the worst of both imperative and FP worlds in the way everything returns as if there were an Either Monad but not enough useful combinators nor a semblance of do-notation/Computation Expressions exist.
Which is fun to see, really, because it is interesting watching someone rediscover FP principles from first practice.
talles
> It may not have the information on the filename or the line of code like the ordinary stack traces, but these are useless information anyway.
These are useless information?
kkukshtel
C# has a much awaited and active proposal to add DUs to C# so I suspect C# will also support this once live (and continue its legacy of plucking great features from F#).
https://github.com/dotnet/csharplang/blob/main/proposals/Typ...
garganzol
Another reason why companies moved to microservices is because it is easier to manage smaller projects. Services have stronger boundaries, they can be swapped with newer and alternative implementations, it is possible to combine the power of different technologies.
I came from the other side: I own a huge monolithic web behemoth which is almost unbearable to maintain now. It was built using a now outdated technology, but: it serves customers, it runs the business, it brings profits. Nevertheless, it is a huge pain to even try to change something in it. The project is at its dead end now, it is a one-trick pony who has become too old.
Nowadays I build newer parts using separate well-defined services. It is not textbook microservices, I would call it just services. Imagine building a mini-product that is not publicly available and only used internally. This kind of productized services work well enough to never look back at the fragile monolithic approach.
Nelkins
I upvoted this because I love F#, but I can't say I agree with the conclusions of the author.
seamossfet
God I remember having to deal with F# when I was an intern, what a nightmare. It's like the worst parts of Java and C# smashed into one esoteric language.
troelsSteegin
This is really about making stack traces easier to understand.
cies
I dont think so, TST and stacktraces are different.
Stacktraces show me call-stack to the point the error happened. Something TST does not always show: different call-stacks can result in the same (or very similar) TST-stacks.
It is possible to return the stacktrace as part of the TST error (not just an error message but also a stacktrace).
cies
> only Odin, F#, OCaml and Zig have [Tagged Unions or Discriminated Unions]
The good old missing sum type story.
Don't Rust, Haskell, Elm, Kotlin (with sealed classes), etc also have them?
squeegee_scream
I got to this part of the article and concluded it wasn’t worth my time, nor probably anyone else’s
1. Absurd claim that file names aren’t important in a stack trace 2. Claims they’re sharing a feature of F# that is never used, then later reveals they’ve been studying F# for 3 months 3. Then this statement about tagged unions
do_not_redeem
3 months, and he's already deep in acronym soup touting DDD, CDD, TDD, and TST. I'm getting major architecture astronaut vibes.
Dansvidania
bit of a weird article to find on the top of HN, to be sure..
But I felt similarly when I picked up Haskell coming from Ruby and Java, so I can understand the attitude. :D
johnnyjeans
Most languages have them, or allow them to be expressed. The name "Tagged Unions" literally comes from C.
AnimalMuppet
Did it? Didn't Pascal have tagged unions?
johnnyjeans
I looked it up and apparently it actually comes from Algol, but I digress. Point being sum types and their equivalents are found all over the place. I wouldn't be surprised to find out modern prologs have them.
jcmontx
I really like F#, but I really miss having a full fledged f#-native ORM
cies
No you dont. Sorry for the blunt answer.
ORMs are a bad idea, even when using OO langs: they make the simple queries slightly simpler (`Users.getById(id: Long)`), they do not help you for hard queries (ORM-using codebases of size usually have hard-SQL-queries "in strings").
Most users of FP langs know this and hence will not even try to implement ORMs.
Look into jOOQ, LINQ-method-syntax (or whatever it is called, without the funny SQLish syntactic sugar), SQLDelight or sqlx for non-ORM options that improve embedding SQL in general purpose langs.
jcmontx
Most of the time we do simple CRUD operations, I want this plumbing to be as straight forward as possible. Don't be patronising...
Dansvidania
I am always confused by this wish. If the operations are simple, why the need for ORM?
rafaelmn
I like the idea of F# type providers but last time I tried (two years ago) they were pretty shit compared to any mainstream ORM/alternative - has that improved ?
Also ORM for simple row selects is a straw-man - their use case is deserializing, updating relationship graphs where they hide away a bunch of code. Should that code be hidden is a different question - but pretending it's there to save you from typing SELECT * FROM foo is kind of disingenuous. Editing graph data structures in most FP languages is just not compatible with OOP approach hence no ORM.
> The question is: Why companies moved from monolithic to microservices? What do they try to avoid?
One of the main reasons why companies move from monoliths to microservices is to promote ownership and accountability in large codebases.
In a monolith where everyone owns the code, developers can break each other's code.
With microservices, each team becomes responsible for one part, and (as long as they keep their SLAs) they can't break each other's code.
When something fails it's easier to identify who needs to fix what.
Microservices don't make much sense for small teams if they don't need or don't have the headcount to split responsibilities.