Skip to content

PostgreSQL support

While drift has originally been designed as a client-side database wrapper for SQLite databases, it can also be used with PostgreSQL database servers. Without having to change your query code, drift can generate Postgres-compatible SQL for most queries. Please keep in mind that some drift APIs, like those for date time modification, are only supported with SQLite. Most queries will work without any modification though.

Setup

Begin by adding both drift and drift_postgres to your pubspec:

dependencies:
  drift: ^2.23.1
  drift_postgres: ^1.3.0

dev_dependencies:
  drift_dev: ^2.23.1
  build_runner: ^2.4.14

Defining a database with Postgres is no different than defining it for sqlite3 - the pages on Dart and SQL explain how to define tables picked up by drift.

Different dialects require changes in generated code in some cases. Since most drift users are targeting sqlite3, drift generates code optimized for sqlite3 by default. To enable code generation for PostgreSQL as well, create a build.yaml next to your pubspec with this content:

targets:
  $default:
    builders:
      drift_dev:
        options:
          sql:
            dialects:
              - sqlite # remove this line if you only need postgres
              - postgres

Then, perhaps this example database is helpful as a starting point:

import 'package:drift/drift.dart';
import 'package:drift_postgres/drift_postgres.dart';
import 'package:postgres/postgres.dart';

part 'postgres.g.dart';

class Users extends Table {
  UuidColumn get id => customType(PgTypes.uuid).withDefault(genRandomUuid())();
  TextColumn get name => text()();
  Column<PgDate> get birthDate => customType(PgTypes.date).nullable()();
}

@DriftDatabase(tables: [Users])
class MyDatabase extends _$MyDatabase {
  MyDatabase(super.e);

  @override
  int get schemaVersion => 1;
}

void main() async {
  final pgDatabase = PgDatabase(
    endpoint: Endpoint(
      host: 'localhost',
      database: 'postgres',
      username: 'postgres',
      password: 'postgres',
    ),
    settings: ConnectionSettings(
      // If you expect to talk to a Postgres database over a public connection,
      // please use SslMode.verifyFull instead.
      sslMode: SslMode.disable,
    ),
  );

  final driftDatabase = MyDatabase(pgDatabase);

  // Insert a new user
  await driftDatabase.users.insertOne(UsersCompanion.insert(name: 'Simon'));

  // Print all of them
  print(await driftDatabase.users.all().get());

  await driftDatabase.close();
}

After starting a database server, for example by running docker run -p 5432:5432 -e POSTGRES_PASSWORD=postgres postgres, you can run the example to see drift talking to Postgres.

Custom connections and connection pools

The unnamed PgDatabase constructor mirrors the options found on the Endpoint class in package:postgres, as it uses that class to establish a connection to PostgreSQL. In some cases, for instance because you already have an existing Postgres connection or because you need something different from the existing Endpoint, you can use PgDatabase.opened with your existing Session from package:postgres.

This technique is also useful for pooling connections to Postgres, as the Pool implementation from package:postgres implements the Session interface:

Future<void> openWithPool() async {
  final pool = Pool.withEndpoints(yourListOfEndpoints);

  final driftDatabase = MyDatabase(PgDatabase.opened(pool));
  await driftDatabase.users.select().get();

  // Note that PgDatabase.opened() doesn't close the underlying connection when
  // the drift database is closed.
  await driftDatabase.close();
  await pool.close();
}

API extensions

The postgres library provides a few custom types enabling you to use postgres-specific types when writing queries in drift. For instance, the PgTypes.uuid type used in the example maps to a native UUID column type in Postgres. The gen_random_uuid() function in postgres is also exposed.

PostgreSQL provides a much larger set of functions, of which currently only a few are exported in the drift_postgres package. You can call others with a FunctionCallExpression - if you do, contributions extending drift_postgres are always welcome!

Avoiding sqlite-specific drift APIs

Early drift versions were designed with SQLite in mind only. Support for PostgreSQL and other database systems has only been added in more recent versions, and this is reflected by some drift APIs being SQLite-specific. These will be moved into separate libraries in a future major release to avoid confusion, but it's best to be aware of them for the time being. This section lists affected APIs and workarounds to make them work PostgreSQL.

