A
arktype•5w ago
Kuba

Throw on failed validation

Does Arktype throws on failed validation? I saw that there's a onFail in docs, so I've configured it and imported it before artype itself in place of use, but still no error was thrown.
// arktype-conf.ts
import { configure } from "arktype";

const config = configure({
onFail: (errors) => errors.throw(),
});

declare global {
interface ArkEnv {
onFail: typeof config.onFail;
}
}
// arktype-conf.ts
import { configure } from "arktype";

const config = configure({
onFail: (errors) => errors.throw(),
});

declare global {
interface ArkEnv {
onFail: typeof config.onFail;
}
}
// example.ts
import "./arktype-conf";
import { type } from "arktype";

const User = type({
name: "string",
age: "number",
});

// const user = User({ name: "John", age: 30 });
const user = User({ name: "John" });
// exits gracefully, but user contains an error
// no error throw happens
// example.ts
import "./arktype-conf";
import { type } from "arktype";

const User = type({
name: "string",
age: "number",
});

// const user = User({ name: "John", age: 30 });
const user = User({ name: "John" });
// exits gracefully, but user contains an error
// no error throw happens
I expected this config to change the behavior of arktype to throw instead. Is it possible or is it by design? Not that I need it, was just curious if this is possible or I did/understood something wrong.
ArkType Docs
Optimized runtime validation for TypeScript syntax
32 Replies
ssalbdivad
ssalbdivad•5w ago
Hi! When you say user contains an error, is user equal to { name: "John" } or is it an ArkErrors array
Kuba
KubaOP•5w ago
It is an ArkErrors array And hello! 👋
ssalbdivad
ssalbdivad•5w ago
Oh the issue might be that you need to import from the config entrypoint: import { configure } from "arktype/config";
Kuba
KubaOP•5w ago
Sweet, now it threw
AggregateError: age must be a number (was missing)
// stacktrace
AggregateError: age must be a number (was missing)
// stacktrace
Completely missed the /config, thanks!
ssalbdivad
ssalbdivad•5w ago
No problem, glad it's working! It's a relatively new unadvertised feature so I'm glad you find it helpful 🙂 I actually added it so that I could have a way to configure match internally to throw by default when the user configures it that way
Kuba
KubaOP•5w ago
It's in the docs, so we can say it's advertised 😄 I was mostly trying out tanstack start and trying if it supports other validators for server functions. I recently watched your appearance on devtools-fm and I picked arktype out of curiosity, so I'm pretty fresh to the library (no idea what match is 😛 )
ssalbdivad
ssalbdivad•5w ago
Yes that is true! I just mean it is not a feature I've been posting about or featuring anywhere centrally other than if you read through that part of the config docs
ssalbdivad
ssalbdivad•5w ago
That podcast was a ton of fun! I actually mentioned match right at the end as an upcoming feature for pattern-matching: https://arktype.io/docs/blog/2.1
ArkType Docs
Optimized runtime validation for TypeScript syntax
Kuba
KubaOP•5w ago
Ah, that's alright. Sometimes you just can't post about everything and probably focus on most important stuff so that's nothing to worry about I would say That's what the docs are for 😄
ssalbdivad
ssalbdivad•5w ago
No need to rush into trying to understand everything at once though. Definitely interested to hear what your experience is like with tanstack with the new forms library just having been released. Well always a good feeling when people read those sections regardless 😊
Kuba
KubaOP•5w ago
It's a bit like "sharpening the axe before cutting the tree". It's just good to know what the library offers. Probably impossible to remember everything, but basic familiarization of capabilities never hurts
ssalbdivad
ssalbdivad•5w ago
I like the analogy! I don't think most users read through the docs before diving in, especially every section! Did you find the intro sections to be a good overview?
Kuba
KubaOP•5w ago
Yes, "Your First Type" was enough to get me started. I only skimmed through constraints, and completely skipped morphs as I didn't have a use for them yet The match.in().at() looks interesting. I've written something similar in my current project for narrowing down the type of objects within the array (all objects were discriminated) of react-hook-form to make my life easier. The match looks much simpler
ssalbdivad
ssalbdivad•5w ago
If you can really just write a pure switch for a discriminated union that will be very efficient and generally reasonable ergonomic in TS. I'd use a matcher especially if you have some groups of conditions or something like number > 0 you can validate inline. But honestly you can use it wherever, definitely a flexible API
Kuba
KubaOP•5w ago
By pure switch you mean the JS provided switch, and use discriminator as switch(discriminator), right?
ssalbdivad
ssalbdivad•5w ago
Yeah
Kuba
KubaOP•5w ago
Okay, I think I ended up with if else if, as I liked it more and didn't have many cases, but switch is also an option. But good to know that other options exist
ssalbdivad
ssalbdivad•5w ago
Yeah, that can work well too especially if the conditions vary a bit (e.g. are not all string literals) and a switch doesn't work as well
Kuba
KubaOP•5w ago
I haven't used tanstack form lib, as I'm mostly using react hook form, so can't share the experience
ssalbdivad
ssalbdivad•5w ago
Ahh okay. Well I think they've adopted standard-schema throughout so hopefully you have a good experience regardless
Kuba
KubaOP•5w ago
True. The Standard Schema is also supported by search params in router, and now I was checking how it behaves in the validators of the Start's server functions Speaking of Start, I'm heading back to testing it more. Thanks for the help once more, and for a chat too! 😄
ssalbdivad
ssalbdivad•5w ago
Likewise, best of luck!
Kuba
KubaOP•5w ago
Thanks!:)
TizzySaurus
TizzySaurus•5w ago
Fwiw, since it doesn't seem to have been mentioned (unless I missed it), this is what type.assert() is for (in your case, User.assert()). You don't actually need to mess with any config if all you want to do is throw the default errors
ssalbdivad
ssalbdivad•5w ago
There's a new top-level onFail config that changes the default top-level invocation behavior For domains where errors are genereally not recoverable So you never have to discriminate output
TizzySaurus
TizzySaurus•5w ago
But it's otherwise the exact same as .assert(), right?
ssalbdivad
ssalbdivad•5w ago
Yeah it just makes assert the default basically
TizzySaurus
TizzySaurus•5w ago
I personally would rather just use .assert() than start messing with config which is why I mentioned it, but maybe that's just me :Shrug:
ssalbdivad
ssalbdivad•5w ago
I can see the appeal if you're using it a lot, but yeah somewhat marginal to avoid the extra prop chain. I mostly added it because I needed to for matching anyways
TizzySaurus
TizzySaurus•5w ago
Yeah, fair enough. The main use case for me would be if I wanted to customise failure beyond just throwing errors (e.g. logging something or w/e)
ssalbdivad
ssalbdivad•5w ago
Oh yeah true you can do lots of stuff with it I18N potentially? I forgot about that haha I would use it if I was an env where I always wanted to throw I think. E.g. before tRPC adopted standard schema when it expected that API Feels a lot cleaner to pass all the types directly
TizzySaurus
TizzySaurus•5w ago
Fair enough. It's certainly nice to be able to customise this

Did you find this page helpful?