-
Notifications
You must be signed in to change notification settings - Fork 53
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
refactor: GQL responses #2872
refactor: GQL responses #2872
Changes from 9 commits
34173ea
e3ed55a
9184773
524fd54
4d847ac
8758c44
fcc576f
b3469c6
2ccc762
bca807b
819090e
ead2482
681db99
585cc16
0935e4c
779959d
340d097
ca7bf81
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -41,7 +41,51 @@ const ( | |
CommitSelection | ||
) | ||
|
||
// ToSelect converts the given [parser.Select] into a [Select]. | ||
// ToOperation converts the given [request.OperationDefinition] into an [Operation]. | ||
// | ||
// In the process of doing so it will construct the document map required to access the data | ||
// yielded by the [Operation]. | ||
func ToOperation( | ||
ctx context.Context, | ||
store client.Store, | ||
operationRequest *request.OperationDefinition, | ||
) (*Operation, error) { | ||
operation := &Operation{ | ||
DocumentMapping: core.NewDocumentMapping(), | ||
} | ||
|
||
for i, s := range operationRequest.Selections { | ||
switch t := s.(type) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. suggestion: The default case is not handled, it might be better to return an error instead of quietly ignoring it - I believe it could only happen if we mess up in the future, but it could save some head-scratching even if it means a line or two of dead code. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I've added a default case with an error. |
||
case *request.CommitSelect: | ||
s, err := toCommitSelect(ctx, store, t, i) | ||
if err != nil { | ||
return nil, err | ||
} | ||
operation.CommitSelects = append(operation.CommitSelects, s) | ||
operation.addSelection(i, t.Field, s.Select) | ||
|
||
case *request.Select: | ||
s, err := toSelect(ctx, store, ObjectSelection, i, t, "") | ||
if err != nil { | ||
return nil, err | ||
} | ||
operation.Selects = append(operation.Selects, s) | ||
operation.addSelection(i, t.Field, *s) | ||
|
||
case *request.ObjectMutation: | ||
m, err := toMutation(ctx, store, t, i) | ||
if err != nil { | ||
return nil, err | ||
} | ||
operation.Mutations = append(operation.Mutations, m) | ||
operation.addSelection(i, t.Field, m.Select) | ||
} | ||
} | ||
|
||
return operation, nil | ||
} | ||
|
||
// ToSelect converts the given [request.Select] into a [Select]. | ||
// | ||
// In the process of doing so it will construct the document map required to access the data | ||
// yielded by the [Select]. | ||
|
@@ -1142,7 +1186,20 @@ func ToCommitSelect( | |
store client.Store, | ||
selectRequest *request.CommitSelect, | ||
) (*CommitSelect, error) { | ||
underlyingSelect, err := ToSelect(ctx, store, CommitSelection, selectRequest.ToSelect()) | ||
return toCommitSelect(ctx, store, selectRequest, 0) | ||
} | ||
|
||
// toCommitSelect converts the given [request.CommitSelect] into a [CommitSelect]. | ||
// | ||
// In the process of doing so it will construct the document map required to access the data | ||
// yielded by the [Select] embedded in the [CommitSelect]. | ||
func toCommitSelect( | ||
ctx context.Context, | ||
store client.Store, | ||
selectRequest *request.CommitSelect, | ||
thisIndex int, | ||
) (*CommitSelect, error) { | ||
underlyingSelect, err := toSelect(ctx, store, CommitSelection, thisIndex, selectRequest.ToSelect(), "") | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
@@ -1160,11 +1217,23 @@ func ToCommitSelect( | |
// In the process of doing so it will construct the document map required to access the data | ||
// yielded by the [Select] embedded in the [Mutation]. | ||
func ToMutation(ctx context.Context, store client.Store, mutationRequest *request.ObjectMutation) (*Mutation, error) { | ||
underlyingSelect, err := ToSelect(ctx, store, ObjectSelection, mutationRequest.ToSelect()) | ||
return toMutation(ctx, store, mutationRequest, 0) | ||
} | ||
|
||
// toMutation converts the given [request.Mutation] into a [Mutation]. | ||
// | ||
// In the process of doing so it will construct the document map required to access the data | ||
// yielded by the [Select] embedded in the [Mutation]. | ||
func toMutation( | ||
ctx context.Context, | ||
store client.Store, | ||
mutationRequest *request.ObjectMutation, | ||
thisIndex int, | ||
) (*Mutation, error) { | ||
underlyingSelect, err := toSelect(ctx, store, ObjectSelection, thisIndex, mutationRequest.ToSelect(), "") | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return &Mutation{ | ||
Select: *underlyingSelect, | ||
Type: MutationType(mutationRequest.Type), | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
// Copyright 2024 Democratized Data Foundation | ||
// | ||
// Use of this software is governed by the Business Source License | ||
// included in the file licenses/BSL.txt. | ||
// | ||
// As of the Change Date specified in that file, in accordance with | ||
// the Business Source License, use of this software will be governed | ||
// by the Apache License, Version 2.0, included in the file | ||
// licenses/APL.txt. | ||
|
||
package mapper | ||
|
||
import ( | ||
"github.com/sourcenetwork/defradb/client/request" | ||
"github.com/sourcenetwork/defradb/internal/core" | ||
) | ||
|
||
// Operation represents an operation such as query or mutation. | ||
// | ||
// It wraps child Selects belonging to this operation. | ||
type Operation struct { | ||
// The document mapping for this select, describing how items yielded | ||
// for this select can be accessed and rendered. | ||
*core.DocumentMapping | ||
|
||
// Selects is the list of selections in the operation. | ||
Selects []*Select | ||
|
||
// Mutations is the list of mutations in the operation. | ||
Mutations []*Mutation | ||
|
||
// CommitSelects is the list of commit selections in the operation. | ||
CommitSelects []*CommitSelect | ||
} | ||
|
||
// addSelection adds a new selection to the operation's document mapping. | ||
// The request.Field is used as the key for the core.RenderKey and the Select | ||
// document mapping is added as a child field. | ||
func (o *Operation) addSelection(i int, f request.Field, s Select) { | ||
renderKey := core.RenderKey{Index: s.Index} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. todo: There is a function There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done |
||
if f.Alias.HasValue() { | ||
renderKey.Key = f.Alias.Value() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. todo: Please add an operation alias test There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Added |
||
} else { | ||
renderKey.Key = f.Name | ||
} | ||
o.DocumentMapping.Add(i, s.Name) | ||
o.DocumentMapping.SetChildAt(i, s.DocumentMapping) | ||
o.DocumentMapping.RenderKeys = append(o.DocumentMapping.RenderKeys, renderKey) | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,185 @@ | ||
// Copyright 2024 Democratized Data Foundation | ||
// | ||
// Use of this software is governed by the Business Source License | ||
// included in the file licenses/BSL.txt. | ||
// | ||
// As of the Change Date specified in that file, in accordance with | ||
// the Business Source License, use of this software will be governed | ||
// by the Apache License, Version 2.0, included in the file | ||
// licenses/APL.txt. | ||
|
||
package planner | ||
|
||
import ( | ||
"github.com/sourcenetwork/defradb/client/request" | ||
"github.com/sourcenetwork/defradb/internal/core" | ||
"github.com/sourcenetwork/defradb/internal/planner/mapper" | ||
) | ||
|
||
const operationNodeKind string = "operationNode" | ||
|
||
// operationNode is the top level node for operations with | ||
// one or more child selections, such as queries or mutations. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nitpick: I understand what you mean by There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree the name could be improved. It should be fairly easy to rename if we find a better name. |
||
type operationNode struct { | ||
documentIterator | ||
docMapper | ||
|
||
children []planNode | ||
childIndexes []int | ||
isDone bool | ||
} | ||
|
||
func (n *operationNode) Spans(spans core.Spans) { | ||
for _, child := range n.children { | ||
child.Spans(spans) | ||
} | ||
} | ||
|
||
func (n *operationNode) Kind() string { | ||
return operationNodeKind | ||
} | ||
|
||
func (n *operationNode) Init() error { | ||
n.isDone = false | ||
for _, child := range n.children { | ||
err := child.Init() | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
func (n *operationNode) Start() error { | ||
for _, child := range n.children { | ||
err := child.Start() | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
func (n *operationNode) Close() error { | ||
for _, child := range n.children { | ||
err := child.Close() | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
func (n *operationNode) Source() planNode { | ||
return nil | ||
} | ||
|
||
func (p *operationNode) Children() []planNode { | ||
return p.children | ||
} | ||
|
||
func (n *operationNode) Explain(explainType request.ExplainType) (map[string]any, error) { | ||
switch explainType { | ||
case request.SimpleExplain: | ||
return map[string]any{}, nil | ||
|
||
case request.ExecuteExplain: | ||
return map[string]any{}, nil | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. question: is it supposed to be empty? Did you add explain tests that assert this? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm not super familiar with the planner package. Is this supposed to return the operations that this node executes? It does appear in the existing explain tests correctly. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I believe you can omit the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. As @jsimnz pointed out, making a node explainable is optional. Moreover, An example of this is |
||
|
||
default: | ||
return nil, ErrUnknownExplainRequestType | ||
} | ||
} | ||
|
||
func (n *operationNode) Next() (bool, error) { | ||
if n.isDone { | ||
return false, nil | ||
} | ||
|
||
n.currentValue = n.documentMapping.NewDoc() | ||
for i, child := range n.children { | ||
switch child.(type) { | ||
case *topLevelNode: | ||
hasChild, err := child.Next() | ||
if err != nil { | ||
return false, err | ||
} | ||
if !hasChild { | ||
return false, ErrMissingChildValue | ||
} | ||
n.currentValue.Fields[n.childIndexes[i]] = child.Value().Fields[0] | ||
|
||
default: | ||
var docs []core.Doc | ||
for { | ||
hasChild, err := child.Next() | ||
if err != nil { | ||
return false, err | ||
} | ||
if !hasChild { | ||
break | ||
} | ||
docs = append(docs, child.Value()) | ||
} | ||
n.currentValue.Fields[n.childIndexes[i]] = docs | ||
} | ||
} | ||
|
||
n.isDone = true | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. question: do you really need this flag? It's being set only in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I tried removing it, but the return value was always incorrect. I believe we need it so that the operation is able to run at least once. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nitpick/todo: The planNode interface and it's consuption dictates that There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should be fixed now. I'm assuming you were referencing the |
||
return true, nil | ||
} | ||
|
||
// Operation creates a new operationNode using the given Selects. | ||
func (p *Planner) Operation(operation *mapper.Operation) (*operationNode, error) { | ||
var children []planNode | ||
var childIndexes []int | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. suggestion: I think a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done |
||
|
||
for _, s := range operation.Selects { | ||
if _, isAgg := request.Aggregates[s.Name]; isAgg { | ||
// If this Select is an aggregate, then it must be a top-level | ||
// aggregate and we need to resolve it within the context of a | ||
// top-level node. | ||
child, err := p.Top(s) | ||
if err != nil { | ||
return nil, err | ||
} | ||
children = append(children, child) | ||
childIndexes = append(childIndexes, s.Index) | ||
} else { | ||
child, err := p.Select(s) | ||
if err != nil { | ||
return nil, err | ||
} | ||
children = append(children, child) | ||
childIndexes = append(childIndexes, s.Index) | ||
} | ||
} | ||
|
||
for _, m := range operation.Mutations { | ||
child, err := p.newObjectMutationPlan(m) | ||
if err != nil { | ||
return nil, err | ||
} | ||
children = append(children, child) | ||
childIndexes = append(childIndexes, m.Index) | ||
} | ||
|
||
for _, s := range operation.CommitSelects { | ||
child, err := p.CommitSelect(s) | ||
if err != nil { | ||
return nil, err | ||
} | ||
children = append(children, child) | ||
childIndexes = append(childIndexes, s.Index) | ||
} | ||
|
||
if len(children) == 0 { | ||
return nil, ErrOperationDefinitionMissingSelection | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. question: Is this spec-compliant? A selection set must always be provided? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I didn't see anything in the spec about empty selections. I've removed this error since it should be fine to return empty results. |
||
} | ||
|
||
return &operationNode{ | ||
docMapper: docMapper{operation.DocumentMapping}, | ||
children: children, | ||
childIndexes: childIndexes, | ||
}, nil | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
question/suggestion: Why is results still an array? Can we return just a
map[string]any
now instead?Same question/suggestion for
RunSelection
.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good catch! I've updated both to return maps.