Need help with TalisORM?
Click the “chat” button below for chat support from the developer who created it, or find similar developers for support.

About the developer

145 Stars 7 Forks 75 Commits 7 Opened issues


An ORM for "ORMless" persistance of DDD-inspired domain models

Services available


Need anything else?

Contributors list

Build Status Code Coverage

About TalisORM

A good design starts with some limitations. You can start simple and keep building until you have a large ORM like Doctrine. Or you can choose not to support a mapping configuration, table inheritance, combined write/read models, navigable object graphs, lazy-loading, etc. That's what I'm looking for with TalisOrm. The rules are:

  • You model a persistable domain object as an Aggregate: one (root) Entity, and optionally some Child entities.
  • The child entities themselves have no children.
  • You use the ORM for your write model only. That is, you don't need to fetch hundreds of these aggregates to show them to the user.
  • Your aggregate internally records domain events, which will automatically be released and dispatched after saving changes to the aggregate.


  • You're going to write your own mapping code, which converts your values or Value objects to and from column values.

I explain more about the motivation for doing this in "ORMless; a Memento-like pattern for object persistence".

You can find some examples of how to use this library in test/TalisOrm/AggregateRepositoryTest/.

Recording and dispatching domain events

A domain event is a simple object indicating that something has happened inside an aggregate (usually this just means that something has changed). You can use the

trait to save yourself from rewriting a couple of simple lines over and over again.

Immediately after saving an aggregate, the

will call the aggregate's

method, which returns previously recorded domain events. It dispatches these events to an object that implements
. As a user of this library you have to provide your own implementation of this interface, which is very simple. Maybe you just want to forward the call to your favorite event dispatcher, or the one that ships with your framework.

Managing the database schema

Aggregates can implement

and, yes, specify their own schema. This can be useful if you want to use a tool to synchronize your current database schema with the schema that your aggregates expect, e.g. the Doctrine DBAL's own
use Doctrine\DBAL\Schema\Synchronizer\SingleDatabaseSynchronizer;
use TalisOrm\Schema\AggregateSchemaProvider;

// set up or reuse a Doctrine\DBAL\Connection instance $connection = ...;

$schemaProvider = new AggregateSchemaProvider($connection, [ // list all the aggregate class names of your application, e.g. User::class, Order::class ]); $synchronizer = new SingleDatabaseSynchronizer($connection); $synchronizer->createSchema($schemaProvider->createSchema());

You could also use Doctrine Migrations to automatically generate migrations based on schema changes. It may need a bit of setup, but once you have it working, you'll notice that this tool needs a

instance (note: this interface is only available in recent versions of
, which requires PHP 7). You can easily set up an adapter for
. For example:
final class AggregateMigrationsSchemaProvider implements SchemaProviderInterface
     * @var AggregateSchemaProvider
    private $aggregateSchemaProvider;

public function __construct(AggregateSchemaProvider $aggregateSchemaProvider)
    $this->aggregateSchemaProvider = $aggregateSchemaProvider;

public function createSchema(): Schema
    return $this->aggregateSchemaProvider->createSchema();


Protecting against concurrent updates

Traditionally, we PHP developers aren't used to protect our aggregates against concurrent updates. Concurrent updates after all are a matter of chance. Maybe there aren't that many users who are working on the same aggregate in your project. But if you're worried that it might happen, there's an easy solution built-in to TalisORM: optimistic concurrency locking.

You need to take the following steps to make it work:

Make sure the table definition for your aggregate has an

column, and that your
methods are aware of it. For example:
final class Order implements Aggregate, SpecifiesSchema
     * @var int
    private $aggregateVersion;

public function state(): array
    // N.B. It's important to increment the version manually every time state() gets called!

    return [
        // ...
        Aggregate::VERSION_COLUMN => $this->aggregateVersion

public static function fromState(array $aggregateState, array $childEntityStatesByType): Aggregate
    $order = new self();

    // ...

    $order->aggregateVersion = $aggregateState[Aggregate::VERSION_COLUMN];

    return $order;

 * Only if your aggregate implements SpecifiesSchema:
public static function specifySchema(Schema $schema): void
    $table = $schema->createTable('orders');

    // ...

    $table->addColumn(Aggregate::VERSION_COLUMN, 'integer');


The above setup will protect your aggregate against concurrent updates between retrieving the aggregate from the database and saving it again. However, you may want to warn a user who's working with the aggregate's data in the user interface that once they store the object, someone else has modified it. To do this, you need to remember the version of the aggregate the user is looking at in the user's session. An outline of this solution:

final class Order implements Aggregate, SpecifiesSchema
    // ...

public function setAggregateVersion(int $version): void
    $this->aggregateVersion = $version;

public function aggregateVersion(): int
    return $this->aggregateVersion;



  • Inside the controller which (for instance) renders a form, allowing the
  • user to modify some aspect of the aggregate:
  • / $order = $repository->getById($orderId); $session->set('aggregate_version', $order->aggregateVersion()); // show form


  • Inside the controller which modifies the aggregate based on the data the
  • user provided:
  • / $order = $repository->getById($orderId); $order->setAggregateVersion($session->get('aggregate_version');


// This will compare the provided version to the version in the database: $repository->save($order);

We use cookies. If you continue to browse the site, you agree to the use of cookies. For more information on our use of cookies please see our Privacy Policy.