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

Ensure MaybeMasked doesn't unwrap if it contains any #12204

Merged
merged 11 commits into from
Dec 12, 2024
5 changes: 5 additions & 0 deletions .changeset/friendly-papayas-breathe.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@apollo/client": patch
---

Fix `Unmasked` unwrapping tuple types into an array of their subtypes.
5 changes: 5 additions & 0 deletions .changeset/three-pandas-smash.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@apollo/client": patch
---

Ensure `MaybeMasked` does not try to unwrap the type as `Unmasked` if the type contains `any`.
36 changes: 35 additions & 1 deletion src/masking/__benches__/types.bench.ts
Original file line number Diff line number Diff line change
Expand Up @@ -299,7 +299,7 @@ test("MaybeMasked handles odd types", (prefix) => {

bench(prefix + "unknown instantiations", () => {
attest<unknown, MaybeMasked<unknown>>();
}).types([52, "instantiations"]);
}).types([54, "instantiations"]);
bench(prefix + "unknown functionality", () => {
expectTypeOf<MaybeMasked<unknown>>().toBeUnknown();
});
Expand Down Expand Up @@ -464,3 +464,37 @@ test("base type, multiple fragments on sub-types", (prefix) => {
}>();
});
});

test("does not detect `$fragmentRefs` if type contains `any`", (prefix) => {
interface Source {
foo: { bar: any[] };
}

bench(prefix + "instantiations", () => {
return {} as MaybeMasked<Source>;
}).types([6, "instantiations"]);

bench(prefix + "functionality", () => {
const x = {} as MaybeMasked<Source>;

expectTypeOf(x).branded.toEqualTypeOf<Source>();
});
});
jerelmiller marked this conversation as resolved.
Show resolved Hide resolved

test("leaves tuples alone", (prefix) => {
interface Source {
coords: [long: number, lat: number];
}

bench(prefix + "instantiations", () => {
return {} as Unmasked<Source>;
}).types([5, "instantiations"]);

bench(prefix + "functionality", () => {
const x = {} as Unmasked<Source>;

expectTypeOf(x).branded.toEqualTypeOf<{
coords: [long: number, lat: number];
}>();
});
});
4 changes: 2 additions & 2 deletions src/masking/internal/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@ export type UnwrapFragmentRefs<TData> =
>
>
>
: TData extends Array<infer TItem> ? Array<UnwrapFragmentRefs<TItem>>
Copy link
Member

Choose a reason for hiding this comment

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

👍

: TData extends object ?
{
[K in keyof TData]: UnwrapFragmentRefs<TData[K]>;
Expand Down Expand Up @@ -184,7 +183,8 @@ export type RemoveFragmentName<T> =
T extends any ? Omit<T, " $fragmentName"> : T;

export type ContainsFragmentsRefs<TData> =
TData extends object ?
true extends IsAny<TData> ? false
: TData extends object ?
" $fragmentRefs" extends keyof TData ?
true
: ContainsFragmentsRefs<TData[keyof TData]>
Expand Down
Loading