Predicate
Boolean (truth-value) expressions that can be evaluated, manipulated, optimized, translated to code, etc.
Example(s)
# Let's build a simple predicate for 'x = 2 and not(y <= 3)'
p = Predicate.eq(:x, 2) & !Predicate.lte(:y, 3)
p.evaluate(:x => 2, :y => 6)
# => true
p.evaluate(:x => 2, :y => 3)
# => false
p.assert!(:x => 2, :y => 3)
# => Minitest::Assertion("Expected false to be truthy")
When building complex expressions, you can use the dsl method.
# This builds the same predicate
p = Predicate.dsl{
  eq(:x, 2) & !lte(:y, 3)
}
The dsl block also have all predicates in camelCase, negated, and full text
variants:
p = Predicate.dsl{
  notEq(:x, "foo") & hasSize(:y, 1..10) & lessThan(:z, 3)
}
If you have complex expressions where many members apply to the same variable,
a currying dsl extension is provided. It allows using all dsl methods
while omitting their first argument.
# Instead of this
p = Predicate.gt(:x, 1) & Predicate.lt(:x, 10)
# or this
p = Predicate.dsl{
  gt(:x, 1) & lt(:x, 10)
}
# do this
p = Predicate.(:x){
  gt(1) & lt(10)
}
p.evaluate(:x => 6)
# => true
Predicate also works if you want to evaluate an expression on a single object
without having to introduce a variable like :x...
p = Predicate.{
  gt(1) & lt(10)
}
p.evaluate(6)
# => true
... or, in contrast, if you want to evaluate boolean expressions over more
complex data structures that a flat Hash like {:x => 6, ...}
x, y = Predicate.vars("items.0.price", "items.1.price")
p = Predicate.eq(x, 6) & Predicate.lt(y, 10)
p.evaluate({
  items: [
    { name: "Candy", price: 6 },
    { name: "Crush", price: 4 }
  ]
})
# => true
The following sections explain a) why we created this library, b) how to build expressions, c) what operators are available, and d) how abstract variables work and what features are supported when using them (because not all are).
Rationale
This reusable library is used in various ruby gems developed and maintained by Enspirit where boolean expressions are first-class citizen. It provides a common API for expressing, evaluating, and manipulating them.
The library represents an expression as an AST internally. This allows for
subsequent manipulations & reasoning. Please check the Predicate::Factory
module for details.
Best-effort simplifications are also performed at construction and when
boolean logic is used (and, or, not). For instance, eq(:x, 6) & eq(:x, 10)
yields a contradiction predicate. There is currently no way to disable those
simplifications that were initially implemented for Bmg.
Building expressions
The following list of operators is currently available.
True and False
Predicate.tautology                  # aka True
Predicate.contradiction              # aka False
Logical operators
For every valid Predicate instances p and q:
p & q                                # Boolean conjunction
p | q                                # Boolean disjunction
!p                                   # Boolean negation
Comparison operators
Predicate.eq(:x, 2)                  # x = 2
Predicate.eq(:x, :y)                 # x = y
Predicate.neq(:x, 2)                 # x != 2
Predicate.neq(:x, :y)                # x != y
Predicate.lt(:x, 2)                  # x < 2
Predicate.lt(:x, :y)                 # x < y
Predicate.lte(:x, 2)                 # x <= 2
Predicate.lte(:x, :y)                # x <= y
Predicate.gt(:x, 2)                  # x > 2
Predicate.gt(:x, :y)                 # x > y
Predicate.gte(:x, 2)                 # x >= 2
Predicate.gte(:x, :y)                # x >= y
Shortcuts (translated immediately, no trace kept in AST) :
Predicate.eq(x: 2, y: 6)             # eq(:x, 2) & eq(:y, 6)
Predicate.eq(x: 2, y: :z)            # eq(:x, 2) & eq(:y, :z)
# ... and so on for neq, lt, lte, gt, gte
Predicate.between(:x, l, h)          # gte(:x, l) & lte(:x, h), for all l and h
Predicate.in(:x, 1..10)              # gte(:x, 1) & lte(:x, 10)
Predicate.in(:x, 1...10)             # gte(:x, 1) & lt(:x, 10)
#
Predicate.is_null(:x)                # eq(:x, nil)
Set-based operators
Predicate.in(:x, [2, 4, 6])          # x ∈ {2, 4, 6}
Predicate.in(:x, :y)                 # x ∈ y
Predicate.intersect(:x, [2, 4, 6])   # x ∩ {2, 4, 6} ≠ ∅
Predicate.intersect(:x, :y)          # x ∩ y ≠ ∅
Predicate.subset(:x, [2, 4, 6])      # x ⊆ {2, 4, 6}
Predicate.subset(:x, :y)             # x ⊆ y
Predicate.superset(:x, [2, 4, 6])    # x ⊇ {2, 4, 6}
Predicate.superset(:x, :y)           # x ⊇ y
Other operators
The following operators have no clear mathematical semantics. Their semantics depends on the underlying type system. Most are currently not supported outside of ruby (e.g. SQL compilation). The documentation below applies to a Ruby usage.
Predicate.match(:x, /abc/)           # ruby's ===
Predicate.empty(:x)                  # ruby's empty?
Predicate.has_size(:x, 1..10)        # ruby's size and ===
Predicate.has_size(:x, 10)           # Same as has_size(:x, 10..10)
Predicate.has_size(:x, :y)           # y must resolve to a Range or Integer
Shortcuts (translated immediately, no trace kept in AST) :
Predicate.min_size(:x, 10)           # has_size(:x, 10..)
Predicate.max_size(:x, 10)           # has_size(:x, 0..10)
Native expressions
Ruby Proc can be used to capture complex predicates. Native predicates always
receive the top evaluation context as first argument.
p = Predicate.native(->(t){
  # t here is the {:x => 2, :y => 6} Hash below
  Foo::Bar.call_to_ruby_code?(t)
})
p.evaluate(:x => 2, :y => 6)
Resulting predicates cannot be translated to, e.g. SQL, and typically prevent optimizations and manipulations:
Available operators
The following operators are available on predicates.
Evaluate
Predicate#evaluate takes a Hash mapping each free variable to a value,
and returns the Boolean evaluation of the expression.
# Let's build a simple predicate for 'x = 2 and not(y <= 3)'
p = Predicate.eq(:x, 2) & !Predicate.lte(:y, 3)
p.evaluate(:x => 2, :y => 6)
# => true
Assert
Predicate#assert! takes a takes a Hash mapping each free variable to a value,
and raises a Minitest::AssertionError if the predicate evaluates to false.
A best effort is made to provide readable error messages on the assertion error.
# Let's build a simple predicate for 'x = 2 and not(y <= 3)'
p = Predicate.eq(:x, 2) & !Predicate.lte(:y, 3)
p.evaluate(:x => 2, :y => 8)
# => Expected false to be truthy (Minitest::Assertion)
Rename
Predicate#rename allows renaming variables.
p = Predicate.eq(:x, 4)       # x = 4
p = p.rename(:x => :z)        # z = 4
Bind
Predicate#bind allows late binding of placeholders to values.
pl = Predicate.placeholder
p = Predicate.eq(:x, pl)      # x = _
p = p.bind(pl, 5)             # x = 5
p.evaluate(:x => 10)
# => false
Quality & Unqualify
Predicate#qualify allows adding a qualifier to each variable, for
disambiguation when composing predicates from different contexts.
Predicate#unqualify does the opposite.
p = Predicate.eq(:x, 2)       # x = 2
p.qualify(:t)                 # t.x = 2
p.unqualify                   # x = 2
Qualify accepts a Hash to use different qualifiers for variables.
p = Predicate.eq(x: 2, y: 4)  # x = 2 & y = 4
p.qualify(:x => :t, :y => :s)       # t.x = 2 & s.y = 4
And split
Predicate#and_split split a predicate p as two predicates p1 and p2
so that p <=> p1 & p2 and p2 makes no reference to any variable of the
given list.
p = Predicate.eq(x: 2, y: 4)  # x = 2 & y = 4
p1, p2 = p.and_split([:x])    # p1 is x = 2 ; p2 is y = 4
Observe that and_split is always possible but may degenerate to an
uninteresting p2, typically when disjunctions are used. For instance,
p = Predicate.eq(x: 2) | Predicate.eq(y: 4)  # x = 2 | y = 4
p1, p2 = p.and_split([:x])    # p1 is x = 2 | y = 4 ; p2 is true
Attr split
Predicate#attr_split can be used to split a predicate p as n+1 predicates
p1, p2, ..., pn, pz, such that p <=> p1 & p2 & ... & pn & pz. Each
predicate pi makes references to variable i only, except pz which can
reference all of them.
The result is a Hash mapping each variable to its predicate. A nil key maps
to pz.
p = Predicate.eq(x: 2, y: 4)  # x = 2 & y = 4
split = p.attr_split
# => {
#   :x => Predicate.eq(:x, 2),
#   :y => Predicate.eq(:y, 4)
# }
Working with PostgreSQL
(experimental) Predicate supports compiling certain high-level expressions to PostgreSQL native operators. It works in an direct or indirect way:
require 'predicate'
require 'predicate/postgres'
# In direct way, you simply create the predicates using PostgreSQL's own
# operators
p = Predicate.pg_array_overlaps(:x, ['foo', 'bar'])
p.to_sequel
# In indirect way, you use high-level predicates and convert them to
# PostgreSQL later using `to_postgres`
p = Predicate.interect(:x, ['foo', 'bar'])
p = p.to_postgres
p.to_sequel
Only a few array operators & translations exist, and only on varchar[]
types. Additional support will be added later. The following
translations are implemented (and methods on the right directly available
on the Predicate class):
          -> pg_array_literal
