Skip to content

Commit 5ccec37

Browse files
authored
adding support for boolean checks (#611)
1 parent 57fe8a2 commit 5ccec37

File tree

4 files changed

+58
-2
lines changed

4 files changed

+58
-2
lines changed

Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ sync: $(INSTALL_STAMP)
5353
lint: $(INSTALL_STAMP) dist
5454
$(POETRY) run isort --profile=black --lines-after-imports=2 ./tests/ $(NAME) $(SYNC_NAME)
5555
$(POETRY) run black ./tests/ $(NAME)
56-
$(POETRY) run flake8 --ignore=W503,E501,F401,E731 ./tests/ $(NAME) $(SYNC_NAME)
56+
$(POETRY) run flake8 --ignore=W503,E501,F401,E731,E712 ./tests/ $(NAME) $(SYNC_NAME)
5757
$(POETRY) run mypy ./tests/ $(NAME) $(SYNC_NAME) --ignore-missing-imports --exclude migrate.py --exclude _compat\.py$
5858
$(POETRY) run bandit -r $(NAME) $(SYNC_NAME) -s B608
5959

aredis_om/model/model.py

+13-1
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,8 @@ class Operators(Enum):
116116
STARTSWITH = 14
117117
ENDSWITH = 15
118118
CONTAINS = 16
119+
TRUE = 17
120+
FALSE = 18
119121

120122
def __str__(self):
121123
return str(self.name)
@@ -582,6 +584,8 @@ def resolve_field_type(
582584
"Only lists and tuples are supported for multi-value fields. "
583585
f"Docs: {ERRORS_URL}#E4"
584586
)
587+
elif field_type is bool:
588+
return RediSearchFieldTypes.TAG
585589
elif any(issubclass(field_type, t) for t in NUMERIC_TYPES):
586590
# Index numeric Python types as NUMERIC fields, so we can support
587591
# range queries.
@@ -676,7 +680,11 @@ def resolve_value(
676680
separator_char,
677681
)
678682
return ""
679-
if isinstance(value, int):
683+
if isinstance(value, bool):
684+
result = "@{field_name}:{{{value}}}".format(
685+
field_name=field_name, value=value
686+
)
687+
elif isinstance(value, int):
680688
# This if will hit only if the field is a primary key of type int
681689
result = f"@{field_name}:[{value} {value}]"
682690
elif separator_char in value:
@@ -1814,6 +1822,8 @@ def schema_for_type(cls, name, typ: Any, field_info: PydanticFieldInfo):
18141822
return ""
18151823
embedded_cls = embedded_cls[0]
18161824
schema = cls.schema_for_type(name, embedded_cls, field_info)
1825+
elif typ is bool:
1826+
schema = f"{name} TAG"
18171827
elif any(issubclass(typ, t) for t in NUMERIC_TYPES):
18181828
vector_options: Optional[VectorFieldOptions] = getattr(
18191829
field_info, "vector_options", None
@@ -2121,6 +2131,8 @@ def schema_for_type(
21212131
raise sortable_tag_error
21222132
if case_sensitive is True:
21232133
schema += " CASESENSITIVE"
2134+
elif typ is bool:
2135+
schema = f"{path} AS {index_field_name} TAG"
21242136
elif any(issubclass(typ, t) for t in NUMERIC_TYPES):
21252137
schema = f"{path} AS {index_field_name} NUMERIC"
21262138
elif issubclass(typ, str):

docs/getting_started.md

+21
Original file line numberDiff line numberDiff line change
@@ -702,6 +702,27 @@ Customer.find((Customer.last_name == "Brookins") | (
702702
) & (Customer.last_name == "Smith")).all()
703703
```
704704

705+
### Saving and querying Boolean values
706+
707+
For historical reasons, saving and querying Boolean values is not supported in `HashModels`, however in JSON models,
708+
you may store and query Boolean values using the `==` syntax:
709+
710+
```python
711+
from redis_om import (
712+
Field,
713+
JsonModel,
714+
Migrator
715+
)
716+
717+
class Demo(JsonModel):
718+
b: bool = Field(index=True)
719+
720+
Migrator().run()
721+
d = Demo(b=True)
722+
d.save()
723+
res = Demo.find(Demo.b == True)
724+
```
725+
705726
## Calling Other Redis Commands
706727

707728
Sometimes you'll need to run a Redis command directly. Redis OM supports this through the `db` method on your model's class. This returns a connected Redis client instance which exposes a function named for each Redis command. For example, let's perform some basic set operations:

tests/test_json_model.py

+23
Original file line numberDiff line numberDiff line change
@@ -971,3 +971,26 @@ async def test_xfix_queries(m):
971971

972972
result = await m.Member.find(m.Member.bio % "*ack*").first()
973973
assert result.first_name == "Steve"
974+
975+
976+
@py_test_mark_asyncio
977+
async def test_boolean():
978+
class Example(JsonModel):
979+
b: bool = Field(index=True)
980+
d: datetime.date = Field(index=True)
981+
name: str = Field(index=True)
982+
983+
await Migrator().run()
984+
985+
ex = Example(b=True, name="steve", d=datetime.date.today())
986+
exFalse = Example(b=False, name="foo", d=datetime.date.today())
987+
await ex.save()
988+
await exFalse.save()
989+
res = await Example.find(Example.b == True).first()
990+
assert res.name == "steve"
991+
992+
res = await Example.find(Example.b == False).first()
993+
assert res.name == "foo"
994+
995+
res = await Example.find(Example.d == ex.d and Example.b == True).first()
996+
assert res.name == ex.name

0 commit comments

Comments
 (0)