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

fix: ZodEffect in discriminatedUnion #2441

Conversation

cdaringe
Copy link

@cdaringe cdaringe commented May 17, 2023

Problem

.refine and .superRefined schemas cannot be used in discriminatedUnion, but should be able to be.

See #2440

Discussion

The fix is very simple.

  1. Support ZodEffects where in the input type is also a ZodObject. Add a union member that says "not only accept ZodObject, but ALSO accept ZodEffect<ZodObject>"
  2. When looking up the schema shape, do a very efficient check to see what type we are, and get the schema shape based on whether we are a ZodEffect or ZodObject

Add a test. Blamo. Easy!

There's a lot of discussion of "problems with z.discriminatedUnion". I see this as independent from those critiques. This, I perceive, was just a basic oversight in the typing of the original implementation. The fix is simple and carries minimal overhead.

@netlify
Copy link

netlify bot commented May 17, 2023

Deploy Preview for guileless-rolypoly-866f8a ready!

Built without sensitive environment variables

Name Link
🔨 Latest commit ff8d2d3
🔍 Latest deploy log https://app.netlify.com/sites/guileless-rolypoly-866f8a/deploys/646e42bc1d33830008378e61
😎 Deploy Preview https://deploy-preview-2441--guileless-rolypoly-866f8a.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site settings.

@@ -2882,13 +2882,20 @@ const getDiscriminator = <T extends ZodTypeAny>(
}
};

export type ZodDiscriminatedUnionOption<Discriminator extends string> =
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is a bad diff 😆 . this type is still present on L2895


export type ZodDiscriminatedUnionOption<Discriminator extends string> =
| ZodDiscriminatedUnionOptionObject<Discriminator>
| ZodDiscriminatedUnionOptionEffectsObject<Discriminator>;
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the creation of this union is the core change in this PR!

@@ -2985,7 +2992,8 @@ export class ZodDiscriminatedUnion<

// try {
for (const type of options) {
const discriminatorValues = getDiscriminator(type.shape[discriminator]);
const shape = "shape" in type ? type.shape : type.sourceType().shape;
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is shape lookup is the only runtime change! nice n simple

@cdaringe cdaringe changed the title fix: support ZodEffect in discriminatedUnion fix: ZodEffect in discriminatedUnion May 18, 2023
@cdaringe
Copy link
Author

hey friends, this is a pretty simple PR to review. let me know if there's anything i can do to reduce your review burden. i annotated the changes, for example, as a mechanism to make your review more straight forward. i know maintenance is a huge effort, so I want to be courteous of your precious time!

@cdaringe
Copy link
Author

hey @colinhacks! hope all is well. this PR has been sitting for a bit, and is a pretty staightforward fix. i don't want to irk anyone on the team, but can i expect feedback anytime soon?

@cdaringe
Copy link
Author

Hey all, friendly bump. Thanks for your time and support

1 similar comment
@cdaringe
Copy link
Author

cdaringe commented Jul 6, 2023

Hey all, friendly bump. Thanks for your time and support

@hfalk
Copy link

hfalk commented Oct 16, 2023

@colinhacks do we have any update on this? Would help us a lot!

@ebramanti
Copy link

I have run into this issue too, would be really helpful if this change could land soon!

@k0rnl
Copy link

k0rnl commented Jan 21, 2024

I have run into this issue too, would be really helpful if this change could land soon ! 🙏

@colinhacks
Copy link
Owner

This is a well-made PR, but the root problem here is that refinements should be stored inside the schema instead of wrapping the schema with ZodEffects. I'm hesitant to merge because this adds more complexity to a feature (discriminated union) that likely won't exist in Zod 4. The Zod 4 implementation will avoid the fundamental messiness of discriminatedUnion and it's right around the corner, so I'm gonna close. Thanks @cdaringe, this was a great PR.

@ms-ati
Copy link

ms-ati commented Mar 29, 2025

Almost a year later, checking in?

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.

6 participants