intersect -> pg_array_overlaps
empty     -> pg_array_empty
Working with abstract variables
WARNING: this var feature is only compatible with Predicate#evaluate
and Predicate#bind so far. Other operators have not been tested and may fail
in unexpected ways or raise a NotImplementedError. Also, predicates using
abstract variables are not properly translated to e.g. SQL.
By default, Predicate expects variable identifiers to be represented by
ruby Symbols. #evaluate then takes a mapping between variables and values as
a Hash:
# :x and :y are variable identifiers
p = Predicate.eq(:x, 2) & !Predicate.lte(:y, 3)
# the Hash below is a mapping between variables and values
p.evaluate(:x => 2, :y => 6)
# => true
There are situations where you would like variables to be kept simple in expressions while evaluating the latter on complex data structures.
Predicate#var can be used as an abstraction mechanism in such cases.
It takes a variable definition as first argument and a semantics as second.
The semantics defines how a value is extracted when the variable value must
be evaluated.
Supported protocols are :dig, :send and :public_send. Only :dig
must be considered safe while the two other ones used with great care.
- :digrelies on Ruby's- digprotocol introduced in Ruby 2.3. It will work out of the box with Hash, Array, Struct, OpenStruct and more generally any object responding to- :dig:
  xyz = Predicate.var([:x, :y, :z], :dig)
  p = Predicate.eq(xyz, 2)
  p.evaluate({ :x => { :y => { :z => 2 } } })
  # => true
