Skip to content
This repository was archived by the owner on Sep 16, 2021. It is now read-only.

[WIP] added phpcr introduction chapter #273

Merged
merged 1 commit into from
Oct 16, 2013
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
369 changes: 369 additions & 0 deletions book/database_layer.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,369 @@
.. index::
single: Database Layer
single: PHPCR-ODM

The Database Layer: PHPCR-ODM
=============================

The Symfony CMF is storage layer agnostic, meaning that it can work with many
storage layers. By default, the Symfony CMF works with the `Doctrine PHPCR-ODM`_.
In this chapter, you will learn how to work with the Doctrine PHPCR-ODM.

.. tip::

Read more about choosing the correct storage layer in
:doc:`../cookbook/database/choosing_storage_layer`

PHPCR: A Tree Structure
-----------------------

The Doctrine PHPCR-ODM is a doctrine object-mapper on top of the
`PHP Content Repository`_ (PHPCR), which is a PHP adaption of the
`JSR-283 specification`_. The most important feature of PHPCR is the tree
structure to store the data. All data is stored in items of a tree, called
nodes. You can think of this like a file system, that makes it perfect to use
in a CMS.

On top of the tree structure, PHPCR also adds features like searching,
versioning and access control.

Doctrine PHPCR-ODM has the same API as the other Doctrine libraries, like the
`Doctrine ORM`_. The Doctrine PHPCR-ODM adds another great feature to PHPCR:
Multilanguage support.

.. sidebar:: PHPCR Implementations

In order to let the Doctrine PHPCR-ODM communicate with the PHPCR, a PHPCR
implementation is needed. `Jackalope`_ is the reference PHPCR implementation,
which can work with `Apache Jackrabbit`_ (with the `jackalope-jackrabbit`_
package) and with Doctrine DBAL (providing support for postgres, sqlite
and mysql) with the `jackalope-doctrine-dbal`_ package.

A Simple Example: A Task
------------------------

The easiest way to get started with the PHPCR-ODM is to see it in action. In
this section, you are going to create a ``Task`` object and learn how to
persist it.

Creating a Document Class
~~~~~~~~~~~~~~~~~~~~~~~~~

Without thinking about Doctrine or PHPCR-ODM, you can create a ``Task`` object
in PHP::

// src/Acme/TaskBundle/Document/Task.php
namespace Acme\TaskBundle\Document;

class Task
{
protected $description;

protected $done = false;
}

This class - often called a "document" in PHPCR-ODM, meaning *a basic class
that holds data* - is simple and helps fulfill the business requirement of
needing tasks in your application. This class can't be persisted to
Doctrine PHPCR-ODM yet - it's just a simple PHP class.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Being picky, I would say that this class isn't a document until it has been mapped. I.e. I wouldn't call this class a document (even though I suspect it might become one, as it is in the Document namespace)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ping

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe this paragraph could be rephrased to say more clearly: The class is a Model for your application. A model tracked by PHPCR-ODM is called a document.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

+1

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hmm, not sure about this one. It's a sentence used in both ORM and MongoDB ODM. I prefer to not change those standard sentences to much.

Also, I don't want to confuse starters with lots of words which all means somewhat the same.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well. "The class" should be "This class" imo. The paragraph doesn't really click with me. But lets not block it.

.. note::

A Document is analogous to the term ``Entity`` employed by the Doctrine
ORM. You must add this object to the ``Document`` sub-namespace of you
bundle, in order register the mapping data automatically.

Add Mapping Information
~~~~~~~~~~~~~~~~~~~~~~~

Doctrine allows you to work with PHPCR in a much more interesting way than
just fetching data back and forth as an array. Instead, Doctrine allows you to
persist entire objects to PHPCR and fetch entire *objects* out of PHPCR.
This works by mapping a PHP class and its properties to the PHPCR tree.

For Doctrine to be able to do this, you just have to create "metadata", or
configuration that tells Doctrine exactly how the ``Task`` document and its
properties should be *mapped* to PHPCR. This metadata can be specified in a
number of different formats including YAML, XML or directly inside the ``Task``
class via annotations:

