Working with Objects

Understanding

In this chapter we will help you understand the DocumentManager and the UnitOfWork. A Unit of Work is similar to an object-level transaction. A new Unit of Work is implicitly started when a DocumentManager is initially created or after DocumentManager#flush() has been invoked. A Unit of Work is committed (and a new one started) by invoking DocumentManager#flush().

A Unit of Work can be manually closed by calling DocumentManager#close(). Any changes to objects within this Unit of Work that have not yet been persisted are lost.

The size of a Unit of Work

The size of a Unit of Work mainly refers to the number of managed documents at a particular point in time.

The cost of flush()

How costly a flush operation is in terms of performance mainly depends on the size. You can get the size of your Unit of Work as follows:

<?php

$uowSize = $dm->getUnitOfWork()->size();

The size represents the number of managed documents in the Unit of Work. This size affects the performance of flush() operations due to change tracking and, of course, memory consumption, so you may want to check it from time to time during development.

Caution

Do not invoke flush after every change to a document or every single invocation of persist/remove/merge/... This is an anti-pattern and unnecessarily reduces the performance of your application. Instead, form units of work that operate on your objects and call flush when you are done. While serving a single HTTP request there should be usually no need for invoking flush more than 0-2 times.

Direct access to a Unit of Work

You can get direct access to the Unit of Work by calling DocumentManager#getUnitOfWork(). This will return the UnitOfWork instance the DocumentManager is currently using.

<?php

$uow = $dm->getUnitOfWork();

Note

Directly manipulating a UnitOfWork is not recommended. When working directly with the UnitOfWork API, respect methods marked as INTERNAL by not using them and carefully read the API documentation.

Persisting documents

A document can be made persistent by passing it to the DocumentManager#persist($document) method. By applying the persist operation on some document, that document becomes MANAGED, which means that its persistence is from now on managed by an DocumentManager. As a result the persistent state of such a document will subsequently be properly synchronized with the database when DocumentManager#flush() is invoked.

Caution

Invoking the persist method on a document does NOT cause an immediate insert to be issued on the database. Doctrine applies a strategy called "transactional write-behind", which means that it will delay most operations until DocumentManager#flush() is invoked which will then issue all necessary queries to synchronize your objects with the database in the most efficient way.

Example:

<?php

$user = new User();
$user->setUsername('jwage');
$user->setPassword('changeme');
$dm->persist($user);
$dm->flush();

Caution

The document identifier is generated during persist if not previously specified. Users cannot rely on a document identifier being available during the prePersist event.

The semantics of the persist operation, applied on a document X, are as follows:

  • If X is a new document, it becomes managed. The document X will be entered into the database as a result of the flush operation.
  • If X is a preexisting managed document, it is ignored by the persist operation. However, the persist operation is cascaded to documents referenced by X, if the relationships from X to these other documents are mapped with cascade=PERSIST or cascade=ALL.
  • If X is a removed document, it becomes managed.
  • If X is a detached document, the behavior is undefined.

Caution

Do not pass detached documents to the persist operation.

Flush Options

When committing your documents you can specify an array of options to the flush method. With it you can send options to the underlying database like safe, fsync, etc.

Example:

<?php

$user = $dm->getRepository('User')->find($userId);
// ...
$user->setPassword('changeme');
$dm->flush(null, array('safe' => true, 'fsync' => true));

You can configure the default flush options on your Configuration object if you want to set them globally for all flushes.

Example:

<?php

$config->setDefaultCommitOptions(array(
    'safe' => true,
    'fsync' => true
));

Note

Safe is set to true by default for all writes when using the ODM.

Removing documents

A document can be removed from persistent storage by passing it to the DocumentManager#remove($document) method. By applying the remove operation on some document, that document becomes REMOVED, which means that its persistent state will be deleted once DocumentManager#flush() is invoked. The in-memory state of a document is unaffected by the remove operation.

Caution

Just like persist, invoking remove on a document does NOT cause an immediate query to be issued on the database. The document will be removed on the next invocation of DocumentManager#flush() that involves that document.

Example:

<?php

$dm->remove($user);
$dm->flush();

