Search code examples
typescripttype-inferencetemplate-literals

Template literal inference and narrowing in Typescript


I have the following code:

export type EntityType = "foo" | "bar";
export interface Foo {
  id: `foo-${string}`;
}
export interface Bar {
  id: `bar-${string}`;
}

export type Ent<T extends EntityType> = T extends "foo" ? Foo : Bar;
export const resolve = <T extends EntityType>(id: `${T}-${string}`): Ent<T> => {
  if (id.startsWith("foo-")) {
    return { id: "foo-a" };
  }
  if (id.startsWith("bar-")) {
    return { id: "bar-a" };
  }
  throw new Error(`Unsupported entity type ${id}`);
};

The return statements are throwing an error that says: Type '{ id: "foo-a"; }' is not assignable to type 'Ent<T>'

This happens because TS can't infer that .startsWith("foo-") actually matches the Foo interface, which is understandable.

Is there a way to hint to the compiler so it can correctly infer and narrow down the expected return type? Right now this only works if I cast to any:

if (id.startsWith("foo-")) {
  return { id: "foo-a" } as any;
}

Solution

  • Perhaps you can try

    if (id.startsWith("foo-")) {
        return { id: "foo-a" } as Foo as Ent<T>;
    }
    if (id.startsWith("bar-")) {
        return { id: "bar-a" } as Bar as Ent<T>;
    }