Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Custom result type with typed errors #999

Merged
merged 2 commits into from
Mar 25, 2025

Conversation

Kailari
Copy link
Contributor

@Kailari Kailari commented Mar 25, 2025

Melko pitkälle viety Proof-of-Concept Result-tyypistä vahvasti tyypitetyillä virheillä.

Toteutettuna TypedResult<Value, Error> -tyyppi ja sille joitain yleisimmin tarvittuja toimintoja. Toteutus muistuttaa rajapinnaltaan vahvasti standardikirjaston Result-tyyppiä, sillä erotuksella, että Success- ja Failure-variaatiot on toteutettu aliluokkina ja itse TypedResult on sealed class. Tämä mahdollistaa mm. exhaustive when-lauseet Success ja Failure-luokkia käyttäen. Tämän hyödyllisyys on tosin melko rajallista (ja/tai johtaa verboosiin koodiin), sillä Kotlin ei mahdollista mm. destrukturointia when lauseiden osana, joskin utility-metodien ja itse TypedResult-luokan toteutus oli tämän ansiosta melko helppoa.

Olennaisimpina muutoksina:

  • kaikki as Error.Validation -tyyliset tyyppimuunnokset putoavat pois
  • .flatMap-kikkailua varten Error luokalle piti lisätä Error.Oppija-aliluokka, kaikkia toOppija/getOppijanumero-putkesta syntyviä virheitä varten

Jälkimmäinen muutos on lähinnä sen demonstrointia varten, että tällainen on ylipäätään mahdollista. Tämä on lähimmäs tyyppi-unioneja (esim. OppijaValidationFailure | OppijanumeroException) mitä Kotlinin rajoitteiden puitteissa sain aikaiseksi.

Improvement/adjustment over stdlib Result<T>, by providing a way to
strongly type the error.

This has the upside of making handling the errors easier (no need to
pattern match against the type of a generic Throwable). Additionally,
enforcing knowing your error conditions might reduce the possibilities
of making bad decisions in life.

However, Kotlin does not support type unions in any meaningful way. This
makes implementing something akin to a 'flatMap' (fold/map which merges
failures produced by the transform function) very difficult. This was
not previously an issue, as everything was just a generic Throwable,
thus easily convertible, and as error handling required pattern matching
the type via when-expressions anyway. With strong error types, nesting
result expressions becomes a bit cumbersome.

This can be somewhat alleviated with .fold trickery for our current
use-cases, but might become an issue in more complex cases. I explored a
number of possible approaches with variations of using
when-exporessions, fold, flatmap-like functions, etc. and this current
iteration is the best I could do.

I hope most of the shortcomings of this approach can be eliminated with
well thought utility functions for any common pain-points that pop up.
Handling the errors without '.fold's makes things a bit more
straightforward, albeit a bit more verbose. For overall readability and
simplicity, this is good.
@Kailari Kailari force-pushed the refact/custom-typed-result branch from 1403b97 to 0bcd50f Compare March 25, 2025 13:44
@Kailari
Copy link
Contributor Author

Kailari commented Mar 25, 2025

Yksinkertaistin hieman oppijanumerohaun virheitä. .flatMap ja Error.Oppija virheluokat olivat hieman turhan raskas ratkaisu tähän tilanteeseen.

@Kailari Kailari merged commit ed0c3b8 into refact/stdlib-result Mar 25, 2025
12 checks passed
@Kailari Kailari deleted the refact/custom-typed-result branch March 25, 2025 13:53
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

1 participant