Migrations

Most parts of the Migrator API are SQLite-specific. You will be able to create tables on PostgreSQL as well, but methods like alterTable will only work with SQLite. While it's possible to use drift migrations with PostgreSQL databases, the recommended approach for now is to export your drift schema and then use dedicated migration tools for PostgreSQL.

In sqlite3, the current schema version is stored in the database file. To support drift's migration API being built on top of this mechanism in Postgres as well, drift creates a __schema table storing the current schema version.

This migration mechanism works for simple deployments, but is unsuitable for large database setups with many application servers connecting to a postgres serve. For those, an existing migration management tool is a more reliable alternative. If you chose to manage migrations with another tool, you can disable migrations in drift by passing enableMigrations: false to the PgDatabase constructor.

DateTime columns

Drift's datetime() columns were designed to work with SQLite, which doesn't have dedicated datetime types. Most of the date time APIs (like currentDateAndTime) will not work with PostgreSQL. When using drift databases with PostgreSQL, we suggest avoiding the default dateTime() column type and instead use PgTypes.date or PgTypes.datetime:

// This table uses proper postgres types to store date/time values.
class TimeStore extends Table {
  Column<PgDate> get date => customType(PgTypes.date)();
  Column<PgDateTime> get timestampWithTimezone =>
      customType(PgTypes.timestampWithTimezone)();
  Column<PgDateTime> get timestampWithoutTimezone =>
      customType(PgTypes.timestampNoTimezone)();
  Column<Interval> get interval => customType(PgTypes.interval)();
}

If you need to support both sqlite3 and Postgres, consider using dialect-aware types:

class _DialectAwareDateTimeType implements DialectAwareSqlType<PgDateTime> {
  /// The underlying type used when this dialect-aware type is used on postgres
  /// databases.
  static const _postgres = PgTypes.timestampWithTimezone;

  /// The fallback type used when we're not talking to postgres.
  static const _other = DriftSqlType.dateTime;

  const _DialectAwareDateTimeType();

  @override
  String mapToSqlLiteral(GenerationContext context, PgDateTime dartValue) {
    return switch (context.dialect) {
      SqlDialect.postgres => _postgres.mapToSqlLiteral(dartValue),
      _ => context.typeMapping.mapToSqlLiteral(dartValue.dateTime),
    };
  }

  @override
  Object mapToSqlParameter(GenerationContext context, PgDateTime dartValue) {
    return switch (context.dialect) {
      SqlDialect.postgres => _postgres.mapToSqlParameter(dartValue),
      _ => context.typeMapping.mapToSqlVariable(dartValue.dateTime)!,
    };
  }

  @override
  PgDateTime read(SqlTypes typeSystem, Object fromSql) {
    return switch (typeSystem.dialect) {
      SqlDialect.postgres => _postgres.read(fromSql),
      _ => PgDateTime(typeSystem.read(_other, fromSql)!),
    };
  }

  @override
  String sqlTypeName(GenerationContext context) {
    return switch (context.dialect) {
      SqlDialect.postgres => _postgres.sqlTypeName(context),
      _ => _other.sqlTypeName(context),
    };
  }
}

const dateTime = _DialectAwareDateTimeType();

class DialectAwareTime extends Table {
  // This will use `timestamp with timezone` on postgres, and fall back to the
  // default date type (integer or text) on sqlite databases.
  Column<PgDateTime> get timeValue => customType(dateTime)();
}

Current state

Drift's support for PostgreSQL is stable in the sense that the current API is unlikely to break. Still, it is a newer implementation and integration tests for PostgreSQL are less extensive than the tests for SQLite databases. And while drift offers typed wrappers around most functions supported by SQLite, only a tiny subset of PostgreSQL's advanced operators and functions are exposed by drift_postgres.

If you're running into problems or bugs with the postgres database, please let us know by creating an issue or a discussion. Contributions expanding wrappers around PosgreSQL functions are also much appreciated.