-
Notifications
You must be signed in to change notification settings - Fork 843
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #495 from bhoriuchi/subscription-execution
Subscription execution
- Loading branch information
Showing
5 changed files
with
669 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,3 @@ | ||
module github.com/graphql-go/graphql | ||
|
||
go 1.13 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,228 @@ | ||
package graphql | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
|
||
"github.com/graphql-go/graphql/gqlerrors" | ||
"github.com/graphql-go/graphql/language/parser" | ||
"github.com/graphql-go/graphql/language/source" | ||
) | ||
|
||
// SubscribeParams parameters for subscribing | ||
type SubscribeParams struct { | ||
Schema Schema | ||
RequestString string | ||
RootValue interface{} | ||
// ContextValue context.Context | ||
VariableValues map[string]interface{} | ||
OperationName string | ||
FieldResolver FieldResolveFn | ||
FieldSubscriber FieldResolveFn | ||
} | ||
|
||
// Subscribe performs a subscribe operation on the given query and schema | ||
// To finish a subscription you can simply close the channel from inside the `Subscribe` function | ||
// currently does not support extensions hooks | ||
func Subscribe(p Params) chan *Result { | ||
|
||
source := source.NewSource(&source.Source{ | ||
Body: []byte(p.RequestString), | ||
Name: "GraphQL request", | ||
}) | ||
|
||
// TODO run extensions hooks | ||
|
||
// parse the source | ||
AST, err := parser.Parse(parser.ParseParams{Source: source}) | ||
if err != nil { | ||
|
||
// merge the errors from extensions and the original error from parser | ||
return sendOneResultAndClose(&Result{ | ||
Errors: gqlerrors.FormatErrors(err), | ||
}) | ||
} | ||
|
||
// validate document | ||
validationResult := ValidateDocument(&p.Schema, AST, nil) | ||
|
||
if !validationResult.IsValid { | ||
// run validation finish functions for extensions | ||
return sendOneResultAndClose(&Result{ | ||
Errors: validationResult.Errors, | ||
}) | ||
|
||
} | ||
return ExecuteSubscription(ExecuteParams{ | ||
Schema: p.Schema, | ||
Root: p.RootObject, | ||
AST: AST, | ||
OperationName: p.OperationName, | ||
Args: p.VariableValues, | ||
Context: p.Context, | ||
}) | ||
} | ||
|
||
func sendOneResultAndClose(res *Result) chan *Result { | ||
resultChannel := make(chan *Result, 1) | ||
resultChannel <- res | ||
close(resultChannel) | ||
return resultChannel | ||
} | ||
|
||
// ExecuteSubscription is similar to graphql.Execute but returns a channel instead of a Result | ||
// currently does not support extensions | ||
func ExecuteSubscription(p ExecuteParams) chan *Result { | ||
|
||
if p.Context == nil { | ||
p.Context = context.Background() | ||
} | ||
|
||
var mapSourceToResponse = func(payload interface{}) *Result { | ||
return Execute(ExecuteParams{ | ||
Schema: p.Schema, | ||
Root: payload, | ||
AST: p.AST, | ||
OperationName: p.OperationName, | ||
Args: p.Args, | ||
Context: p.Context, | ||
}) | ||
} | ||
var resultChannel = make(chan *Result) | ||
go func() { | ||
defer close(resultChannel) | ||
defer func() { | ||
if err := recover(); err != nil { | ||
e, ok := err.(error) | ||
if !ok { | ||
return | ||
} | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(e), | ||
} | ||
} | ||
return | ||
}() | ||
|
||
exeContext, err := buildExecutionContext(buildExecutionCtxParams{ | ||
Schema: p.Schema, | ||
Root: p.Root, | ||
AST: p.AST, | ||
OperationName: p.OperationName, | ||
Args: p.Args, | ||
Context: p.Context, | ||
}) | ||
|
||
if err != nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(err), | ||
} | ||
|
||
return | ||
} | ||
|
||
operationType, err := getOperationRootType(p.Schema, exeContext.Operation) | ||
if err != nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(err), | ||
} | ||
|
||
return | ||
} | ||
|
||
fields := collectFields(collectFieldsParams{ | ||
ExeContext: exeContext, | ||
RuntimeType: operationType, | ||
SelectionSet: exeContext.Operation.GetSelectionSet(), | ||
}) | ||
|
||
responseNames := []string{} | ||
for name := range fields { | ||
responseNames = append(responseNames, name) | ||
} | ||
responseName := responseNames[0] | ||
fieldNodes := fields[responseName] | ||
fieldNode := fieldNodes[0] | ||
fieldName := fieldNode.Name.Value | ||
fieldDef := getFieldDef(p.Schema, operationType, fieldName) | ||
|
||
if fieldDef == nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(fmt.Errorf("the subscription field %q is not defined", fieldName)), | ||
} | ||
|
||
return | ||
} | ||
|
||
resolveFn := fieldDef.Subscribe | ||
|
||
if resolveFn == nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(fmt.Errorf("the subscription function %q is not defined", fieldName)), | ||
} | ||
return | ||
} | ||
fieldPath := &ResponsePath{ | ||
Key: responseName, | ||
} | ||
|
||
args := getArgumentValues(fieldDef.Args, fieldNode.Arguments, exeContext.VariableValues) | ||
info := ResolveInfo{ | ||
FieldName: fieldName, | ||
FieldASTs: fieldNodes, | ||
Path: fieldPath, | ||
ReturnType: fieldDef.Type, | ||
ParentType: operationType, | ||
Schema: p.Schema, | ||
Fragments: exeContext.Fragments, | ||
RootValue: exeContext.Root, | ||
Operation: exeContext.Operation, | ||
VariableValues: exeContext.VariableValues, | ||
} | ||
|
||
fieldResult, err := resolveFn(ResolveParams{ | ||
Source: p.Root, | ||
Args: args, | ||
Info: info, | ||
Context: p.Context, | ||
}) | ||
if err != nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(err), | ||
} | ||
|
||
return | ||
} | ||
|
||
if fieldResult == nil { | ||
resultChannel <- &Result{ | ||
Errors: gqlerrors.FormatErrors(fmt.Errorf("no field result")), | ||
} | ||
|
||
return | ||
} | ||
|
||
switch fieldResult.(type) { | ||
case chan interface{}: | ||
sub := fieldResult.(chan interface{}) | ||
for { | ||
select { | ||
case <-p.Context.Done(): | ||
return | ||
|
||
case res, more := <-sub: | ||
if !more { | ||
return | ||
} | ||
resultChannel <- mapSourceToResponse(res) | ||
} | ||
} | ||
default: | ||
resultChannel <- mapSourceToResponse(fieldResult) | ||
return | ||
} | ||
}() | ||
|
||
// return a result channel | ||
return resultChannel | ||
} |
Oops, something went wrong.