th3mouk / materialized-view
Declarative, PostgreSQL-native management of materialized views: define views as versioned SQL, synchronise, rebuild, refresh (concurrently) and read them safely. Runs on Doctrine DBAL or a bare PDO connection.
Requires
- php: >=8.4
- psr/log: ^3.0
Requires (Dev)
- ext-pdo: *
- doctrine/dbal: ^4.4
- doctrine/orm: ^3.6
- friendsofphp/php-cs-fixer: ^3.64
- phpstan/phpstan: ^2.1
- phpunit/phpunit: ^13.0
- rector/rector: ^2.0
Suggests
- ext-pdo_pgsql: To run on a bare PDO connection (no Doctrine) via MaterializedViewManager::forPdo().
- doctrine/dbal: To wire the manager with MaterializedViewManager::forConnection() and keep DBAL's primary/replica routing, middlewares and profiling.
- doctrine/orm: To map a materialized view as a read-only entity via Th3Mouk\MaterializedView\DoctrineOrm.
- th3mouk/materialized-view-bundle: Symfony integration: autoconfiguration, console commands and the locked deploy lane.
This package is auto-updated.
Last update: 2026-06-15 14:28:02 UTC
README
Declarative, PostgreSQL-native management of materialized views — on Doctrine DBAL or a bare PDO connection.
Define a materialized view once, as a versioned .sql file plus a small PHP definition, and let the library create it, detect drift, rebuild it safely, refresh it (including CONCURRENTLY), and let your app read it without surprises. No ORM owns the DDL; PostgreSQL stays the source of truth for the physical object.
This is the framework-agnostic core. It talks to PostgreSQL through a tiny Connection port, so Doctrine is optional: run it on a plain PDO handle, or hand it a Doctrine DBAL connection to additionally get primary/replica routing, middlewares, profiling and read-only ORM mapping. For Symfony — autoconfiguration, console commands, the locked deploy lane and async refresh — use th3mouk/materialized-view-bundle.
Why this library exists
A materialized view is a physical PostgreSQL object with its own lifecycle: CREATE MATERIALIZED VIEW … AS …, REFRESH MATERIALIZED VIEW [CONCURRENTLY], no CREATE OR REPLACE, destructive redefinition, unique-index preconditions for concurrent refresh, dependency ordering. Doctrine ORM can read the rows as a read-only projection, but it cannot express any of that — and at the time of writing no maintained, framework-agnostic PHP library manages PostgreSQL materialized views. This library fills that gap — natively on Doctrine DBAL, or on a bare PDO connection when you don't run an ORM.
See docs/internals/design-rationale.md for the full reasoning and the competitive landscape.
Highlights
- Declarative: SQL lives in
db/matviews/*.sql; a tiny PHP class declares name, indexes, rebuild & population policy. - Drift detection via a canonical hash stored in
COMMENT ON MATERIALIZED VIEW(travels with database clones). - Safe rebuilds:
drop_createand a low-lockside_by_sidestrategy, with index and GRANT re-application. - Refresh runtime:
CONCURRENTLYwith precondition validation, primary/replica awareness (on the Doctrine backend),lock_timeout/statement_timeout, advisory locks,ANALYZE. - Catalog-derived dependency ordering (
pg_depend/pg_rewrite) — no hand-maintained graph, no drift. - Doctrine optional: the engine runs on a bare
PDOhandle with zero extra Composer dependencies; a Doctrine DBAL connection is natively supported and adds primary/replica routing, middlewares, profiling and read-only ORM mapping. Seedocs/guide/connection-backends.md. - Read-only ORM mapping (optional, Doctrine ORM) with a write guard and an unpopulated-read readiness guard.
Installation
composer require th3mouk/materialized-view
Requirements: PHP ≥ 8.4, PostgreSQL (12+; tested against 17), and one connection backend:
- Doctrine DBAL ≥ 4.4 —
composer require doctrine/dbal. Recommended: unlocks primary/replica routing, middlewares, profiling, and the optional read-only ORM mapping (doctrine/orm≥ 3.6). - or the
pdo_pgsqlextension — no extra Composer dependency; wire the manager withMaterializedViewManager::forPdo().
The core itself depends only on php and psr/log; the backend is your choice.
60-second example (framework-agnostic)
Given a base table orders (id bigint, category text, amount numeric, created_at timestamptz):
db/matviews/sales_by_category.sql
SELECT category, count(*) AS order_count, sum(amount) AS total_amount FROM orders GROUP BY category
use Th3Mouk\MaterializedView\Core\Definition\MaterializedViewDefinition; use Th3Mouk\MaterializedView\Core\Definition\MaterializedViewIndex; use Th3Mouk\MaterializedView\Core\Definition\SqlFileSource; use Th3Mouk\MaterializedView\Core\Registry\MaterializedViewRegistry; $definition = MaterializedViewDefinition::create('public.sales_by_category') ->fromSql(SqlFileSource::fromProjectPath('db/matviews/sales_by_category.sql')) ->withIndex(MaterializedViewIndex::unique( name: 'ux_sales_by_category_category', columns: ['category'], )); $registry = MaterializedViewRegistry::fromDefinitions([$definition]);
Wire the manager onto your connection backend:
use Th3Mouk\MaterializedView\Core\MaterializedViewManager; // With Doctrine DBAL (recommended — primary/replica routing, middlewares, profiling): $manager = MaterializedViewManager::forConnection($dbalConnection); // …or without Doctrine, on a bare PDO connection: $manager = MaterializedViewManager::forPdo(new PDO($dsn, $user, $password));
$manager->syncAll($registry); // create / rebuild on drift $manager->refresh($definition); // REFRESH MATERIALIZED VIEW (CONCURRENTLY when possible)
Documentation
| Tier | Audience | Start here |
|---|---|---|
| Getting started | Users — set it up fast | docs/getting-started.md |
| Guide | Users — advanced concepts | docs/guide/ |
| Internals | Maintainers — design & upstream references | docs/internals/ |
A full table of contents is in docs/README.md.
Compatibility
| This library | PHP | Doctrine DBAL (optional) | Doctrine ORM (optional) | PostgreSQL |
|---|---|---|---|---|
^1.x |
≥ 8.4 | ^4.4 | ^3.6 | 12 – 17 |
The core needs only a PostgreSQL connection — Doctrine DBAL or pdo_pgsql. Doctrine is optional and natively supported; it adds primary/replica routing, middlewares, profiling and read-only ORM mapping. We track DBAL major versions deliberately and do not pin a tight upper bound that would strand the library (see docs/internals/compatibility-and-evolution.md).
License
Apache-2.0 — Copyright © 2026 Jérémy Marodon (th3mouk). See NOTICE.
If you use or redistribute this package, keep the NOTICE attribution —
crediting Jérémy Marodon (th3mouk) and naming this library in your product's
documentation or credits. Please contribute upstream rather than
maintaining a public fork.