You are browsing a version that has not yet been released.

Bi-Directional References

By default when you map a bi-directional reference, the reference is maintained on both sides of the relationship and there is not a single owning side. Both sides are considered owning and changes are tracked and persisted separately. Here is an example:

1<?php #[Document] class BlogPost { // ... #[ReferenceOne(targetDocument: User::class)] private User $user; } #[Document] class User { // ... /** @var Collection<BlogPost> */ #[ReferenceMany(targetDocument: BlogPost::class)] private Collection $posts; }
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

When I persist some instances of the above classes the references would exist on both sides! The BlogPost collection would have a DBRef stored on the $user property and the User collection would have a DBRef stored in the $posts property.

Owning and Inverse Sides

A user may have lots of posts and we don't need to store a reference to each post on the user, we can get the users post by running a query like the following:

1db.BlogPost.find({ 'user.$id' : user.id })

In order to map this you can use the inversedBy and mappedBy options. Here is the same example above where we implement this:

One to Many

1<?php #[Document] class BlogPost { // ... #[ReferenceOne(targetDocument: User::class, inversedBy: 'posts')] private User $user; } #[Document] class User { // ... /** @var Collection<BlogPost> */ #[ReferenceMany(targetDocument: BlogPost::class, mappedBy: 'user')] private Collection $posts; }
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

So now when we persist a User and multiple BlogPost instances for that User:

1<?php $user = new User(); $post1 = new BlogPost(); $post1->setUser($user); $post2 = new BlogPost(); $post2->setUser($user); $post3 = new BlogPost(); $post3->setUser($user); $dm->persist($post1); $dm->persist($post2); $dm->persist($post3); $dm->flush();
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

And we retrieve the User later to access the posts for that user:

1<?php $user = $dm->find(User::class, $user->id); $posts = $user->getPosts(); foreach ($posts as $post) { // ... }
2
3
4
5
6
7
8

The above will execute a query like the following to lazily load the collection of posts to iterate over:

1db.BlogPost.find( { 'user.$id' : user.id } )

Remember that the inverse side, the side which specified mappedBy is immutable and any changes to the state of the reference will not be persisted.

Other Examples

Here are several examples which implement the inversedBy and mappedBy options:

One to One

Here is an example where we have a one to one relationship between Cart and Customer:

1<?php #[Document] class Cart { // ... #[ReferenceOne(targetDocument: Customer::class, inversedBy: 'cart')] public Customer $customer; } #[Document] class Customer { // ... #[ReferenceOne(targetDocument: Cart::class, mappedBy: 'customer')] public Cart $cart; }
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

The owning side is on Cart.customer and the Customer.cart referenced is loaded with a query like this:

1db.Cart.find( { 'customer.$id' : customer.id } )

If you want to nullify the relationship between a Cart instance and Customer instance you must null it out on the Cart.customer side:

1<?php $cart->setCustomer(null); $dm->flush();
2
3
4

When specifying inverse one-to-one relationships the referenced document is loaded directly when the owning document is hydrated instead of using a proxy. In the example above, loading a Customer object from the database would also cause the corresponding Cart to be loaded. This can cause performance issues when loading many Customer objects at once.

Self-Referencing Many to Many

1<?php namespace Documents; #[Document] class User { // ... /** @var Collection<User> */ #[ReferenceMany(targetDocument: User::class, mappedBy: 'myFriends')] public Collection $friendsWithMe; /** @var Collection<User> */ #[ReferenceMany(targetDocument: User::class, inversedBy: 'friendsWithMe')] public Collection $myFriends; public function __construct($name) { $this->name = $name; $this->friendsWithMe = new \Doctrine\Common\Collections\ArrayCollection(); $this->myFriends = new \Doctrine\Common\Collections\ArrayCollection(); } public function addFriend(User $user): void { $user->friendsWithMe[] = $this; $this->myFriends[] = $user; } }
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30