'Retrieve records of objects with the correct types

I have the following Typescript function that query a table in Postgresql;I pass a generic type and I expect to have an array of objects of the same type passed as input, but it doesn't:

export async function pgQuery<T>(
  query: string,
  postgreSqlClient: Client = postgreSqlClientGlobal
) {
  const log = extendLogger(_log, pgQuery);
  try {
    const res = await postgreSqlClient.query<T>(query);
    return res;
  } catch (error) {
    log.error({ error }, `PostgreSql query ERROR`);
    throw new Error(`PostgreSql query ERROR - ${error}`);
  }
}

The input type T is:

{country: string,
 age_min: number,
 age_max: number}

And in the table of the Postgresql DB I have records like this:

{
  loc_name: "Albania",
  age_min: 15,
  age_max: 64
}

but I have, as output of the executed query, object like this:

{country: string,
 age_min: string,
 age_max: string}

Example:

{
  loc_name: "Albania",
  age_min: "15",
  age_max: "64"
}

How to retrieve the correct types?
pg stands for: https://github.com/brianc/node-postgres



Solution 1:[1]

It depends on how postgreSqlClient.query<T>(query) is implemented.

When you interact with an external database, just typing the request as type T is not enough to guarantee that you will actually get a value of type T.

When you declare the type as T, Typescript will just trust you and assume that the type is T. However, since you are fetching an external resource, you cannot be sure at runtime that the value is actually of type T.

If you want to be 100% sure, you must implement some sort of runtime validation. Libraries like zod and io-ts are made for this purpose.

Solution 2:[2]

The fastest solution (I'm writing a "disposable code"...) is to cast the fields in the postgresql query (es.: select loc_name, age_min::int, age_max::int from table;); for more info see:
pg-promise returns integers as strings

Sources

This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.

Source: Stack Overflow

Solution Source
Solution 1 Sandro Maglione
Solution 2 user1