Skip to content

Commit

Permalink
Minimal range query support
Browse files Browse the repository at this point in the history
  • Loading branch information
danslapman committed Sep 21, 2022
1 parent f3d2752 commit 2c6f244
Show file tree
Hide file tree
Showing 4 changed files with 65 additions and 3 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,13 @@ object ElasticQueryCompiler extends Backend[QExpr, ElasticQueryNode, JsonNode] {
case QExpr.Eq(QExpr.Prop(path), QExpr.Constant(s)) => EQN.Term(EQN.Field(path), EQN.Constant(s))
case QExpr.Ne(QExpr.Prop(path), QExpr.Constant(s)) =>
EQN.Bool(mustNot = EQN.Term(EQN.Field(path), EQN.Constant(s)) :: Nil)
case QExpr.And(exprs) => EQN.Bool(must = exprs map opt)
case QExpr.Or(exprs) => EQN.Bool(should = exprs map opt)
case QExpr.Not(expr) => EQN.Bool(mustNot = opt(expr) :: Nil)
case QExpr.Gte(QExpr.Prop(path), QExpr.Constant(s)) => EQN.Range(EQN.Field(path), gte = Some(EQN.Constant(s)))
case QExpr.Lte(QExpr.Prop(path), QExpr.Constant(s)) => EQN.Range(EQN.Field(path), lte = Some(EQN.Constant(s)))
case QExpr.Gt(QExpr.Prop(path), QExpr.Constant(s)) => EQN.Range(EQN.Field(path), gt = Some(EQN.Constant(s)))
case QExpr.Lt(QExpr.Prop(path), QExpr.Constant(s)) => EQN.Range(EQN.Field(path), lt = Some(EQN.Constant(s)))
case QExpr.And(exprs) => EQN.Bool(must = exprs map opt)
case QExpr.Or(exprs) => EQN.Bool(should = exprs map opt)
case QExpr.Not(expr) => EQN.Bool(mustNot = opt(expr) :: Nil)
case QExpr.Exists(QExpr.Prop(path), QExpr.Constant(true)) => EQN.Exists(EQN.Field(path))
case QExpr.Exists(QExpr.Prop(path), QExpr.Constant(false)) =>
EQN.Bool(mustNot = EQN.Exists(EQN.Field(path)) :: Nil)
Expand Down Expand Up @@ -49,13 +53,24 @@ object ElasticQueryCompiler extends Backend[QExpr, ElasticQueryNode, JsonNode] {
case EQN.Constant(s: Any) => s.toString
case EQN.Exists(EQN.Field(path)) =>
s"""{ "exists": { "field": "${path.mkString(".")}" }}"""
case EQN.Range(EQN.Field(path), gt, gte, lt, lte) =>
val bounds = Seq(
renderKeyMap(""""gt"""", gt),
renderKeyMap(""""gte"""", gte),
renderKeyMap(""""lt"""", lt),
renderKeyMap(""""lte"""", lte)
).flatten
s"""{"range": {"${path.mkString(".")}": {${bounds.mkString(",")}}}}"""
case EQN.Field(field) =>
// TODO: adjust error message
report.errorAndAbort(s"There is no filter condition on field ${field.mkString(".")}")
case _ => "AST can't be rendered"
}
}

private def renderKeyMap(key: String, node: Option[ElasticQueryNode])(using quotes: Quotes): Option[String] =
node.map(render(_)).map(v => s"""$key:$v""")

override def target(optRepr: ElasticQueryNode)(using quotes: Quotes): Expr[JsonNode] = {
import quotes.reflect.*

Expand All @@ -74,6 +89,19 @@ object ElasticQueryCompiler extends Backend[QExpr, ElasticQueryNode, JsonNode] {
'{ JsonNode.obj("term" -> JsonNode.obj(${ Expr(path.mkString(".")) } -> ${ handleValues(x) })) }
case EQN.Exists(EQN.Field(path)) =>
'{ JsonNode.obj("exists" -> JsonNode.obj("field" -> JsonNode.Str(${ Expr(path.mkString(".")) }))) }
case EQN.Range(EQN.Field(path), gt, gte, lt, lte) =>
'{
JsonNode.obj(
"range" -> JsonNode.obj(
${ Expr(path.mkString(".")) } -> JsonNode.obj(
"gt" -> ${ gt.map(handleValues(_)).getOrElse('{ JsonNode.`null` }) },
"gte" -> ${ gte.map(handleValues(_)).getOrElse('{ JsonNode.`null` }) },
"lt" -> ${ lt.map(handleValues(_)).getOrElse('{ JsonNode.`null` }) },
"lte" -> ${ lte.map(handleValues(_)).getOrElse('{ JsonNode.`null` }) }
)
)
)
}
case _ => report.errorAndAbort("given node can't be in that position")
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,4 +38,12 @@ object ElasticQueryNode {
}
}
}

case class Range(
field: Field,
gt: Option[ElasticQueryNode] = None,
gte: Option[ElasticQueryNode] = None,
lt: Option[ElasticQueryNode] = None,
lte: Option[ElasticQueryNode] = None
) extends ElasticQueryNode
}
Original file line number Diff line number Diff line change
Expand Up @@ -47,4 +47,28 @@ class QuerySpec extends AnyFunSuite {

q.render shouldBe """{"query":{"bool":{"must":[],"should":[],"must_not":[{"exists":{"field":"field3.optionalInnerField"}}]}}}"""
}

test("> query") {
val q = query[TestClass](_.field2 > 4)

q.render shouldBe """{"query":{"range":{"field2":{"gt":4,"gte":null,"lt":null,"lte":null}}}}"""
}

test(">= query") {
val q = query[TestClass](_.field2 >= 4)

q.render shouldBe """{"query":{"range":{"field2":{"gt":null,"gte":4,"lt":null,"lte":null}}}}"""
}

test("< query") {
val q = query[TestClass](_.field2 < 4)

q.render shouldBe """{"query":{"range":{"field2":{"gt":null,"gte":null,"lt":4,"lte":null}}}}"""
}

test("<= query") {
val q = query[TestClass](_.field2 <= 4)

q.render shouldBe """{"query":{"range":{"field2":{"gt":null,"gte":null,"lt":null,"lte":4}}}}"""
}
}
2 changes: 2 additions & 0 deletions oolong-json/src/main/scala/ru/tinkoff/oolong/JsonNode.scala
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ private[oolong] object JsonNode {
override def render: String = value.map((k, v) => s"\"$k\":${v.render}").mkString("{", ",", "}")
}

val `null`: JsonNode = Null

def obj(head: (String, JsonNode), tail: (String, JsonNode)*): Obj =
Obj((head +: tail).to(Map))
}

0 comments on commit 2c6f244

Please sign in to comment.