Module: Mongoid::Criteria::Queryable::Storable Private
- Included in:
- Mongoid::Criteria::Queryable
- Defined in:
- lib/mongoid/criteria/queryable/storable.rb
Overview
This module is part of a private API. You should avoid using this module if possible, as it may be removed or be changed in the future.
This module encapsulates methods that write query expressions into the Criteria’s selector.
The query expressions must have already been expanded as necessary. The methods of this module do not perform processing on expression values.
Methods in this module do not handle negation - if negation is needed, it must have already been handled upstream of these methods.
Instance Method Summary collapse
-
#add_field_expression(field, value) ⇒ Storable
private
Adds a field expression to the query.
-
#add_logical_operator_expression(operator, op_expr) ⇒ Storable
private
Adds a logical operator expression to the selector.
-
#add_one_expression(field, value) ⇒ Storable
private
Adds an arbitrary expression to the query.
-
#add_operator_expression(operator, op_expr) ⇒ Storable
private
Adds an operator expression to the selector.
Instance Method Details
#add_field_expression(field, value) ⇒ Storable
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Adds a field expression to the query.
field
must be a field name, and it must be a string. The upstream code must have converted other field/key types to the simple string form by the time this method is invoked.
value
can be of any type, it is written into the selector unchanged.
This method performs no processing on the provided field value.
Mutates the receiver.
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 |
# File 'lib/mongoid/criteria/queryable/storable.rb', line 36 def add_field_expression(field, value) unless field.is_a?(String) raise ArgumentError, "Field must be a string: #{field}" end if field.start_with?('$') raise ArgumentError, "Field cannot be an operator (i.e. begin with $): #{field}" end if selector[field] # We already have a restriction by the field we are trying # to restrict, combine the restrictions. if value.is_a?(Hash) && selector[field].is_a?(Hash) && value.keys.all? { |key| key_s = key.to_s key_s.start_with?('$') && !selector[field].keys.map(&:to_s).include?(key_s) } then # Multiple operators can be combined on the same field by # adding them to the existing hash. new_value = selector[field].merge(value) selector.store(field, new_value) elsif selector[field] != value add_operator_expression('$and', [{field => value}]) end else selector.store(field, value) end self end |
#add_logical_operator_expression(operator, op_expr) ⇒ Storable
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Adds a logical operator expression to the selector.
This method only handles logical operators ($and, $nor and $or). It raises ArgumentError if called with another operator. Note that in MQL, $not is a field-level operator and not a query-level one, and therefore $not is not handled by this method.
This method takes the operator and the operator value expression separately for callers’ convenience. It can be considered to handle storing the hash {operator => op_expr}.
If the selector consists of a single condition which is the specified operator (on the top level), the new condition given in op_expr is added to the existing conditions for the specified operator. For example, if the selector is currently:
{'$or' => [{'hello' => 'world'}]}
… and operator is ‘$or’ and op_expr is [{‘test’ => 123’}], the resulting selector will be:
{'$or' => [{'hello' => 'world'}, {'test' => 123}]}
This method always adds the new conditions as additional requirements; in other words, it does not implement the ActiveRecord or/nor behavior where the receiver becomes one of the operands. It is expected that code upstream of this method implements such behavior.
This method does not simplify values (i.e. if the selector is currently empty and operator is $and, op_expr is written to the selector with $and even if the $and can in principle be elided). Such simplification is also expected to have already been performed by the upstream code.
This method mutates the receiver.
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 |
# File 'lib/mongoid/criteria/queryable/storable.rb', line 108 def add_logical_operator_expression(operator, op_expr) unless operator.is_a?(String) raise ArgumentError, "Operator must be a string: #{operator}" end unless %w($and $nor $or).include?(operator) raise ArgumentError, "This method only handles logical operators ($and, $nor, $or). Operator given: #{operator}" end unless op_expr.is_a?(Array) raise Errors::InvalidQuery, "#{operator} argument must be an array: #{Errors::InvalidQuery.truncate_expr(op_expr)}" end if selector.length == 1 && selector.keys.first == operator new_value = selector.values.first + op_expr selector.store(operator, new_value) elsif operator == '$and' || selector.empty? # $and can always be added to top level and it will be combined # with whatever other conditions exist. if current_value = selector[operator] new_value = current_value + op_expr selector.store(operator, new_value) else selector.store(operator, op_expr) end else # Other operators need to be added separately if selector[operator] add_logical_operator_expression('$and', [operator => op_expr]) else selector.store(operator, op_expr) end end self end |
#add_one_expression(field, value) ⇒ Storable
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Adds an arbitrary expression to the query.
Field can either be a field name or an operator.
Mutates the receiver.
223 224 225 226 227 228 229 230 231 232 233 |
# File 'lib/mongoid/criteria/queryable/storable.rb', line 223 def add_one_expression(field, value) unless field.is_a?(String) raise ArgumentError, "Field must be a string: #{field}" end if field.start_with?('$') add_operator_expression(field, value) else add_field_expression(field, value) end end |
#add_operator_expression(operator, op_expr) ⇒ Storable
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Adds an operator expression to the selector.
This method takes the operator and the operator value expression separately for callers’ convenience. It can be considered to handle storing the hash {operator => op_expr}.
The operator value can be of any type.
If the selector already has the specified operator in it (on the top level), the new condition given in op_expr is added to the existing conditions for the specified operator. This is straightforward for $and; for other logical operators, the behavior of this method is to add the new conditions to the existing operator. For example, if the selector is currently:
{'foo' => 'bar', '$or' => [{'hello' => 'world'}]}
… and operator is ‘$or’ and op_expr is {‘test’ => 123’}, the resulting selector will be:
{'foo' => 'bar', '$or' => [{'hello' => 'world'}, {'test' => 123}]}
This does not implement an OR between the existing selector and the new operator expression - handling this is the job of upstream methods. This method simply stores op_expr into the selector on the assumption that the existing selector is the correct left hand side of the operation already.
For non-logical query-level operators like $where and $text, if there already is a top-level operator with the same name, the op_expr is added to the selector via a top-level $and operator, thus producing a selector having both operator values.
This method does not simplify values (i.e. if the selector is currently empty and operator is $and, op_expr is written to the selector with $and even if the $and can in principle be elided).
This method mutates the receiver.
188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 |
# File 'lib/mongoid/criteria/queryable/storable.rb', line 188 def add_operator_expression(operator, op_expr) unless operator.is_a?(String) raise ArgumentError, "Operator must be a string: #{operator}" end unless operator.start_with?('$') raise ArgumentError, "Operator must begin with $: #{operator}" end if %w($and $nor $or).include?(operator) return add_logical_operator_expression(operator, op_expr) end # For other operators, if the operator already exists in the # query, add the new condition with $and, otherwise add the # new condition to the top level. if selector[operator] add_logical_operator_expression('$and', [{operator => op_expr}]) else selector.store(operator, op_expr) end self end |