When using :dig the variable definition can be passed as a String
  that will be automatically decomposed for you. Variable names are
  transformed to Symbols and integer literals to Integers. You must
  use the explicit version above if you don't want those conversions.
  # this
  Predicate.var("x.0.y", :dig)
  # is equivalent to
  Predicate.var([:x, 0, :y], :dig)
- :sendrelies on Ruby's- __send__method and is generally less safe if variable definitions are not strictly controlled. But it allows evaluating predicates over any data structure made of pure ruby objects:
  class C
    attr_reader :x
    def initialize(x)
      @x = x
    end
  end
  xy = Predicate.var([:x, :y], :send)
  p = Predicate.eq(xy, 2)
  p.evaluate(C.new(OpenStruct.new(y: 2)))
  # => true
    ```
  The variable can similarly be passed as a dotted String that will be
  decomposed as a sequence of Symbols.
  ```ruby
  xy = Predicate.var("x.y", :send)
  p = Predicate.eq(xy, 2)
  p.evaluate(C.new(OpenStruct.new(y: 2)))
  # => true
- :public_sendis similar to- :sendbut slightly safer as it only allows calling Ruby's public methods.
Public API
This library follows semantics versioning 2.0. Its public API is:
- Class methods of the - Predicateclass, such as those covered in the "Building expressions" section above.
- DSL methods contributed by - Predicate::Factory,- Predicate::Sugar, and- Predicate::Dslmodules ; including dynamic ones (negation, camelCase, etc.)
- Instance methods of the - Predicateclass, such as those covered in the "Available operators" section above.
- Instance and class methods contributed by plugins (e.g. - predicate/sequel).
- Exception classes: - Predicate::NotSupportedError,- Predicate::UnboundErrorand- Predicate::TypeError.
The AST representation of predicate expressions is NOT part of the public API. We bump the minor version of the library when it changes, though.
Everything else is condidered private and may change any time (i.e. on patch releases).
Contributing
Please use github issues and pull requests, and favor the latter if possible.
This repository uses the help of jeny to
generate code snippets when adding new predicates. It supports predicate
and sugar snippets and add code to be completed in various places:
bundle exec jeny s predicate -d op_name:my_predicate -d arity:unary
bundle exec jeny s sugar     -d op_name:my_shortcut
Licence
This software is distributed by Enspirit SRL under a MIT Licence. Please contact Bernard Lambeau ([email protected]) with any question.