content/guides/04.connect/2.filter-rules.md
Filters are used in permissions, validations, and automations, as well as throughout the APIs and in extensions. All filters use standard syntax and operators which are described on this page.
| Operator | Description |
|---|---|
_eq <sup>[1]</sup> | Equals |
_neq <sup>[1]</sup> | Doesn't equal |
_lt | Less than |
_lte | Less than or equal to |
_gt | Greater than |
_gte | Greater than or equal to |
_in | Is one of |
_nin | Is not one of |
_null | Is null |
_nnull | Isn't null |
_contains | Contains |
_ncontains | Doesn't contain |
_icontains | Contains (case-insensitive) |
_nicontains | Doesn't contain (case-insensitive) |
_starts_with | Starts with |
_istarts_with | Starts with (case-insensitive) |
_nstarts_with | Doesn't start with |
_nistarts_with | Doesn't start with (case-insensitive) |
_ends_with | Ends with |
_iends_with | Ends with (case-insensitive) |
_nends_with | Doesn't end with |
_niends_with | Doesn't end with (case-insensitive) |
_between | Is between two values (inclusive) |
_nbetween | Is not between two values (inclusive) |
_empty | Is empty (null or falsy) |
_nempty | Isn't empty (null or falsy) |
_intersects <sup>[2]</sup> | Intersects a point |
_nintersects <sup>[2]</sup> | Doesn't intersect a point |
_intersects_bbox <sup>[2]</sup> | Intersects a bounding box |
_nintersects_bbox <sup>[2]</sup> | Doesn't intersect a bounding box |
_regex <sup>[3]</sup> | Regular expression (escape backslashes) |
_some <sup>[4]</sup> | At least one related value is true |
_none <sup>[4]</sup> | No related values are true |
<sup>[1]</sup> Compared value is not strictly typed for numeric values, allowing comparisons between numbers and their string representations.
<sup>[2]</sup> Only available on geometry fields.
<sup>[3]</sup> Only available in validation permissions.
<sup>[4]</sup> Only available on One to Many relationship fields.
{
"field": {
"operator": "value"
}
}
The field can exist on the current collection or a relational collection.
The operator must be any valid filter operator such as 'equals' or 'contains'.
The value can be any fixed static value or one of the provided dynamic variables.
::example
This filter checks the title field contains the case-sensitive substring 'Directus':
{
"title": {
"_contains": "Directus"
}
}
::
You can filter items based on related data by nesting field names in your query. This allows you to query items not just by their own fields, but also by values in related collections.
You can filter items with Many-to-One relations by specifying values of their respective fields.
::example
For example, if you have an articles collection with a relational Many-to-One author field, you can filter articles based on the author's details—such as their name.
{
"author": {
"name": {
"_eq": "Rijk van Zanten"
}
}
}
::
When using Many-to-Many relationships, a junction table will be created and the filter applies to the junction table itself. For
example, if you have a books collection, with a Many-to-Many relationship to authors of each book, the junction collection will
probably be named books_authors and have 3 fields : id, books_id and authors_id. To filter specific books
depending on their authors you must go through the junction table and the authors_id field :
{
"authors": {
"authors_id": {
"name": {
"_eq": "Rijk van Zanten"
}
}
}
}
_some vs _none in One-to-Many and Many-to-ManyThe _some and _none filter operators can be used for filtering One-to-Many and Many-to-Many relational fields in API queries. They allow you to filter items based on conditions applied to their related collections.
The _some operator matches items where at least one related item meets the condition.
By default, Directus will apply the _some operator when querying One-to-Many and Many-to-Many relational queries.
::example
This filter matches all items where at least one related category has the name "Recipe":
{
"categories": {
"_some": {
"name": {
"_eq": "Recipe"
}
}
}
}
In the above, categories is the relational field in the collection. _some checks if at least one related category has the name "Recipe". If an item has one or more categories named "Recipe", it will be included in the result.
Since _some is applied by default, the below is equivalent:
{
"categories": {
"name": {
"_eq": "Recipe"
}
}
}
::
The _none operator matches items where none of the related items meet the condition.
::example This filter matches all items where none of the categories has the name "Recipe":
{
"categories": {
"_none": {
"name": {
"_eq": "Recipe"
}
}
}
}
In the above, categories is the relational field in the collection. _none checks if no related category has the name "Recipe". If an item has no categories named "Recipe", it will be included in the result.
::
| Variable | Description |
|---|---|
$CURRENT_USER | The primary key of the currently authenticated user. |
$CURRENT_ROLE | The primary key of the role for the currently authenticated user |
$NOW | The current timestamp |
$NOW(<adjustment>) | The current timestamp plus/minus a given distance, for example $NOW(-1 year), $NOW(+2 hours) |
::example{title="Examples"}
::tabs
::div{class="pr-6"}
---
label: "$CURRENT_USER"
---
json { "owner": { "_eq": "$CURRENT_USER" } }
::
::div{class="pr-6"}
---
label: "$NOW"
---
```json
{
"datetime": {
"_lte": "$NOW"
}
}
```
::
:: ::
::example
Nested user and role variables in permissions
When configuring permissions, $CURRENT_USER and $CURRENT_ROLE allow you to specify any related field, such as $CURRENT_ROLE.name or $CURRENT_USER.avatar.filesize.
::
You can group multiple rules using the _and or _or logical operators. Each logical operator holds an array of filter rules. Logical operators can be nested directly inside of each other, but not inside of other filter rules.
{
"_and": [
{
"field": {
"operator": "value"
}
},
{
"field": {
"operator": "value"
}
}
]
}
::example
{
"_or": [
{
"_and": [
{
"user_created": {
"_eq": "$CURRENT_USER"
}
},
{
"status": {
"_in": ["published", "draft"]
}
}
]
},
{
"_and": [
{
"user_created": {
"_neq": "$CURRENT_USER"
}
},
{
"status": {
"_in": ["published"]
}
}
]
}
]
}
::
:partial{content="query-functions"}
::example
{
_and: [
{
"year(published_date)": {
_eq: 1968,
},
},
{
"month(published_date)": {
_eq: 4,
},
},
],
},
::
Filters allow you to query relations from collections directly.
For cases where you wish to query an indirect relation, such as countries to which cities have an M2O relation, you can use the $FOLLOW(target-collection, relation-field) syntax.
This is useful when there is a relation from the target collection to the current collection, but no relation field has been configured in the current collection to the target collection.
::example
There exists both a cities and a countries collection. cities have an M2O relationship with countries via the country_id field. You can query countries via the fields of its related cities using the following:
{
"filter": {
"name": "Germany",
"$FOLLOW(cities, country_id)": {
"name": "Berlin"
}
}
}
::