Knex is an SQL query builder for Node.js. This guide targets v0.13.0.

Getting started

Connect

require('knex')({
  client: 'pg',
  connection: 'postgres://user:pass@localhost:5432/dbname'
})

See: Connect

Create table

knex.schema.createTable('user', (table) => {
  table.increments('id')
  table.string('name')
  table.integer('age')
})
.then(() => ···)

See: Schema

Select

knex('users')
  .where({ email: 'hi@example.com' })
  .then(rows => ···)

See: Select

Insert

knex('users')
  .insert({ email: 'hi@example.com' })

See: Insert

Update

knex('users')
  .where({ id: 135 })
  .update({ email: 'hi@example.com' })

See: Update

Migrations

knex init
knex migrate:make migration_name
knex migrate:make migration_name -x ts # Generates a TypeScript migration file
knex migrate:latest
knex migrate:rollback

See: Migrations

Seeds

knex seed:make seed_name
knex seed:make seed_name -x ts # Generates a TypeScript seed file
knex seed:run # Runs all seed files
knex seed:run --specific=seed-filename.js # Runs a specific seed file

See: Seeds

Connect

Libraries

pg PostgreSQL
mysql MySQL or MariaDB
sqlite3 Sqlite3
mssql MSSQL

Install any of these packages along with knex.

See: Node.js installation

Connect via host

var knex = require('knex')({
  client: 'mysql',
  connection: {
    host: '127.0.0.1',
    user: 'your_database_user',
    password: 'your_database_password',
    database: 'myapp_test'
  },
  pool: { min: 0, max: 7 }
})

See: Initializing the library

Connect via URL

var pg = require('knex')({
  client: 'pg',
  connection: process.env.DATABASE_URL,
  searchPath: 'knex,public',
  pool: { min: 0, max: 7 }
})

Connect via Sqlite

var knex = require('knex')({
  client: 'sqlite3',
  connection: { filename: './mydb.sqlite' }
})

Select

Where

knex
  .from('books')
  .select('title', 'author', 'year')

Where

  .where('title', 'Hello')
  .where({ title: 'Hello' })
  .whereIn('id', [1, 2, 3])
  .whereNot(···)
  .whereNotIn('id', [1, 2, 3])

Where conditions

  .whereNull('updated_at')
  .whereNotNull(···)
  .whereExists('updated_at')
  .whereNotExists(···)
  .whereBetween('votes', [1, 100])
  .whereNotBetween(···)
  .whereRaw('id = ?', [1])

Where grouping

  .where(function () {
    this
      .where('id', 1)
      .orWhere('id', '>', 10)
  })

See: Where clauses

Join

knex('users')

Basic join

  .join('contacts', 'users.id', '=', 'contacts.id')
  .join('contacts', {'users.id': 'contacts.id'})

Strings

  .join('accounts', 'accounts.type', '=', knex.raw('?', ['admin']))

Directions

  .leftJoin(···)
  .leftOuterJoin(···)
  .rightJoin(···)
  .rightOuterJoin(···)
  .outerJoin(···)
  .fullOuterJoin(···)
  .crossJoin(···)

Raw

  .joinRaw('natural full join table1')

Grouping

  .join('accounts', function () {
    this
      .on('accounts.id', '=', 'users.account_id')
      .orOn('accounts.owner_id', '=', 'users.id')

      .onIn('accounts.id', [1, 2, 3, 5, 8])
      .onNotIn(···)

      .onNull('accounts.email')
      .onNotNull(···)

      .onExists(function () {
        this.select(···)
      })
      .onNotExists(···)
  })

See: Join methods

Others

knex('users')
  .distinct()

Group

  .groupBy('count')
  .groupByRaw('year WITH ROLLUP')

Order

  .orderBy('name', 'desc')
  .orderByRaw('name DESC')

Offset/limit

  .offset(10)
  .limit(20)

Having

  .having('count', '>', 100)
  .havingIn('count', [1, 100])

Union

  .union(function() {
    this.select(···)
  })
  .unionAll(···)

See: Query builder

Etc

knex('users')
  .pluck('id')
  .then(ids => { ··· })
knex('users')
  .first()
  .then(user => { ··· })

Booleans

  .count('active')
  .count('active as is_active')

Numbers

  .min('age')
  .max('age')
  .sum('age')
  .sumDistinct('age')
  .avg('age')

See: Query builder

Schema

Create table

knex.schema.createTable('accounts', table => {

Columns

  table.increments('id')
  table.string('account_name')
  table.integer('age')
  table.float('age')
  table.decimal('balance', 8, 2)
  table.boolean('is_admin')
  table.date('birthday')
  table.time('created_at')
  table.timestamp('created_at').defaultTo(knex.fn.now())
  table.json('profile')
  table.jsonb('profile')
  table.uuid('id').primary()

Constraints

  table.unique('email')
  table.unique(['email', 'company_id'])
  table.dropUnique(···)

Indices

  table.foreign('company_id')
    .references('companies.id')
  table.dropForeign(···)

Variations

  table.integer('user_id')
    .unsigned()
    .references('users.id')
})
.then(() => ···)

See: Schema builder

Alter table

knex.schema.table('accounts', table => {

Create

  table.string('first_name')

Alter

  table.string('first_name').alter()
  table.renameColumn('admin', 'is_admin')

Drop

  table.dropColumn('admin')
  table.dropTimestamps('created_at')
})

See: Schema builder

Other methods

knex.schema
  .renameTable('persons', 'people')
  .dropTable('persons')
  .hasTable('users').then(exists => ···)
  .hasColumn('users', 'id').then(exists => ···)

See: Schema builder

Modifying

Insert

knex('users')

Insert one

  .insert({ name: 'John' })

Insert many

  .insert([
    { name: 'Starsky' },
    { name: 'Hutch' }
  ])

See: Insert

Update

knex('users')
  .where({ id: 2 })
  .update({ name: 'Homer' })

See: Update

Delete

knex('users')
  .where({ id: 2 })
  .del()

See: Delete

Migrations

Setting up

Create knexfile.js

./node_modules/.bin/knex init

Create a migration

knex migrate:make migration_name
knex migrate:make migration_name --env production

Run migrations

knex migrate:latest
knex migrate:latest --env production

Rollback

knex migrate:rollback
knex migrate:rollback --env production

See: Migrations