Skip to content

Commit

Permalink
Fix(GraphQL): Add filter in DQL query in case of reverse predicate (#…
Browse files Browse the repository at this point in the history
…7728)

* Add code

* Fix GraphQL schemagen with reverse Dgraph predicates (#7689)

(cherry picked from commit f7c199e)

* Add yaml test

* Fix tests

(cherry picked from commit 1743501)
  • Loading branch information
vmrajas committed Apr 16, 2021
1 parent f7c199e commit b1c6ef3
Show file tree
Hide file tree
Showing 4 changed files with 107 additions and 5 deletions.
30 changes: 29 additions & 1 deletion graphql/resolve/query_rewriter.go
Original file line number Diff line number Diff line change
Expand Up @@ -243,6 +243,7 @@ func aggregateQuery(query schema.Query, authRw *authRewriter) []*gql.GraphQuery
// var(func: type(Tweets)) {
// scoreVar as Tweets.score
// }

mainQuery.Children = append(mainQuery.Children, child)
isAggregateFieldVisited[constructedForField] = true
}
Expand Down Expand Up @@ -1042,6 +1043,13 @@ func buildAggregateFields(

// addedAggregateFields is a map from aggregate field name to boolean
addedAggregateField := make(map[string]bool)
// isAggregateFieldVisited is a map from field name to boolean. It is used to
// Add type filter in case the Dgraph predicate for which the aggregate
// field belongs to is a reverse edge
if strings.HasPrefix(constructedForDgraphPredicate, "~") {
addTypeFilter(mainField, f.ConstructedFor())
}

// isAggregateFieldVisited is a map from field name to boolean. It is used to
// ensure that a field is added to Var query at maximum once.
// Eg. Even if scoreMax and scoreMin are queried, the corresponding field will
Expand Down Expand Up @@ -1069,7 +1077,21 @@ func buildAggregateFields(
// As count fields are always part of an AggregateField by
// default (added just before this for loop). We continue
// in case of a count field.
if aggregateField.DgraphAlias() == "count" {
if aggregateField.Name() == "count" {
aggregateChild := &gql.GraphQuery{
Alias: aggregateField.DgraphAlias() + "_" + f.DgraphAlias(),
Attr: "count(" + constructedForDgraphPredicate + ")",
}
// Add filter to count aggregation field.
_ = addFilter(aggregateChild, constructedForType, fieldFilter)

// Add type filter in case the Dgraph predicate for which the aggregate
// field belongs to is a reverse edge
if strings.HasPrefix(constructedForDgraphPredicate, "~") {
addTypeFilter(aggregateChild, f.ConstructedFor())
}

aggregateChildren = append(aggregateChildren, aggregateChild)
continue
}
// Handle other aggregate functions than count
Expand Down Expand Up @@ -1268,6 +1290,12 @@ func addSelectionSetFrom(
if includeField := addFilter(child, f.Type(), filter); !includeField {
continue
}

// Add type filter in case the Dgraph predicate is a reverse edge
if strings.HasPrefix(f.DgraphPredicate(), "~") {
addTypeFilter(child, f.Type())
}

addOrder(child, f)
addPagination(child, f)
addCascadeDirective(child, f)
Expand Down
53 changes: 50 additions & 3 deletions graphql/resolve/query_test.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -2395,9 +2395,8 @@
query {
queryMovie(func: type(Movie)) {
name : Movie.name
director : ~directed.movies {
director : ~directed.movies @filter(type(MovieDirector)) {
name : MovieDirector.name
dgraph.uid : uid
}
dgraph.uid : uid
}
Expand Down Expand Up @@ -3183,4 +3182,52 @@
name : Author.name
dgraph.uid : uid
}
}
}
-
name: "Query fields linked to reverse predicates in Dgraph"
gqlquery: |
query {
queryLinkX(filter:{f9:{eq: "Alice"}}) {
f1(filter: {f6: {eq: "Eve"}}) {
f6
}
f2(filter: {f7: {eq: "Bob"}}) {
f7
}
f1Aggregate(filter: {f6: {eq: "Eve"}}) {
count
f6Max
}
f2Aggregate(filter: {f7: {eq: "Bob"}}) {
count
f7Min
}
}
}
dgquery: |-
query {
queryLinkX(func: type(LinkX)) @filter(eq(LinkX.f9, "Alice")) {
LinkX.f1 : ~link @filter((eq(LinkY.f6, "Eve") AND type(LinkY))) {
LinkY.f6 : LinkY.f6
dgraph.uid : uid
}
LinkX.f2 : ~link @filter((eq(LinkZ.f7, "Bob") AND type(LinkZ))) {
LinkZ.f7 : LinkZ.f7
dgraph.uid : uid
}
LinkX.f1Aggregate : ~link @filter((eq(LinkY.f6, "Eve") AND type(LinkY))) {
LinkX.f1Aggregate_f6Var as LinkY.f6
dgraph.uid : uid
}
LinkYAggregateResult.count_LinkX.f1Aggregate : count(~link) @filter((eq(LinkY.f6, "Eve") AND type(LinkY)))
LinkYAggregateResult.f6Max_LinkX.f1Aggregate : max(val(LinkX.f1Aggregate_f6Var))
LinkX.f2Aggregate : ~link @filter((eq(LinkZ.f7, "Bob") AND type(LinkZ))) {
LinkX.f2Aggregate_f7Var as LinkZ.f7
dgraph.uid : uid
}
LinkZAggregateResult.count_LinkX.f2Aggregate : count(~link) @filter((eq(LinkZ.f7, "Bob") AND type(LinkZ)))
LinkZAggregateResult.f7Min_LinkX.f2Aggregate : min(val(LinkX.f2Aggregate_f7Var))
dgraph.uid : uid
}
}
16 changes: 15 additions & 1 deletion graphql/resolve/schema.graphql
Original file line number Diff line number Diff line change
Expand Up @@ -378,4 +378,18 @@ type Workflow {

type Node {
name: String!
}
}

type LinkX {
f9: String! @id
f1: [LinkY] @dgraph(pred: "~link")
f2: [LinkZ] @dgraph(pred: "~link")
}
type LinkY {
f6: String! @id
f3: [LinkX] @dgraph(pred: "link")
}
type LinkZ {
f7: String! @id
f4: [LinkX] @dgraph(pred: "link")
}
13 changes: 13 additions & 0 deletions graphql/schema/gqlschema_test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -3121,3 +3121,16 @@ valid_schemas:
type Z {
f4: [X] @dgraph(pred: "link")
}
- name: "Same reverse dgraph predicate can be used by two different GraphQL fields"
input: |
type X {
f1: [Y] @dgraph(pred: "~link")
f2: [Z] @dgraph(pred: "~link")
}
type Y {
f3: [X] @dgraph(pred: "link")
}
type Z {
f4: [X] @dgraph(pred: "link")
}

0 comments on commit b1c6ef3

Please sign in to comment.