blob: 7f5ce6345d348045f8d2eead197a0c316d69ed58 [file] [log] [blame]
# Licensed under the Apache License, Version 2.0 (the "License"); you may not
# use this file except in compliance with the License. You may obtain a copy of
# the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations under
# the License.
import json
import mango
import unittest
import user_docs
import math
from hypothesis import given, assume, example
import hypothesis.strategies as st
@unittest.skipIf(mango.has_text_service(), "text service exists")
class TextIndexCheckTests(mango.DbPerClass):
def test_create_text_index(self):
body = json.dumps({
'index': {
},
'type': 'text'
})
resp = self.db.sess.post(self.db.path("_index"), data=body)
assert resp.status_code == 503, resp
@unittest.skipUnless(mango.has_text_service(), "requires text service")
class BasicTextTests(mango.UserDocsTextTests):
def test_simple(self):
docs = self.db.find({"$text": "Stephanie"})
assert len(docs) == 1
assert docs[0]["name"]["first"] == "Stephanie"
def test_with_integer(self):
docs = self.db.find({"name.first": "Stephanie", "age": 48})
assert len(docs) == 1
assert docs[0]["name"]["first"] == "Stephanie"
assert docs[0]["age"] == 48
def test_with_boolean(self):
docs = self.db.find({"name.first": "Stephanie", "manager": False})
assert len(docs) == 1
assert docs[0]["name"]["first"] == "Stephanie"
assert docs[0]["manager"] == False
def test_with_array(self):
faves = ["Ruby", "C", "Python"]
docs = self.db.find({"name.first": "Stephanie", "favorites": faves})
assert docs[0]["name"]["first"] == "Stephanie"
assert docs[0]["favorites"] == faves
def test_array_ref(self):
docs = self.db.find({"favorites.1": "Python"})
assert len(docs) == 4
for d in docs:
assert "Python" in d["favorites"]
# Nested Level
docs = self.db.find({"favorites.0.2": "Python"})
print len(docs)
assert len(docs) == 1
for d in docs:
assert "Python" in d["favorites"][0][2]
def test_number_ref(self):
docs = self.db.find({"11111": "number_field"})
assert len(docs) == 1
assert docs[0]["11111"] == "number_field"
docs = self.db.find({"22222.33333": "nested_number_field"})
assert len(docs) == 1
assert docs[0]["22222"]["33333"] == "nested_number_field"
def test_lt(self):
docs = self.db.find({"age": {"$lt": 22}})
assert len(docs) == 0
docs = self.db.find({"age": {"$lt": 23}})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": {"$lt": 33}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (1, 9)
docs = self.db.find({"age": {"$lt": 34}})
assert len(docs) == 3
for d in docs:
assert d["user_id"] in (1, 7, 9)
docs = self.db.find({"company": {"$lt": "Dreamia"}})
assert len(docs) == 1
assert docs[0]["company"] == "Affluex"
def test_lte(self):
docs = self.db.find({"age": {"$lte": 21}})
assert len(docs) == 0
docs = self.db.find({"age": {"$lte": 22}})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": {"$lte": 33}})
assert len(docs) == 3
for d in docs:
assert d["user_id"] in (1, 7, 9)
docs = self.db.find({"company": {"$lte": "Dreamia"}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (0, 11)
def test_eq(self):
docs = self.db.find({"age": 21})
assert len(docs) == 0
docs = self.db.find({"age": 22})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": {"$eq": 22}})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": 33})
assert len(docs) == 1
assert docs[0]["user_id"] == 7
def test_ne(self):
docs = self.db.find({"age": {"$ne": 22}})
assert len(docs) == len(user_docs.DOCS) - 1
for d in docs:
assert d["age"] != 22
docs = self.db.find({"$not": {"age": 22}})
assert len(docs) == len(user_docs.DOCS) - 1
for d in docs:
assert d["age"] != 22
def test_gt(self):
docs = self.db.find({"age": {"$gt": 77}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (3, 13)
docs = self.db.find({"age": {"$gt": 78}})
assert len(docs) == 1
assert docs[0]["user_id"] == 3
docs = self.db.find({"age": {"$gt": 79}})
assert len(docs) == 0
docs = self.db.find({"company": {"$gt": "Zialactic"}})
assert len(docs) == 0
def test_gte(self):
docs = self.db.find({"age": {"$gte": 77}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (3, 13)
docs = self.db.find({"age": {"$gte": 78}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (3, 13)
docs = self.db.find({"age": {"$gte": 79}})
assert len(docs) == 1
assert docs[0]["user_id"] == 3
docs = self.db.find({"age": {"$gte": 80}})
assert len(docs) == 0
docs = self.db.find({"company": {"$gte": "Zialactic"}})
assert len(docs) == 1
assert docs[0]["company"] == "Zialactic"
def test_and(self):
docs = self.db.find({"age": 22, "manager": True})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": 22, "manager": False})
assert len(docs) == 0
docs = self.db.find({"$and": [{"age": 22}, {"manager": True}]})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"$and": [{"age": 22}, {"manager": False}]})
assert len(docs) == 0
docs = self.db.find({"$text": "Ramona", "age": 22})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"$and": [{"$text": "Ramona"}, {"age": 22}]})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"$and": [{"$text": "Ramona"}, {"$text": "Floyd"}]})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
def test_or(self):
docs = self.db.find({"$or": [{"age": 22}, {"age": 33}]})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (7, 9)
q = {"$or": [{"$text": "Ramona"}, {"$text": "Stephanie"}]}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (0, 9)
q = {"$or": [{"$text": "Ramona"}, {"age": 22}]}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 9
def test_and_or(self):
q = {
"age": 22,
"$or": [
{"manager": False},
{"location.state": "Missouri"}
]
}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 9
q = {
"$or": [
{"age": 22},
{"age": 43, "manager": True}
]
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (9, 10)
q = {
"$or": [
{"$text": "Ramona"},
{"age": 43, "manager": True}
]
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (9, 10)
def test_nor(self):
docs = self.db.find({"$nor": [{"age": 22}, {"age": 33}]})
assert len(docs) == 13
for d in docs:
assert d["user_id"] not in (7, 9)
def test_in_with_value(self):
docs = self.db.find({"age": {"$in": [1, 5]}})
assert len(docs) == 0
docs = self.db.find({"age": {"$in": [1, 5, 22]}})
assert len(docs) == 1
assert docs[0]["user_id"] == 9
docs = self.db.find({"age": {"$in": [1, 5, 22, 31]}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (1, 9)
docs = self.db.find({"age": {"$in": [22, 31]}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (1, 9)
# Limits on boolean clauses?
docs = self.db.find({"age": {"$in": range(1000)}})
assert len(docs) == 15
def test_in_with_array(self):
vals = ["Random Garbage", 52, {"Versions": {"Alpha": "Beta"}}]
docs = self.db.find({"favorites": {"$in": vals}})
assert len(docs) == 1
assert docs[0]["user_id"] == 1
vals = ["Lisp", "Python"]
docs = self.db.find({"favorites": {"$in": vals}})
assert len(docs) == 10
vals = [{"val1": 1, "val2": "val2"}]
docs = self.db.find({"test_in": {"$in": vals}})
assert len(docs) == 1
assert docs[0]["user_id"] == 2
def test_nin_with_value(self):
docs = self.db.find({"age": {"$nin": [1, 5]}})
assert len(docs) == len(user_docs.DOCS)
docs = self.db.find({"age": {"$nin": [1, 5, 22]}})
assert len(docs) == len(user_docs.DOCS) - 1
for d in docs:
assert d["user_id"] != 9
docs = self.db.find({"age": {"$nin": [1, 5, 22, 31]}})
assert len(docs) == len(user_docs.DOCS) - 2
for d in docs:
assert d["user_id"] not in (1, 9)
docs = self.db.find({"age": {"$nin": [22, 31]}})
assert len(docs) == len(user_docs.DOCS) - 2
for d in docs:
assert d["user_id"] not in (1, 9)
# Limits on boolean clauses?
docs = self.db.find({"age": {"$nin": range(1000)}})
assert len(docs) == 0
def test_nin_with_array(self):
vals = ["Random Garbage", 52, {"Versions": {"Alpha": "Beta"}}]
docs = self.db.find({"favorites": {"$nin": vals}})
assert len(docs) == len(user_docs.DOCS) - 1
for d in docs:
assert d["user_id"] != 1
vals = ["Lisp", "Python"]
docs = self.db.find({"favorites": {"$nin": vals}})
assert len(docs) == 5
vals = [{"val1": 1, "val2": "val2"}]
docs = self.db.find({"test_in": {"$nin": vals}})
assert len(docs) == 0
def test_all(self):
vals = ["Ruby", "C", "Python", {"Versions": {"Alpha": "Beta"}}]
docs = self.db.find({"favorites": {"$all": vals}})
assert len(docs) == 1
assert docs[0]["user_id"] == 1
# This matches where favorites either contains
# the nested array, or is the nested array. This is
# notably different than the non-nested array in that
# it does not match a re-ordered version of the array.
# The fact that user_id 14 isn't included demonstrates
# this behavior.
vals = [["Lisp", "Erlang", "Python"]]
docs = self.db.find({"favorites": {"$all": vals}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (3, 9)
def test_exists_field(self):
docs = self.db.find({"exists_field": {"$exists": True}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (7, 8)
docs = self.db.find({"exists_field": {"$exists": False}})
assert len(docs) == len(user_docs.DOCS) - 2
for d in docs:
assert d["user_id"] not in (7, 8)
def test_exists_array(self):
docs = self.db.find({"exists_array": {"$exists": True}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (9, 10)
docs = self.db.find({"exists_array": {"$exists": False}})
assert len(docs) == len(user_docs.DOCS) - 2
for d in docs:
assert d["user_id"] not in (9, 10)
def test_exists_object(self):
docs = self.db.find({"exists_object": {"$exists": True}})
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (11, 12)
docs = self.db.find({"exists_object": {"$exists": False}})
assert len(docs) == len(user_docs.DOCS) - 2
for d in docs:
assert d["user_id"] not in (11, 12)
def test_exists_object_member(self):
docs = self.db.find({"exists_object.should": {"$exists": True}})
assert len(docs) == 1
assert docs[0]["user_id"] == 11
docs = self.db.find({"exists_object.should": {"$exists": False}})
assert len(docs) == len(user_docs.DOCS) - 1
for d in docs:
assert d["user_id"] != 11
def test_exists_and(self):
q = {"$and": [
{"manager": {"$exists": True}},
{"exists_object.should": {"$exists": True}}
]}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 11
q = {"$and": [
{"manager": {"$exists": False}},
{"exists_object.should": {"$exists": True}}
]}
docs = self.db.find(q)
assert len(docs) == 0
# Translates to manager exists or exists_object.should doesn't
# exist, which will match all docs
q = {"$not": q}
docs = self.db.find(q)
assert len(docs) == len(user_docs.DOCS)
def test_value_chars(self):
q = {"complex_field_value": "+-(){}[]^~&&*||\"\\/?:!"}
docs = self.db.find(q)
assert len(docs) == 1
def test_regex(self):
docs = self.db.find({
"age": {"$gt": 40},
"location.state": {"$regex": "(?i)new.*"}
})
assert len(docs) == 2
assert docs[0]["user_id"] == 2
assert docs[1]["user_id"] == 10
# test lucene syntax in $text
@unittest.skipUnless(mango.has_text_service(), "requires text service")
class ElemMatchTests(mango.FriendDocsTextTests):
def test_elem_match_non_object(self):
q = {"bestfriends":{
"$elemMatch":
{"$eq":"Wolverine", "$eq":"Cyclops"}
}
}
docs = self.db.find(q)
print len(docs)
assert len(docs) == 1
assert docs[0]["bestfriends"] == ["Wolverine", "Cyclops"]
q = {"results": {"$elemMatch": {"$gte": 80, "$lt": 85}}}
docs = self.db.find(q)
print len(docs)
assert len(docs) == 1
assert docs[0]["results"] == [82, 85, 88]
def test_elem_match(self):
q = {"friends": {
"$elemMatch":
{"name.first": "Vargas"}
}
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (0, 1)
q = {
"friends": {
"$elemMatch": {
"name.first": "Ochoa",
"name.last": "Burch"
}
}
}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 4
# Check that we can do logic in elemMatch
q = {
"friends": {"$elemMatch": {
"name.first": "Ochoa", "type": "work"
}}
}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 1
q = {
"friends": {
"$elemMatch": {
"name.first": "Ochoa",
"$or": [
{"type": "work"},
{"type": "personal"}
]
}
}
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (1, 4)
# Same as last, but using $in
q = {
"friends": {
"$elemMatch": {
"name.first": "Ochoa",
"type": {"$in": ["work", "personal"]}
}
}
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (1, 4)
q = {
"$and": [{
"friends": {
"$elemMatch": {
"id": 0,
"name": {
"$exists": True
}
}
}
},
{
"friends": {
"$elemMatch": {
"$or": [
{
"name": {
"first": "Campos",
"last": "Freeman"
}
},
{
"name": {
"$in": [{
"first": "Gibbs",
"last": "Mccarty"
},
{
"first": "Wilkins",
"last": "Chang"
}
]
}
}
]
}
}
}
]
}
docs = self.db.find(q)
assert len(docs) == 3
for d in docs:
assert d["user_id"] in (10, 11,12)
@unittest.skipUnless(mango.has_text_service(), "requires text service")
class AllMatchTests(mango.FriendDocsTextTests):
def test_all_match(self):
q = {"friends": {
"$allMatch":
{"type": "personal"}
}
}
docs = self.db.find(q)
assert len(docs) == 2
for d in docs:
assert d["user_id"] in (8, 5)
# Check that we can do logic in allMatch
q = {
"friends": {
"$allMatch": {
"name.first": "Ochoa",
"$or": [
{"type": "work"},
{"type": "personal"}
]
}
}
}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 15
# Same as last, but using $in
q = {
"friends": {
"$allMatch": {
"name.first": "Ochoa",
"type": {"$in": ["work", "personal"]}
}
}
}
docs = self.db.find(q)
assert len(docs) == 1
assert docs[0]["user_id"] == 15
# Test numeric strings for $text
@unittest.skipUnless(mango.has_text_service(), "requires text service")
class NumStringTests(mango.DbPerClass):
@classmethod
def setUpClass(klass):
super(NumStringTests, klass).setUpClass()
klass.db.recreate()
if mango.has_text_service():
klass.db.create_text_index()
# not available for python 2.7.x
def isFinite(num):
not (math.isinf(num) or math.isnan(num))
@given(f=st.floats().filter(isFinite).map(str)
| st.floats().map(lambda f: f.hex()))
@example('NaN')
@example('Infinity')
def test_floating_point_val(self,f):
doc = {"number_string": f}
self.db.save_doc(doc)
q = {"$text": f}
docs = self.db.find(q)
if len(docs) == 1:
assert docs[0]["number_string"] == f
if len(docs) == 2:
if docs[0]["number_string"] != f:
assert docs[1]["number_string"] == f
q = {"number_string": f}
docs = self.db.find(q)
if len(docs) == 1:
assert docs[0]["number_string"] == f
if len(docs) == 2:
if docs[0]["number_string"] != f:
assert docs[1]["number_string"] == f