.. configuration-block::

.. code-block:: php-annotations

// src/Acme/TaskBundle/Document/Task.php
namespace Acme\TaskBundle\Document;

use Doctrine\ODM\PHPCR\Mapping\Annotations as PHPCR;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i tend to use PHPCRODM as alias. what do others think?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

PHPCRODM is a bit long for me .. i prefer PHPCR or ODM

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we should check if the doc is consistent and promote one way to do this in our slides and examples (and i will even update my projects to train my brain on the consistent way...). PHPCR or ODM? i think PHPCR is more immediately clear.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

then again, it promotes the confusion between the PHPCR and PHPCR-ODM - not sure how critical that is for people to distinguish.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well the ORM uses ORM, MongoDB ODM uses MongoDB, so I think we should use PHPCR for the PHPCR ODM.


/**
* @PHPCR\Document()
*/
class Task
{
/**
* @PHPCR\Id()
*/
protected $id;

/**
* @PHPCR\String()
*/
protected $description;

/**
* @PHPCR\Boolean()
*/
protected $done = false;

/**
* @PHPCR\ParentDocument()
*/
protected $parent;
}

.. code-block:: yaml

# src/Acme/TaskBundle/Resources/config/doctrine/Task.odm.yml
Acme\TaskBundle\Document\Task:
id: id

fields:
description: string
done: boolean

parent_document: parent

.. code-block:: xml

<!-- src/Acme/TaskBundle/Resources/config/doctrine/Task.odm.xml -->
<?xml version="1.0" encoding="UTF-8" ?>
<doctrine-mapping
xmlns="http://doctrine-project.org/schemas/phpcr-odm/phpcr-mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://doctrine-project.org/schemas/phpcr-odm/phpcr-mapping
https://github.com/doctrine/phpcr-odm/raw/master/doctrine-phpcr-odm-mapping.xsd"
>

<document name="Acme\TaskBundle\Document\Task">

<id name="id" />

<field name="description" type="string" />
<field name="done" type="boolean" />

<parent-document name="parent" />
</document>

</doctrine-mapping>

After this, you have to create getters and setters for the properties.
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Am I correct that there is no doctrine:phpcr:generate:entity command for PHPCR?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unfortunately yes. i would love to have it. we might reuse most of the orm code for that, depending on how well they did it... but not for 1.0

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why not for 1.0? If you really want it, I have 1 hour today and 3 hours tomorrow to work on it :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we said we don't want to merge any non-critical things into the code anymore this week, to increase the chances that 1.0 does not have new bugs introduced... there are still many parts of the doc not written if you have some time you want to invest ;-)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, I already planned to invest time into the docs :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

DoctrinePHPCRBundle is already released as 1.0
but feel free to work on it now and we can open a 1.0 branch and merge this into master


.. note::

This Document uses the parent document and a node name to determine its
position in the tree. Because there isn't any name set, it is generated
automatically. If you want to use a specific node name, such as a
sluggified version of the title, you need to add a property mapped as
``Nodename``.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would be great at somepoint to have some mechanisim for automatically generating names for Documents.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

how do you mean? the auto id strategy?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

More like some way to automatically, for example, slugify the title field and use that as the node name. Similar to what happens in RuotungAuto.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you could do that with a repository strategy. a generic way is difficult as you would need to know which field is the "title"


A Document must have an id property. This represents the full path (parent
+ name) of the Document. This will be set by Doctrine by default and it is
not recommend to use the id to determine the location of a Document.

For more information about identifier generation strategies, refer to the
`doctrine documentation`_

.. seealso::

You can also check out Doctrine's `Basic Mapping Documentation`_ for all
details about mapping information. If you use annotations, you'll need to
prepend all annotations with ``PHPCR\``, which is the name of the imported
namespace (e.g. ``PHPCR\Document(..)``), this is not shown in Doctrine's
documentation. You'll also need to include the use
``use Doctrine\ODM\PHPCR\Mapping\Annotations as PHPCR;`` statement, which
imports the PHPCR annotations prefix.

