forked from doctrine/mongodb-odm
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathDocumentRepository.php
239 lines (217 loc) · 6.96 KB
/
DocumentRepository.php
1
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
<?php
/*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* This software consists of voluntary contributions made by many individuals
* and is licensed under the LGPL. For more information, see
* <http://www.doctrine-project.org>.
*/
namespace Doctrine\ODM\MongoDB;
/**
* An DocumentRepository serves as a repository for documents with generic as well as
* business specific methods for retrieving documents.
*
* This class is designed for inheritance and users can subclass this class to
* write their own repositories with business-specific methods to locate documents.
*
* @since 2.0
* @author Benjamin Eberlei <kontakt@beberlei.de>
* @author Guilherme Blanco <guilhermeblanco@hotmail.com>
* @author Jonathan Wage <jonwage@gmail.com>
* @author Roman Borschel <roman@code-factory.org>
*/
class DocumentRepository
{
/**
* @var string
*/
protected $_documentName;
/**
* @var DocumentManager
*/
protected $_dm;
/**
* @var MongoCollection
*/
protected $_collection;
/**
* @var Doctrine\ODM\MongoDB\Mapping\ClassMetadata
*/
protected $_class;
/**
* The Document hydrator instance.
*
* @var Doctrine\ODM\MongoDB\Hydrator
*/
protected $_hydrator;
/**
* The UnitOfWork used to coordinate object-level transactions.
*
* @var Doctrine\ORM\UnitOfWork
*/
private $_unitOfWork;
/**
* Initializes a new <tt>DocumentRepository</tt>.
*
* @param DocumentManager $dm The DocumentManager to use.
* @param ClassMetadata $classMetadata The class descriptor.
*/
public function __construct($dm, Mapping\ClassMetadata $class)
{
$this->_documentName = $class->name;
$this->_dm = $dm;
$this->_unitOfWork = $this->_dm->getUnitOfWork();
$this->_hydrator = $this->_dm->getHydrator();
$this->_collection = $this->_dm->getDocumentCollection($class->name);
$this->_class = $class;
}
/**
* Create a new QueryBuilder instance that is prepopulated for this document name
*
* @return QueryBuilder $qb
*/
public function createQuery()
{
return $this->_dm->createQuery()
->from($this->_documentName);
}
/**
* Clears the repository, causing all managed documents to become detached.
*/
public function clear()
{
$this->_dm->clear($this->_class->rootDocumentName);
}
/**
* Find a single document by its identifier or multiple by a given criteria.
*
* @param mixed $query A single identifier or an array of criteria.
* @param array $select The fields to select.
* @return Doctrine\ODM\MongoDB\MongoCursor $cursor
* @return object $document
*/
public function find($query = array(), array $select = array())
{
if (is_string($query)) {
$result = $this->_collection->findOne(array('_id' => new \MongoId($query)));
if ($result !== null) {
return $this->_unitOfWork->getOrCreateDocument($this->_documentName, $result);
}
return null;
} else {
$cursor = $this->_collection->find($query, $select);
return new MongoCursor($this->_dm, $this->_hydrator, $this->_class, $cursor);
}
}
/**
* Find a single document with the given query and select fields.
*
* @param string $documentName The document to find.
* @param array $query The query criteria.
* @param array $select The fields to select
* @return object $document
*/
public function findOne(array $query = array(), array $select = array())
{
$result = $this->_collection->findOne($query, $select);
if ($result !== null) {
return $this->_unitOfWork->getOrCreateDocument($this->_documentName, $result);
}
return null;
}
/**
* Finds all documents in the repository.
*
* @param int $hydrationMode
* @return array The documents.
*/
public function findAll()
{
return $this->find();
}
/**
* Finds documents by a set of criteria.
*
* @param string $column
* @param string $value
* @return array
*/
public function findBy(array $criteria)
{
return $this->find($criteria);
}
/**
* Finds a single document by a set of criteria.
*
* @param string $column
* @param string $value
* @return object
*/
public function findOneBy(array $criteria)
{
return $this->findOne($criteria);
}
/**
* Adds support for magic finders.
*
* @return array|object The found document/documents.
* @throws BadMethodCallException If the method called is an invalid find* method
* or no find* method at all and therefore an invalid
* method call.
*/
public function __call($method, $arguments)
{
if (substr($method, 0, 6) == 'findBy') {
$by = substr($method, 6, strlen($method));
$method = 'findBy';
} elseif (substr($method, 0, 9) == 'findOneBy') {
$by = substr($method, 9, strlen($method));
$method = 'findOneBy';
} else {
throw new \BadMethodCallException(
"Undefined method '$method'. The method name must start with ".
"either findBy or findOneBy!"
);
}
if ( ! isset($arguments[0])) {
throw ORMException::findByRequiresParameter($method.$by);
}
$fieldName = lcfirst(\Doctrine\Common\Util\Inflector::classify($by));
if ($this->_class->hasField($fieldName)) {
return $this->$method(array($fieldName => $arguments[0]));
} else {
throw MongoDBException::invalidFindByCall($this->_documentName, $fieldName, $method.$by);
}
}
/**
* @return string
*/
public function getDocumentName()
{
return $this->_documentName;
}
/**
* @return DocumentManager
*/
public function getDocumentManager()
{
return $this->_dm;
}
/**
* @return Mapping\ClassMetadata
*/
public function getClassMetadata()
{
return $this->_class;
}
}