The semantics of the remove operation, applied to a document X are as follows:

  • If X is a new document, it is ignored by the remove operation. However, the remove operation is cascaded to documents referenced by X, if the relationship from X to these other documents is mapped with cascade=REMOVE or cascade=ALL.
  • If X is a managed document, the remove operation causes it to become removed. The remove operation is cascaded to documents referenced by X, if the relationships from X to these other documents is mapped with cascade=REMOVE or cascade=ALL.
  • If X is a detached document, an InvalidArgumentException will be thrown.
  • If X is a removed document, it is ignored by the remove operation.
  • A removed document X will be removed from the database as a result of the flush operation.

Detaching documents

A document is detached from a DocumentManager and thus no longer managed by invoking the DocumentManager#detach($document) method on it or by cascading the detach operation to it. Changes made to the detached document, if any (including removal of the document), will not be synchronized to the database after the document has been detached.

Doctrine will not hold on to any references to a detached document.

Example:

<?php

$dm->detach($document);

The semantics of the detach operation, applied to a document X are as follows:

  • If X is a managed document, the detach operation causes it to become detached. The detach operation is cascaded to documents referenced by X, if the relationships from X to these other documents is mapped with cascade=DETACH or cascade=ALL. Documents which previously referenced X will continue to reference X.
  • If X is a new or detached document, it is ignored by the detach operation.
  • If X is a removed document, the detach operation is cascaded to documents referenced by X, if the relationships from X to these other documents is mapped with cascade=DETACH or cascade=ALL/Documents which previously referenced X will continue to reference X.

There are several situations in which a document is detached automatically without invoking the detach method:

  • When DocumentManager#clear() is invoked, all documents that are currently managed by the DocumentManager instance become detached.
  • When serializing a document. The document retrieved upon subsequent unserialization will be detached (This is the case for all documents that are serialized and stored in some cache).

The detach operation is usually not as frequently needed and used as persist and remove.

Merging documents

Merging documents refers to the merging of (usually detached) documents into the context of a DocumentManager so that they become managed again. To merge the state of a document into an DocumentManager use the DocumentManager#merge($document) method. The state of the passed document will be merged into a managed copy of this document and this copy will subsequently be returned.

Example:

<?php

$detachedDocument = unserialize($serializedDocument); // some detached document
$document = $dm->merge($detachedDocument);
// $document now refers to the fully managed copy returned by the merge operation.
// The DocumentManager $dm now manages the persistence of $document as usual.

The semantics of the merge operation, applied to a document X, are
as follows:
  • If X is a detached document, the state of X is copied onto a pre-existing managed document instance X' of the same iddocument or a new managed copy X' of X is created.
  • If X is a new document instance, an InvalidArgumentException will be thrown.
  • If X is a removed document instance, an InvalidArgumentException will be thrown.
  • If X is a managed document, it is ignored by the merge operation, however, the merge operation is cascaded to documents referenced by relationships from X if these relationships have been mapped with the cascade element value MERGE or ALL.
  • For all documents Y referenced by relationships from X having the cascade element value MERGE or ALL, Y is merged recursively as Y'. For all such Y referenced by X, X' is set to reference Y'. (Note that if X is managed then X is the same object as X'.)
  • If X is a document merged to X', with a reference to another document Y, where cascade=MERGE or cascade=ALL is not specified, then navigation of the same association from X' yields a reference to a managed object Y' with the same persistent iddocument as Y.

The merge operation is usually not as frequently needed and used as persist and remove. The most common scenario for the merge operation is to reattach documents to an DocumentManager that come from some cache (and are therefore detached) and you want to modify and persist such a document.

Note

If you load some detached documents from a cache and you do not need to persist or delete them or otherwise make use of them without the need for persistence services there is no need to use merge. I.e. you can simply pass detached objects from a cache directly to the view.

References

References between documents and embedded documents are represented just like in regular object-oriented PHP, with references to other objects or collections of objects.

Establishing References

Establishing a reference to another document is straight forward:

Here is an example where we add a new comment to an article:

<?php

$comment = new Comment();
// ...

$article->getComments()->add($comment);

Or you can set a single reference:

<?php

$address = new Address();
// ...

$user->setAddress($address);

Removing References

Removing an association between two documents is similarly straight-forward. There are two strategies to do so, by key and by element. Here are some examples:

<?php

$article->getComments()->removeElement($comment);
$article->getComments()->remove($ithComment);