Persisting Documents to PHPCR
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Now that you have a mapped ``Task`` document, complete with getter and setter
methods, you're ready to persist data to PHPCR. From inside a controller,
this is pretty easy, add the following method to the ``DefaultController`` of the
AcmeTaskBundle::

// src/Acme/TaskBundle/Controller/DefaultController.php

// ...
use Acme\TaskBundle\Document\Task;
use Symfony\Component\HttpFoundation\Response;

// ...
public function createAction()
{
$documentManager = $this->get('doctrine_phpcr')->getManager();

$rootTask = $documentManager->find(null, '/tasks');

$task = new Task();
$task->setDescription('Finish CMF project');
$task->setParent($rootTask);

$documentManager->persist($task);

$documentManager->flush();

return new Response('Created task "'.$task->getDescription().'"');
}

Take a look at the previous example in more detail:

* **line 10** This line fetches Doctrine's *document manager* object, which is
responsible for handling the process of persisting and fetching objects to
and from PHPCR.
* **line 12** This line fetches the root document for the tasks, as each
Document needs to have a parent. To create this root document, you can
configure a :ref:`Repository Initializer <phpcr-odm-repository-initializers>`,
which will be executed when running ``doctrine:phpcr:repository:init``.
* **lines 14-16** In this section, you instantiate and work with the ``$task``
object like any other, normal PHP object.
* **line 18** The ``persist()`` method tells Doctrine to "manage" the
``$task`` object. This does not actually cause a query to be made to PHPCR
(yet).
* **line 20** When the ``flush()`` method is called, Doctrine looks through
all of the objects that it is managing to see if they need to be persisted to
PHPCR. In this example, the ``$task`` object has not been persisted yet, so
the document manager makes a query to PHPCR, which adds a new document.

When creating or updating objects, the workflow is always the same. In the
next section, you'll see how Doctrine is smart enough to update documents if
they already exist in PHPCR.

Fetching Objects from PHPCR
~~~~~~~~~~~~~~~~~~~~~~~~~~~

Fetching an object back out of PHPCR is even easier. For example, suppose
you've configured a route to display a specific task by name::

public function showAction($id)
{
$repository = $this->get('doctrine_phpcr')->getRepository('AcmeTaskBundle:Task');
$task = $repository->find('/task/'.$id);

if (!$task) {
throw $this->createNotFoundException('No task found for id '.$name);
}

return new Response('['.($task->isDone() ? 'x' : ' ').'] '.$task->getDescription());
}

To retrieve objects from the document repository using both the ``find`` and
``findMany`` methods and all helper methods of a class-specific repository. In
PHPCR, it's often unkown for developers which node has the data for a specific
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"You can retrieve objects from the document repository using both its find and findAll methods. "

I don't think ->find('/foobar') counts as a query (as I think of it), its a retrieval of a specific document. Also typo: unkown

I don't know if we should explain here that arbitrary documents should be retrieved with $dm->find(null, '/path/to/doc')

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

+1 to mention that the class FQN is not needed in the find() - note that for the Doctrine Repository, find does not have a class parameter and only finds classes of that repository. but imho repositories are prone to get into the way with phpcr-odm and using the documentmanager->find(null, $path) is the easiest way to work.

document, in that case you should use the document manager to find the nodes
(for instance, when you want to get the root document). In example above, we
know they are ``Task`` documents and so we can use the repository.

The repository contains all sorts of helpful methods::

// query by the id (full path)
$task = $repository->find($id);

// query for one task matching be name and done
$task = $repository->findOneBy(array('name' => 'foo', 'done' => false));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure if this would work, @lsmith77

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it turns out it works

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the keys here are the document fields, not the phpcr property names, right?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes indeed.


// query for all tasks matching the name, ordered by done
$tasks = $repository->findBy(
array('name' => 'foo'),
array('done' => 'ASC')
);

.. tip::

If you use the repository class, you can also create a custom repository
for a specific document. This helps Seperation of Concern when using more
complex queries. This is similair to how it's done in Doctrine ORM, for
more information read "`Custom Repository Classes`_" in the core
documentation.

.. tip::

