mirror of
https://github.com/hex248/ob248.com.git
synced 2026-02-08 02:33:02 +00:00
244 lines
6.8 KiB
TypeScript
244 lines
6.8 KiB
TypeScript
// @ts-ignore TS6133
|
|
import { expect, test } from "vitest";
|
|
|
|
import * as z from "zod/v3";
|
|
import { ZodNullable, ZodOptional } from "zod/v3";
|
|
import { util } from "../helpers/util.js";
|
|
|
|
const nested = z.object({
|
|
name: z.string(),
|
|
age: z.number(),
|
|
outer: z.object({
|
|
inner: z.string(),
|
|
}),
|
|
array: z.array(z.object({ asdf: z.string() })),
|
|
});
|
|
|
|
test("shallow inference", () => {
|
|
const shallow = nested.partial();
|
|
type shallow = z.infer<typeof shallow>;
|
|
type correct = {
|
|
name?: string | undefined;
|
|
age?: number | undefined;
|
|
outer?: { inner: string } | undefined;
|
|
array?: { asdf: string }[];
|
|
};
|
|
util.assertEqual<shallow, correct>(true);
|
|
});
|
|
|
|
test("shallow partial parse", () => {
|
|
const shallow = nested.partial();
|
|
shallow.parse({});
|
|
shallow.parse({
|
|
name: "asdf",
|
|
age: 23143,
|
|
});
|
|
});
|
|
|
|
test("deep partial inference", () => {
|
|
const deep = nested.deepPartial();
|
|
const asdf = deep.shape.array.unwrap().element.shape.asdf.unwrap();
|
|
asdf.parse("asdf");
|
|
type deep = z.infer<typeof deep>;
|
|
type correct = {
|
|
array?: { asdf?: string }[];
|
|
name?: string | undefined;
|
|
age?: number | undefined;
|
|
outer?: { inner?: string | undefined } | undefined;
|
|
};
|
|
|
|
util.assertEqual<deep, correct>(true);
|
|
});
|
|
|
|
test("deep partial parse", () => {
|
|
const deep = nested.deepPartial();
|
|
|
|
expect(deep.shape.name instanceof z.ZodOptional).toBe(true);
|
|
expect(deep.shape.outer instanceof z.ZodOptional).toBe(true);
|
|
expect(deep.shape.outer._def.innerType instanceof z.ZodObject).toBe(true);
|
|
expect(deep.shape.outer._def.innerType.shape.inner instanceof z.ZodOptional).toBe(true);
|
|
expect(deep.shape.outer._def.innerType.shape.inner._def.innerType instanceof z.ZodString).toBe(true);
|
|
});
|
|
|
|
test("deep partial runtime tests", () => {
|
|
const deep = nested.deepPartial();
|
|
deep.parse({});
|
|
deep.parse({
|
|
outer: {},
|
|
});
|
|
deep.parse({
|
|
name: "asdf",
|
|
age: 23143,
|
|
outer: {
|
|
inner: "adsf",
|
|
},
|
|
});
|
|
});
|
|
|
|
test("deep partial optional/nullable", () => {
|
|
const schema = z
|
|
.object({
|
|
name: z.string().optional(),
|
|
age: z.number().nullable(),
|
|
})
|
|
.deepPartial();
|
|
|
|
expect(schema.shape.name.unwrap()).toBeInstanceOf(ZodOptional);
|
|
expect(schema.shape.age.unwrap()).toBeInstanceOf(ZodNullable);
|
|
});
|
|
|
|
test("deep partial tuple", () => {
|
|
const schema = z
|
|
.object({
|
|
tuple: z.tuple([
|
|
z.object({
|
|
name: z.string().optional(),
|
|
age: z.number().nullable(),
|
|
}),
|
|
]),
|
|
})
|
|
.deepPartial();
|
|
|
|
expect(schema.shape.tuple.unwrap().items[0].shape.name).toBeInstanceOf(ZodOptional);
|
|
});
|
|
|
|
test("deep partial inference", () => {
|
|
const mySchema = z.object({
|
|
name: z.string(),
|
|
array: z.array(z.object({ asdf: z.string() })),
|
|
tuple: z.tuple([z.object({ value: z.string() })]),
|
|
});
|
|
|
|
const partialed = mySchema.deepPartial();
|
|
type partialed = z.infer<typeof partialed>;
|
|
type expected = {
|
|
name?: string | undefined;
|
|
array?:
|
|
| {
|
|
asdf?: string | undefined;
|
|
}[]
|
|
| undefined;
|
|
tuple?: [{ value?: string }] | undefined;
|
|
};
|
|
util.assertEqual<expected, partialed>(true);
|
|
});
|
|
|
|
test("required", () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
nullableField: z.number().nullable(),
|
|
nullishField: z.string().nullish(),
|
|
});
|
|
|
|
const requiredObject = object.required();
|
|
expect(requiredObject.shape.name).toBeInstanceOf(z.ZodString);
|
|
expect(requiredObject.shape.age).toBeInstanceOf(z.ZodNumber);
|
|
expect(requiredObject.shape.field).toBeInstanceOf(z.ZodDefault);
|
|
expect(requiredObject.shape.nullableField).toBeInstanceOf(z.ZodNullable);
|
|
expect(requiredObject.shape.nullishField).toBeInstanceOf(z.ZodNullable);
|
|
});
|
|
|
|
test("required inference", () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
nullableField: z.number().nullable(),
|
|
nullishField: z.string().nullish(),
|
|
});
|
|
|
|
const requiredObject = object.required();
|
|
|
|
type required = z.infer<typeof requiredObject>;
|
|
type expected = {
|
|
name: string;
|
|
age: number;
|
|
field: string;
|
|
nullableField: number | null;
|
|
nullishField: string | null;
|
|
};
|
|
util.assertEqual<expected, required>(true);
|
|
});
|
|
|
|
test("required with mask", () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
country: z.string().optional(),
|
|
});
|
|
|
|
const requiredObject = object.required({ age: true });
|
|
expect(requiredObject.shape.name).toBeInstanceOf(z.ZodString);
|
|
expect(requiredObject.shape.age).toBeInstanceOf(z.ZodNumber);
|
|
expect(requiredObject.shape.field).toBeInstanceOf(z.ZodDefault);
|
|
expect(requiredObject.shape.country).toBeInstanceOf(z.ZodOptional);
|
|
});
|
|
|
|
test("required with mask -- ignore falsy values", () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
country: z.string().optional(),
|
|
});
|
|
|
|
// @ts-expect-error
|
|
const requiredObject = object.required({ age: true, country: false });
|
|
expect(requiredObject.shape.name).toBeInstanceOf(z.ZodString);
|
|
expect(requiredObject.shape.age).toBeInstanceOf(z.ZodNumber);
|
|
expect(requiredObject.shape.field).toBeInstanceOf(z.ZodDefault);
|
|
expect(requiredObject.shape.country).toBeInstanceOf(z.ZodOptional);
|
|
});
|
|
|
|
test("partial with mask", async () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
country: z.string(),
|
|
});
|
|
|
|
const masked = object.partial({ age: true, field: true, name: true }).strict();
|
|
|
|
expect(masked.shape.name).toBeInstanceOf(z.ZodOptional);
|
|
expect(masked.shape.age).toBeInstanceOf(z.ZodOptional);
|
|
expect(masked.shape.field).toBeInstanceOf(z.ZodOptional);
|
|
expect(masked.shape.country).toBeInstanceOf(z.ZodString);
|
|
|
|
masked.parse({ country: "US" });
|
|
await masked.parseAsync({ country: "US" });
|
|
});
|
|
|
|
test("partial with mask -- ignore falsy values", async () => {
|
|
const object = z.object({
|
|
name: z.string(),
|
|
age: z.number().optional(),
|
|
field: z.string().optional().default("asdf"),
|
|
country: z.string(),
|
|
});
|
|
|
|
// @ts-expect-error
|
|
const masked = object.partial({ name: true, country: false }).strict();
|
|
|
|
expect(masked.shape.name).toBeInstanceOf(z.ZodOptional);
|
|
expect(masked.shape.age).toBeInstanceOf(z.ZodOptional);
|
|
expect(masked.shape.field).toBeInstanceOf(z.ZodDefault);
|
|
expect(masked.shape.country).toBeInstanceOf(z.ZodString);
|
|
|
|
masked.parse({ country: "US" });
|
|
await masked.parseAsync({ country: "US" });
|
|
});
|
|
|
|
test("deeppartial array", () => {
|
|
const schema = z.object({ array: z.string().array().min(42) }).deepPartial();
|
|
|
|
// works as expected
|
|
schema.parse({});
|
|
|
|
// should be false, but is true
|
|
expect(schema.safeParse({ array: [] }).success).toBe(false);
|
|
});
|