Data types
When talking about data types in Kysely we need to make a distinction between the two kinds of types:
- Typescript types
- Runtime javascript types
Typescript types
In Kysely, you only define typescript types for your tables and columns. Since typescript is entirely a compile-time concept, typescript types can't affect runtime javascript types. If you define your column to be a string
in typescript but the database returns a number
, the runtime type doesn't magically change to string
. You'll see a string
in the typescript code, but observe a number when you run the program.
It's up to you to select correct typescript types for your columns based on what the driver returns.
Runtime javascript types
The database driver, such as pg
or mysql2
, decides the runtime javascript types the queries return. Kysely never touches the runtime types the driver returns. In fact, Kysely doesn't touch the data returned by the driver in any way. It simply executes the query and returns whatever the driver returns. An exception to this rule is when you use a plugin like CamelCasePlugin
, in which case Kysely does change the column names.
You need to read the underlying driver's documentation or otherwise figure out what the driver returns and then align the typescript types to match them.
Configuring runtime javascript types
Most drivers provide a way to change the returned types. For example pg
returns bigint
and numeric
types as strings by default, but often you want to configure it to return numbers instead.
Postgres
When using the pg
driver, you can use the pg-types package to configure the types. For example here's how you'd configure the bigint
to be returned as a number:
import { Kysely, PostgresDialect } from 'kysely'
import * as pg from 'pg'
const int8TypeId = 20
// Map int8 to number.
pg.types.setTypeParser(int8TypeId, (val) => {
return parseInt(val, 10)
})
export const db = new Kysely<Database>({
dialect: new PostgresDialect({
pool: new pg.Pool(config),
}),
})
See the documentation here on how to figure out the correct type id.
MySQL
When using the mysql2
driver, you an use the typeCast pool property.
For example here's how you'd map tinyint(1)
to a boolean:
import { Kysely, MysqlDialect } from 'kysely'
import { createPool } from 'mysql2'
export const db = new Kysely<Database>({
dialect: new MysqlDialect({
pool: createPool({
...config,
// Map tinyint(1) to boolean
typeCast(field, next) {
if (field.type === 'TINY' && field.length === 1) {
return field.string() === '1'
} else {
return next()
}
},
}),
}),
})
Type generators
There are third-party type generators such as kysely-codegen and kanel-kysely that automatically generate typescript types based on the database schema. Find out more at "Generating types".
If these tools generate a type that doesn't match the runtime type you observe, please refer to their documentation or open an issue in their github. Kysely has no control over these libraries.