Or you can remove a single reference:

<?php

$user->setAddress(null);

When working with collections, keep in mind that a Collection is essentially an ordered map (just like a PHP array). That is why the remove operation accepts an index/key. removeElement is a separate method that has O(n) complexity, where n is the size of the map.

Transitive persistence

Persisting, removing, detaching and merging individual documents can become pretty cumbersome, especially when a larger object graph with collections is involved. Therefore Doctrine provides a mechanism for transitive persistence through cascading of these operations. Each reference to another document or a collection of documents can be configured to automatically cascade certain operations. By default, no operations are cascaded.

The following cascade options exist:

  • persist : Cascades persist operations to the associated documents.
  • remove : Cascades remove operations to the associated documents.
  • merge : Cascades merge operations to the associated documents.
  • detach : Cascades detach operations to the associated documents.
  • all : Cascades persist, remove, merge and detach operations to associated documents.

The following example shows an association to a number of addresses. If persist() or remove() is invoked on any User document, it will be cascaded to all associated Address documents in the $addresses collection.

<?php

class User
{
    //...
    /**
     * @ReferenceMany(targetDocument="Address", cascade={"persist", "remove"})
     */
    private $addresses;
    //...
}

Even though automatic cascading is convenient it should be used with care. Do not blindly apply cascade=all to all associations as it will unnecessarily degrade the performance of your application.

Querying

Doctrine provides the following ways, in increasing level of power and flexibility, to query for persistent objects. You should always start with the simplest one that suits your needs.

By Primary Key

The most basic way to query for a persistent object is by its identifier / primary key using the DocumentManager#find($documentName, $id) method. Here is an example:

<?php

$user = $dm->find('User', $id);

The return value is either the found document instance or null if no instance could be found with the given identifier.

Essentially, DocumentManager#find() is just a shortcut for the following:

<?php

$user = $dm->getRepository('User')->find($id);

DocumentManager#getRepository($documentName) returns a repository object which provides many ways to retrieve documents of the specified type. By default, the repository instance is of type Doctrine\ODM\MongoDB\DocumentRepository. You can also use custom repository classes.

By Simple Conditions

To query for one or more documents based on several conditions that form a logical conjunction, use the findBy and findOneBy methods on a repository as follows:

<?php

// All users that are 20 years old
$users = $dm->getRepository('User')->findBy(array('age' => 20));

// All users that are 20 years old and have a surname of 'Miller'
$users = $dm->getRepository('User')->findBy(array('age' => 20, 'surname' => 'Miller'));

// A single user by its nickname
$user = $dm->getRepository('User')->findOneBy(array('nickname' => 'romanb'));

A DocumentRepository also provides a mechanism for more concise calls through its use of __call. Thus, the following two examples are equivalent:

<?php

// A single user by its nickname
$user = $dm->getRepository('User')->findOneBy(array('nickname' => 'romanb'));

// A single user by its nickname (__call magic)
$user = $dm->getRepository('User')->findOneByNickname('romanb');

Note

You can learn more about Repositories in a dedicated chapter.

By Lazy Loading

Whenever you have a managed document instance at hand, you can traverse and use any associations of that document as if they were in-memory already. Doctrine will automatically load the associated objects on demand through the concept of lazy-loading.

By Query Builder Objects

The most powerful and flexible method to query for persistent objects is the QueryBuilder object. The QueryBuilder object enables you to query for persistent objects with a fluent object oriented interface.

You can create a query using DocumentManager#createQueryBuilder($documentName = null). Here is a simple example:

<?php

// All users with an age between 20 and 30 (inclusive).
$qb = $dm->createQueryBuilder('User')
    ->field('age')->range(20, 30);
$q = $qb->getQuery()
$users = $q->execute();

By Reference

To query documents with a ReferenceOne association to another document, use the references($document) expression:

<?php

$group = $dm->find('Group', $id);
$usersWithGroup = $dm->createQueryBuilder('User')
    ->field('group')->references($group)
    ->getQuery()->execute();

To find documents with a ReferenceMany association that includes a certain document, use the includesReferenceTo($document) expression:

<?php

$users = $dm->createQueryBuilder('User')
    ->field('groups')->includesReferenceTo($group)
    ->getQuery()->execute();
Fork me on GitHub