alehechka/grpc-graphql-gateway

Refactor Proto type marshaling

alehechka opened this issue · 2 comments

Due to the changes made on #14, the default behavior of the generated GraphQL resolvers is to use the custom MarshalResponse function which makes use of a series of recursive calls to marshal the provided interface into a more generic structure with json_name appropriate keys.

However, It is not ideal to use this because it is one extra thing this package has to deal with and since it is now the default behavior after generation, it leads to a much greater dependency on this custom marshaling.

A more ideal solution would be to of course not introduce an intermediate marshaling step at all, but due to the protobuf compiler not generating the json tag with appropriate json_name from the .proto spec file, it leads to naming conventions of fields not matching and thus not being returned correctly from GraphQL.

The following code change shows a possible solution that simply uses the jsonpb package (that is also being used for MarshalRequest.

Console logging both the above code changes as well as the output of the current custom marshaling shows that they are near identical. The only difference being that in the starwars.proto example the returned id field in the jsonpb functions returns as a string instead of int64. However, this has shown to be an oddity with int64 in general and after it goes through the graphql resolver it is outputted as an integer regardless.

However, after running through the graphql resolver, all enum values are null. The output for these enums is the same after either custom marshaling steps, but after the graphql resolver, the jsonpb version only returns nulls. If this can be overcome, this would be the much preferred solution to this problem as it would result in much less custom code and rely on a more battle hardened solution.

Figured this problem out. After re-reading the docs for the standard encoding/json packages Marshal and Unmarshal libraries, it reminded me that JSON numbers get unmarshaled as float64. The current template.go will generate the enum values as int32(0) which causes the graphql resolver to not equate the values for deciding which enum string to return.

Changing the generated enum values to instead cast as float64(0) does resolve the issue of enums returning from the graphql-gateway as null.

However, after further thought I'm releasing that this shorter, simpler solution may actually be less performant. The reasoning is due to first marshaling the proto message into a bytes.Buffer as encoded JSON. Then decoding the JSON buffer into a Go interface{} that will be returned from the function. This requires both of those steps before it hits the graphql resolver which undoubtedly also does some kind of encoding.

The current custom solution in place, while more complex, and and extra thing to manage for this code base, does lend itself to not having this marshal/unmarshal step. It simply takes a raw interface{} and uses reflection to transform the entire provided object into a new one that uses the protobuf tag to extract the key for each variable in the object.

Before I close this issue, I am going to give this some more thought and potentially try and test some larger datasets to check performance.

Additionally, the current jsonpb marshaling solution takes protoiface.MessageV1 as its only parameter which is fine for most cases, but if the pluck option is used within a service it will pluck a child object from the parent and in the base example for starwars, the humans and droids queries end up returning a slice of protoiface.MessageV1 objects which causes a mismatch type compile error. This would undoubtedly need to be coded for and end up adding more complexity to the jsonpb solution.

Sadly another con against the jsonpb marshaling solution.