Inquire allows you to generate advanced query strings. Currently supports armet syntax query strings.
Table of Contents
- Installation
- The Problem
- A Solution
- Background
- Usage
- Examples
Inquire is available on the npm registry: inquire
You can install it with npm.
npm install inquire
Currently, query strings only conjoin predicates together with equality. Armet attempts to extend this in two ways:
- Allowing predicates to be conjoined, disjoined and negated.
- Allowing more than just equality, e.g. inequalities.
This ends up creating a problem. We now have to generate these advanced query strings. Generating these advanced query strings by hand gets unwieldy.
For example, let's say we are dealing with a REST api for shapes.
We can GET
on /api/shape
and return all of the shapes.
Now let's try to get more detailed in our result.
Let's say we want red shapes wider than 30 pixels or
we want all shapes with no more than 12 sides or
we want all of the squares that either aren't black or that were created by bob.
Our GET
request becomes:
/api/shape?(color=red&width>30);sides<=12;(shape=square&(color!=black;user=bob))
We need a better way to generate query strings for armet's consumption. Inquire allows you do the same query without trying to man-handle the string.
import Inquire
query = (("color" `eq` "red") |&| ("width" `gt` "30")) ||| ("sides" `le` "12") ||| (("shape" `eq` "square") |&| (("color" `ne` "black") ||| ("user" `eq` "bob")))
url = "/api/shape?" ++ show query
-- url => "/api/shape?(color=red&width>30);sides<=12;(shape=square&(color!=black;user=bob))"
LiveScript:
require! I: \inquire.Inquire
query = I.or(I.or(I.and(I.eq(\color)(\red))(I.gt(\width)(30)))(I.le(\sides)(12)))(I.and(I.eq(\shape)(\square))(I.or(I.ne(\color)(\black))(I.eq(\user)(\bob))))
url = "/api/shape?#{I.generate query}"
# url => '/api/shape?(color=red&width>30);sides<=12;(shape=square&(color!=black;user=bob))'
Javascript:
var I = require('inquire').Inquire;
var query = I.or(I.or(I.and(I.eq('color')('red'))(I.gt('width')(30)))(I.le('sides')(12)))(I.and(I.eq('shape')('square'))(I.or(I.ne('color')('black'))(I.eq('user')('bob'))))
var url = '/api/shape?' + I.generate(query)
// url => '/api/shape?(color=red&width>30);sides<=12;(shape=square&(color!=black;user=bob))'
Note: Inquire removes as many parens as logically possible.
Of course this still looks horrible to have to deal with because it's all on one line. So, as with most software issues, it's better to decompose this into smaller pieces and then rejoin the pieces at the end.
N.B. Please report any misinformation, pull requests are welcome!
At its core Inquire is a Boolean Algebra. This means that all of the properties of other boolean algebras can be applied to Inquire. This includes things like:
- Associativity
- Commutativity
- Idempotence
- DeMorgan's Laws
- Implication
- Equivalence
Because of this, we can optimize our queries. In some cases, we can optimize away the entire query!
The basic building block of Inquire is a predicate.
Each Predicate has a key, a relational operator and a value.
The supported relational operators are ==
, !=
, <
, <=
, >
, and >=
.
A key or a value can be any thing.
N.B. Currently in JS, only the basic JS types are supported (Boolean, Number, String).
We can construct more complex queries by joining our predicates together.
There are two binary combinators and a unary combinator: and
, or
and not
, respectively.
Also, since Boolean Algebras are Complemented Distributive Lattices, we get all of the properties of those for free as well!
The complemented part means that every query has a complement. We construct the complement for an entire query by wrapping it in a negation. This is the lazy way, and not actually true to the definition. The way to actually get the complement is to take each predicate and change its relational operator.
== |
<=> | != |
> |
<=> | <= |
< |
<=> | >= |
The distributive part means that our binary combinators can be distributed over each other. This is similar to how multiplication can distribute over addition in Semirings like the Natural numbers with 0, or Integers or the reals, etc. The difference is, each operation can distribute over the other.
So the following are equivalent: p AND (q OR r)
, (p OR q) AND (p OR r)
As well as: p OR (q AND r)
, (p AND q) OR (p AND r)
The lattice part has two equivalent interpretations. One states that there is a poset nestled underneath our structure. The other states that we have some of the axioms mentioned above.
Inquire has a few basic combinators on which most of everything else is built. Since Inquire is tied closely to armet, most of this is about filtering responses based on model attributes. However, the fundamentals are the same for any purpose.
Constructs an Inquire from two values with the relation ==
.
Will return results where the key
on the model equals the supplied val
Constructs an Inquire from two values with the relation !=
Will return results where the key
on the model does not equal the supplied val
Constructs an Inquire from two values with the relation >
Will return results where the key
on the model is greater than the supplied val
Constructs an Inquire from two values with the relation >=
Will return results where the key
on the model is greater than or equal to the supplied val
Constructs an Inquire from two values with the relation <
Will return results where the key
on the model is less than the supplied val
Constructs an Inquire from two values with the relation <=
Will return results where the key
on the model is less than or equal to the supplied val
Constructs an Inquire from two Inquires with the conjunction &
Will return results where both left
and right
subqueries are true.
Constructs an Inquire from two Inquires with the disjunction ;
Will return results where either one, or both, left
and right
subqueries are true.
Constructs an Inquire from one Inquire with the negation !
Will return results where subquery is false.
From these simple combinators, we can create more complex ideas:
Constructs an Inquire from two Inquires.
This is the direct definition of implication, namely (NOT p) OR q
.
Constructs an Inquire from two Inquires.
This is the direct definition of equivalence, namely (p AND q) OR ((NOT p) AND (NOT q))
.
Constructs an Inquire from two Inquires.
This is the direct definition of exclusive or, namely (p AND (NOT q)) OR ((NOT p) AND q)
.
Simplifies specific Inquires.
p AND (p OR q)
=> p
p OR (p AND q)
=> p
Reorders parens from what they are now to the other possible way.
p AND (q AND r)
=> (p AND q) AND r
(p AND q) AND r
=> p AND (q AND r)
p OR (q OR r)
=> (p OR q) OR r
(p OR q) OR r
=> p OR (q OR r)
Reorders parens to the left.
p AND (q AND r)
=> (p AND q) AND r
p OR (q OR r)
=> (p OR q) OR r
Reorders parens to the right.
(p AND q) AND r
=> p AND (q AND r)
(p OR q) OR r
=> p OR (q OR r)
Reorders queries.
p AND q
=> q AND p
p OR q
=> q OR p
Distributes one query over its dual.
p AND (q OR r)
=> (p OR q) AND (p OR r)
p OR (q AND r)
=> (p AND q) OR (p AND r)
Performs the opposite of distribution.
(p OR q) AND (p OR r)
=> p AND (q OR r)
(p AND q) OR (p AND r)
=> p OR (q AND r)
Simplifies the Inquire with the Idempotency rule.
p AND p
=> p
p OR p
=> p
Creating new Inquires is done with either single key
val
arguments,
an array of [key, val]
pairs,
or an array of {key: k, val: v}
objects.
The predicate combinators above correspond to the single key
, val
option.
This constructs a single Inquire from the conjunction of all the Inquires in the array.
This constructs an Inquire with the default relation of ==
for all predicates,
and the default junction of &
for all combinations of predicates.
This constructs an Inquire with the default relation of ==
for all predicates,
and the default junction of &
for all combinations of predicates.
This expects each object to have two entries: key
and val
.
These should correspond to the intended predicate key
and val
.
This constructs an Inquire with the default relation of ==
for all predicates,
and the default junction of &
for all combinations of predicates.
Since Inquire is also a Foldable
and Functor
,
it provides the ability to fold
and map
over all of val
s, respectively.
These operations are similar to the native reduce
and map
for javascript arrays,
the fantasy land(fantasy land) specification,
or something you might get from a library like underscore.
So, if you can use Array.map
, _.reduceRight
or similar,
you already know how to use these functions.
This is a right associative "catamorphism". What that means is that it takes each value and joins it to the next given the supplied function.
The first argument is a binary function. The second argument is the initial value. The last argument is the Inquire.
This is a left associative "catamorphism". What that means is that it takes each value and joins it to the next given the supplied function.
The first argument is a binary function. The second argument is the initial value. The last argument is the Inquire.
This will modify every value in the Inquire with the given function.
There are a few modification combinators available.
Removes all predicates that fail the supplied boolean test.
The first argument must be a function that tests a key
and returns a boolean.
Removes all predicates that fail the supplied boolean test.
The first argument must be a function that tests a val
and returns a boolean.
Searches for the first element which has the specified key
.
If successful, will return a Just predicate
.
If unsuccessful, will return a Nothing
.
This is biased to the right.
Searches for the first element which has the specified val
.
If successful, will return a Just predicate
.
If unsuccessful, will return a Nothing
.
This is biased to the right.
Helper function for remove
and removeAll
.
Requires a binary function that takes two inquires and returns a boolean indicating whether the to continue removal.
Removes the rightmost instance of the first Inquire from the second Inquire.
Removes every instance of the first Inquire from the second Inquire.
Replaces each Inquire that has the specified key
with the given val
.
The first argument is the val
to replace with.
The second argument is the key
to test with.
Replaces each Inquire that has the specified val
with the given val
.
The first argument is the val
to replace with.
The second argument is the val
to test with.
Unsafe version of findByKey.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
Unsafe version of findByVal.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
Unsafe way to drop an object into an Inquire.
This just iterates through the object and tries to take all the top level keys and vals as arguments to the Inquire.
Conjoins everything with the relation equals.
Note, there's no telling what will happen if you put in something that's not a simple key val object.
Unsafe version of remove.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
Unsafe version of removeAll.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
Unsafe version of replaceValByKey.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
Unsafe version of replaceValByVal.
This is unsafe in the sense that it uses Prelude.unsafeRefEq
for equality testing.
AKA ===
in javascript.
So it could fail depending on what the values are. Use with caution!
However, it provides a nice wrapper around the safe version. i.e. you don't have to pass a two typeclass dictionary references to the safe version.
This is unsafe in many senses. Primarily it relies on the current implementation of the compiler to check if any js value is an Inquire. Use with caution as it can generate false positives.
The Zipper is a purely functional data structure for manipulating other data structures.
An immense amount of information can be found on Zippers.
It was first written about as a Functional Pearl by Gérard Huet.
The basic idea is that you create a hole
in a data structure that you want to modify,
and keep track of the context
to that hole
.
Once you modify the hole
, you can then use the context
to find your way back to the top of the structure.
In our case, the hole
will be an Inquire, and the context
the path to the top of the Inquire.
We can move in the four cardinal directions: Up, Down, Left, Right.
This allows us to go to any position in our Inquire,
modify that Predicate, Junction, or Wrap, and return the new structure easily.
Creates a new Zipper from an Inquire. This is how we can start to manipulate our Inquire.
Returns our modified Inquire from a Zipper.
Moves to the left of the current hole
, if possible, or does nothing.
Moves to the right of the current hole
, if possible, or does nothing.
Moves down or to the right of the current hole
, if possible, or does nothing.
Moves up of the current hole
, if possible, or does nothing.
Moves to the top of the Zipper.
Moves as far to the left of the Zipper as possible.
Moves as far to the right of the Zipper as possible.
Peeks into the Zipper and returns the current Inquire in the hole
.
Applies a function to the current Inquire in the hole
and returns the result.
Applies a function to the current Inquire in the hole
.
So let's say we've got a Todo list available at some endpoint. We'd like to retrieve all of the items in the list that are due after tax day 2014. We'll use moment to make things a bit easier.
> inquire = require('inquire')
> I = inquire.Inquire
> moment = require('moment')
> query0 = I.gt('due')(moment('Apr 15, 2014'))
> I.generate(query0)
'due>Tue%20Apr%2015%202014%2000%3A00%3A00%20GMT-0700'
Later on if you decide to change the date value, there are combinators that make it much easier than doing this by hand:
> IC = inquire.Inquire_Combinators
> query1 = IC.unsafeReplaceValByKey('tonight')('due')(query0)
> I.generate(query1)
'due>tonight'
Maybe instead you want to move the cut-off point a month sooner, so March 15, 2014. This can be done using the fact that Inquire is a Functor.
> prevMonth = function(x) { return moment(x).subtract('month', 1); }
> query2 = IC.map(prevMonth)(query0)
> I.generate(query2)
'due>Sat%20Mar%2015%202014%2000%3A00%3A00%20GMT-0700'
Of course, this will fail miserably if you have more values that aren't moments. The more robust way currently is to use the Zipper. Let's start by making our query a bit bigger and seeing it fail.
> query3 = I.and(query0)(IC.fromArrayPair([['user', 'Joe'], ['subject', 'bills']]))
> I.generate(query3)
'due>Tue%20Apr%2015%202014%2000%3A00%3A00%20GMT-0700&subject=bills&user=Joe'
> query4 = IC.map(prevMonth)(query3)
> I.generate(query4)
'due>Sat%20Mar%2015%202014%2000%3A00%3A00%20GMT-0700&subject=Invalid%20date&user=Invalid%20date'
So as we can see, it failed because the function was expecting to operate on moments.
N.B. This wouldn't be a problem in a language like purescript, as the line wouldn't have even compile. This is only a concern in javascript because it does not respect types.
So let's use our zipper to find that specific date, and change it. We know it's the leftmost predicate in our query, so let's go there, and modify that.
The basic idea with the Zipper is that it allows you to traverse a data structure, modify some part of it, and then return the new modified structure.
> IZ = inquire.Inquire_Zipper
> zip0 = IZ.toInquireZ(query3)
> zip1 = IZ.zipLeftmost(zip0)
> zip2 = IZ.modify(IC.map(prevMonth))(zip1)
> query5 = IZ.fromInquireZ(zip2)
> I.generate(query5)
'due>Sat%20Mar%2015%202014%2000%3A00%3A00%20GMT-0700&subject=bills&user=Joe'
Wonderful! We see that the Zipper made it so our logic was expressed almost verbatim in the code. It was composable and reusable.