У меня болит голова, которой я хотел бы поделиться с вами 🙂
Это мой код (часть небольшого проекта, который я делаю в php + mongo):
<?php
$db = new MongoClient('mongodb://localhost');
$db->connect();
$collection = $db->{'someDb'}->{'someCollection'};
print '********************************************' . PHP_EOL;
print '******** SAVE ***********' . PHP_EOL;
print '********************************************' . PHP_EOL;
$obj = array(
"_id" =>new MongoId(),
"dob" => new MongoDate(strtotime("1983-08-31T04:00:00Z")),
"name" =>"Author [541a88934d1ed8.64628062]",
"password" => "229fe88b25ae8307601bf6c9c050bf02755b7e26",
"timezone" => "Australia/Sydney",
"expiry" => new MongoDate(strtotime("2015-09-18T07:24:03Z")),
"token" => array(
0 => array(
"title" => "API Test",
"hash" => "ce9808c5063f114f21cbf2d7e194caeccd17d0ae",
"expiry" => new MongoDate(strtotime("2015-09-18T07:24:03Z"))
)
)
);
print_r($obj);
$collection->save($obj);
print '********************************************' . PHP_EOL;
print '******** SEARCH ***********' . PHP_EOL;
print '********************************************' . PHP_EOL;
$where = array(
'token.hash' => 'ce9808c5063f114f21cbf2d7e194caeccd17d0ae',
'expiry' => array('$lt' => new \MongoDate())
);
print '>>> CRITERIA ' . PHP_EOL;
print_r($where);
print '>>> RESULT ' . PHP_EOL;
$cursor = $collection->find()->limit(1);
$result = iterator_to_array($cursor);
$result = array_shift($result);
print_r($result);
print '********************************************' . PHP_EOL;
print '******** NESTED SEARCH ***********' . PHP_EOL;
print '********************************************' . PHP_EOL;
$where = array(
'token.hash' => 'ce9808c5063f114f21cbf2d7e194caeccd17d0ae',
'token.expiry' => array('$lt' => new \MongoDate())
);
print '>>> CRITERIA ' . PHP_EOL;
print_r($where);
$cursor = $collection->find($where)->limit(1);
$result = iterator_to_array($cursor);
$result = array_shift($result);
print '>>> RESULT ' . PHP_EOL;
var_dump($result);
Первый поиск работает отлично, но второй поиск не работает должным образом, проверьте результат:
********************************************
******** SAVE ***********
********************************************
Array
(
[_id] => MongoId Object
(
[$id] => 541a9111c48e5859108b4567
)
[dob] => MongoDate Object
(
[sec] => 431150400
[usec] => 0
)
[name] => Author [541a88934d1ed8.64628062]
[password] => 229fe88b25ae8307601bf6c9c050bf02755b7e26
[timezone] => Australia/Sydney
[expiry] => MongoDate Object
(
[sec] => 1442561043
[usec] => 0
)
[token] => Array
(
[0] => Array
(
[title] => API Test
[hash] => ce9808c5063f114f21cbf2d7e194caeccd17d0ae
[expiry] => MongoDate Object
(
[sec] => 1442561043
[usec] => 0
)
)
)
)
********************************************
******** SEARCH ***********
********************************************
>>> CRITERIA
Array
(
[token.hash] => ce9808c5063f114f21cbf2d7e194caeccd17d0ae
[expiry] => Array
(
[$lt] => MongoDate Object
(
[sec] => 1411027217
[usec] => 868000
)
)
)
>>> RESULT
Array
(
[_id] => MongoId Object
(
[$id] => 541a8e23c48e58700f8b4567
)
[dob] => MongoDate Object
(
[sec] => 431150400
[usec] => 0
)
[name] => Author [541a88934d1ed8.64628062]
[password] => 229fe88b25ae8307601bf6c9c050bf02755b7e26
[timezone] => Australia/Sydney
[token] => Array
(
[0] => Array
(
[title] => API Test
[hash] => ce9808c5063f114f21cbf2d7e194caeccd17d0ae
[expiry] => MongoDate Object
(
[sec] => 1442561043
[usec] => 0
)
)
)
)
********************************************
******** NESTED SEARCH ***********
********************************************
>>> CRITERIA
Array
(
[token.hash] => ce9808c5063f114f21cbf2d7e194caeccd17d0ae
[token.expiry] => Array
(
[$lt] => MongoDate Object
(
[sec] => 1411027217
[usec] => 870000
)
)
)
>>> RESULT
NULL
Я не гуру MongoDb, просто плохой программист, пытающийся спасти свой день, у кого-нибудь есть идеи, почему это происходит? Что я делаю неправильно?
Версия PHP:
valerio@beatrice:~$ php -v
PHP 5.5.9-1ubuntu4.4 (cli) (built: Sep 4 2014 06:56:34)
Copyright (c) 1997-2014 The PHP Group
Zend Engine v2.5.0, Copyright (c) 1998-2014 Zend Technologies
with Zend OPcache v7.0.3, Copyright (c) 1999-2014, by Zend Technologies
valerio@beatrice:~$
Информация о драйвере MongoDb (из php -info)
MongoDB Support => enabled
Version => 1.4.5
SSL Support => enabled
Streams Support => enabled
Directive => Local Value => Master Value
mongo.allow_empty_keys => 0 => 0
mongo.chunk_size => 262144 => 262144
mongo.cmd => $ => $
mongo.default_host => localhost => localhost
mongo.default_port => 27017 => 27017
mongo.is_master_interval => 15 => 15
mongo.long_as_object => 0 => 0
mongo.native_long => 0 => 0
mongo.ping_interval => 5 => 5
Предполагая, что предоставленный вами документ является единственным в коллекции, он не соответствует ни одному из перечисленных шаблонов критериев. Единственная причина, по которой первый пример «SEARCH» возвращает документ, заключается в том, что вы не передаете $where
к find()
метод:
$collection->find()->limit(1);
Напомним, что поскольку ваш второй шаблон критериев (т. Е. «NESTED SEARCH») запрашивает два вложенных поля в одном и том же массиве, я предполагаю, что вы на самом деле хотите убедиться, что они соответствуют вложенным полям в одном и том же элементе массива. $elemMatch
Оператор запроса может помочь с этим:
$where = [
'token' => [
'$elemMatch' => [
'hash' => 'ce9808c5063f114f21cbf2d7e194caeccd17d0ae',
'expiry' => ['$lt' => new \MongoDate()],
],
],
];
Без $elemMatch
исходные критерии могут соответствовать документу с несколькими объектами-токенами, где один из них соответствует только hash
пункт, а другой соответствует expiry
,
Других решений пока нет …