You can also query objects by using the Query Builder provided by
Doctrine PHPCR-ODM. For more information, read
`the QueryBuilder documentation`_.

Updating an Object
~~~~~~~~~~~~~~~~~~

Once you've fetched an object from Doctrine, updating it is easy. Suppose you
have a route that maps a task ID to an update action in a controller::

public function updateAction($name)
{
$documentManager = $this->get('doctrine_phpcr')->getManager();
$repository = $documentManager->getRepository('AcmeTaskBundle:Task');
$task = $repository->find('/tasks/'.$name);

if (!$task) {
throw $this->createNotFoundException('No task found for id '.$name);
}

if (!$task->isDone()) {
$task->setDone(true);
}

$documentManager->flush();

return new Response('[x] '.$task->getDescription());
}

Updating an object involves just three steps:

#. fetching the object from Doctrine;
#. modifying the object;
#. calling ``flush()`` on the document manager

Notice that calling ``$documentManger->persist($task)`` isn't necessary.
Recall that this method simply tells Doctrine to manage or "watch" the
``$task`` object. In this case, since you fetched the ``$task`` object from
Doctrine, it's already managed.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will stop correcting these contractions now. As a native english speaker, I think expanding them is more "correct", but thats just a feeling, so do whatever you want.


Deleting an Object
~~~~~~~~~~~~~~~~~~

Deleting an object is very similar, but requires a call to the ``remove()`` method
of the document manager after you fetched the document from PHPCR::

$documentManager->remove($task);
$documentManager->flush();

As you might expect, the ``remove()`` method notifies Doctrine that you'd like to
remove the given document from PHPCR. The actual delete operation
however, is not actually executed until the ``flush()`` method is called.

Summary
-------

With Doctrine, you can focus on your objects and how they're useful in your
application and worry about database persistence second. This is because
Doctrine allows you to use any PHP object to hold your data and relies on
mapping metadata information to map an object's data to a particular database
table.

And even though Doctrine revolves around a simple concept, it's incredibly
powerful, allowing you to create complex queries and subscribe to events that
allow you to take different actions as objects go through their persistence
lifecycle.

.. _`Doctrine PHPCR-ODM`: http://docs.doctrine-project.org/projects/doctrine-phpcr-odm/en/latest/index.html
.. _`PHP Content Repository`: http://phpcr.github.io/
.. _`JSR-283 specification`: http://jcp.org/en/jsr/detail?id=283
.. _`Doctrine ORM`: http://symfony.com/doc/current/book/doctrine.html
.. _`Jackalope`: http://jackalope.github.io/
.. _`Apache Jackrabbit`: http://jackrabbit.apache.org/
.. _`jackalope-jackrabbit`: https://github.com/jackalope/jackalope-jackrabbit
.. _`jackalope-doctrine-dbal`: https://github.com/jackalope/jackalope-doctrine-dbal
.. _`doctrine documentation`: http://docs.doctrine-project.org/projects/doctrine-phpcr-odm/en/latest/reference/basic-mapping.html#basicmapping-identifier-generation-strategies
.. _`Basic Mapping Documentation`: http://docs.doctrine-project.org/projects/doctrine-phpcr-odm/en/latest/reference/annotations-reference.html
.. _`the QueryBuilder documentation`: http://docs.doctrine-project.org/projects/doctrine-phpcr-odm/en/latest/reference/query-builder.html
.. _`Custom Repository Classes`: http://symfony.com/doc/current/book/doctrine.html#custom-repository-classes
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it is on line 379

1 change: 1 addition & 0 deletions book/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ Getting Started
.. toctree::
:hidden:

database_layer
installation
routing
simplecms
Expand Down
1 change: 1 addition & 0 deletions book/map.rst.inc
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
* :doc:`installation`
* :doc:`simplecms`
* :doc:`routing`
* :doc:`database_layer`
* :doc:`static_content`
* :doc:`structuring_content`
1 change: 1 addition & 0 deletions index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ will typically want to keep this close at hand.
book/installation
book/simplecms
book/routing
book/database_layer
book/static_content
book/structuring_content

Expand Down