From b46ef1945d6b7616f6b61ab79b42bb38054e2de0 Mon Sep 17 00:00:00 2001 From: Tom Pallister Date: Thu, 12 Apr 2018 17:48:43 +0100 Subject: [PATCH] Feature/graphql (#312) * #298 initial hacking around better aggregation * #298 bit more hacking around * #298 abstraction over httpresponsemessage * #298 tidying up * #298 docs * #298 missed this * #306 example of how to do GraphQL --- docs/features/graphql.rst | 15 +++ docs/index.rst | 1 + docs/introduction/notsupported.rst | 19 ++- samples/OcelotGraphQL/OcelotGraphQL.csproj | 18 +++ samples/OcelotGraphQL/Program.cs | 131 +++++++++++++++++++++ samples/OcelotGraphQL/README.md | 71 +++++++++++ samples/OcelotGraphQL/configuration.json | 19 +++ 7 files changed, 270 insertions(+), 4 deletions(-) create mode 100644 docs/features/graphql.rst create mode 100644 samples/OcelotGraphQL/OcelotGraphQL.csproj create mode 100644 samples/OcelotGraphQL/Program.cs create mode 100644 samples/OcelotGraphQL/README.md create mode 100644 samples/OcelotGraphQL/configuration.json diff --git a/docs/features/graphql.rst b/docs/features/graphql.rst new file mode 100644 index 00000000..1b527314 --- /dev/null +++ b/docs/features/graphql.rst @@ -0,0 +1,15 @@ +GraphQL +======= + +OK you got me Ocelot doesn't directly support GraphQL but so many people have asked about it I wanted to show how easy it is to integrate +the `graphql-dotnet `_ library. + + +Please see the sample project `OcelotGraphQL `_. +Using a combination of the graphql-dotnet project and Ocelot's DelegatingHandler features this is pretty easy to do. +However I do not intend to integrate more closely with GraphQL at the moment. Check out the samples readme and that should give +you enough instruction on how to do this! + +Good luck and have fun :> + + diff --git a/docs/index.rst b/docs/index.rst index 78b305b2..7989fcdd 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -21,6 +21,7 @@ Thanks for taking a look at the Ocelot documentation. Please use the left hand n features/configuration features/routing features/requestaggregation + features/graphql features/servicediscovery features/servicefabric features/authentication diff --git a/docs/introduction/notsupported.rst b/docs/introduction/notsupported.rst index 35c916a0..37e5b5eb 100644 --- a/docs/introduction/notsupported.rst +++ b/docs/introduction/notsupported.rst @@ -7,7 +7,10 @@ Ocelot does not support... * Fowarding a host header - The host header that you send to Ocelot will not be forwarded to the downstream service. Obviously this would break everything :( -* Swagger - I have looked multiple times at building swagger.json out of the Ocelot configuration.json but it doesnt fit into the vision I have for Ocelot. If you would like to have Swagger in Ocelot then you must roll your own swagger.json and do the following in your Startup.cs or Program.cs. The code sample below registers a piece of middleware that loads your hand rolled swagger.json and returns it on /swagger/v1/swagger.json. It then registers the SwaggerUI middleware from Swashbuckle.AspNetCore +* Swagger - I have looked multiple times at building swagger.json out of the Ocelot configuration.json but it doesnt fit into the vision +I have for Ocelot. If you would like to have Swagger in Ocelot then you must roll your own swagger.json and do the following in your +Startup.cs or Program.cs. The code sample below registers a piece of middleware that loads your hand rolled swagger.json and returns +it on /swagger/v1/swagger.json. It then registers the SwaggerUI middleware from Swashbuckle.AspNetCore .. code-block:: csharp @@ -25,8 +28,16 @@ Ocelot does not support... app.UseOcelot().Wait(); -The main reasons why I don't think Swagger makes sense is we already hand roll our definition in configuration.json. If we want people developing against Ocelot to be able to see what routes are available then either share the configuration.json with them (This should be as easy as granting access to a repo etc) or use the Ocelot administration API so that they can query Ocelot for the configuration. +The main reasons why I don't think Swagger makes sense is we already hand roll our definition in configuration.json. +If we want people developing against Ocelot to be able to see what routes are available then either share the configuration.json +with them (This should be as easy as granting access to a repo etc) or use the Ocelot administration API so that they can query Ocelot for the configuration. -In addition to this many people will configure Ocelot to proxy all traffic like /products/{everything} to there product service and you would not be describing what is actually available if you parsed this and turned it into a Swagger path. Also Ocelot has no concept of the models that the downstream services can return and linking to the above problem the same endpoint can return multiple models. Ocelot does not know what models might be used in POST, PUT etc so it all gets a bit messy and finally the Swashbuckle package doesnt reload swagger.json if it changes during runtime. Ocelot's configuration can change during runtime so the Swagger and Ocelot information would not match. Unless I rolled my own Swagger implementation. +In addition to this many people will configure Ocelot to proxy all traffic like /products/{everything} to there product service +and you would not be describing what is actually available if you parsed this and turned it into a Swagger path. Also Ocelot has +no concept of the models that the downstream services can return and linking to the above problem the same endpoint can return +multiple models. Ocelot does not know what models might be used in POST, PUT etc so it all gets a bit messy and finally the Swashbuckle +package doesnt reload swagger.json if it changes during runtime. Ocelot's configuration can change during runtime so the Swagger and Ocelot +information would not match. Unless I rolled my own Swagger implementation. -If the user wants something to easily test against the Ocelot API then I suggest using Postman as a simple way to do this. It might even be possible to write something that maps configuration.json to the postman json spec. However I don't intend to do this. \ No newline at end of file +If the user wants something to easily test against the Ocelot API then I suggest using Postman as a simple way to do this. It might +even be possible to write something that maps configuration.json to the postman json spec. However I don't intend to do this. \ No newline at end of file diff --git a/samples/OcelotGraphQL/OcelotGraphQL.csproj b/samples/OcelotGraphQL/OcelotGraphQL.csproj new file mode 100644 index 00000000..2f29e395 --- /dev/null +++ b/samples/OcelotGraphQL/OcelotGraphQL.csproj @@ -0,0 +1,18 @@ + + + netcoreapp2.0 + + + + PreserveNewest + + + + + + + + + + + \ No newline at end of file diff --git a/samples/OcelotGraphQL/Program.cs b/samples/OcelotGraphQL/Program.cs new file mode 100644 index 00000000..94e1295b --- /dev/null +++ b/samples/OcelotGraphQL/Program.cs @@ -0,0 +1,131 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Threading.Tasks; +using Microsoft.AspNetCore; +using Microsoft.AspNetCore.Hosting; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Logging; +using Ocelot.Middleware; +using Ocelot.DependencyInjection; +using GraphQL.Types; +using GraphQL; +using Ocelot.Requester; +using Ocelot.Responses; +using System.Net.Http; +using System.Net; +using Microsoft.Extensions.DependencyInjection; +using System.Threading; + +namespace OcelotGraphQL +{ + public class Hero + { + public int Id { get; set; } + public string Name { get; set; } + } + + public class Query + { + private List _heroes = new List + { + new Hero { Id = 1, Name = "R2-D2" }, + new Hero { Id = 2, Name = "Batman" }, + new Hero { Id = 3, Name = "Wonder Woman" }, + new Hero { Id = 4, Name = "Tom Pallister" } + }; + + [GraphQLMetadata("hero")] + public Hero GetHero(int id) + { + return _heroes.FirstOrDefault(x => x.Id == id); + } + } + + public class GraphQlDelegatingHandler : DelegatingHandler + { + private readonly ISchema _schema; + + public GraphQlDelegatingHandler(ISchema schema) + { + _schema = schema; + } + + protected async override Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + //try get query from body, could check http method :) + var query = await request.Content.ReadAsStringAsync(); + + //if not body try query string, dont hack like this in real world.. + if(query.Length == 0) + { + var decoded = WebUtility.UrlDecode(request.RequestUri.Query); + query = decoded.Replace("?query=", ""); + } + + var result = _schema.Execute(_ => + { + _.Query = query; + }); + + //maybe check for errors and headers etc in real world? + var response = new HttpResponseMessage(HttpStatusCode.OK) + { + Content = new StringContent(result) + }; + + //ocelot will treat this like any other http request... + return response; + } + } + + public class Program + { + public static void Main(string[] args) + { + var schema = Schema.For(@" + type Hero { + id: Int + name: String + } + + type Query { + hero(id: Int): Hero + } + ", _ => { + _.Types.Include(); + }); + + new WebHostBuilder() + .UseKestrel() + .UseContentRoot(Directory.GetCurrentDirectory()) + .ConfigureAppConfiguration((hostingContext, config) => + { + config + .SetBasePath(hostingContext.HostingEnvironment.ContentRootPath) + .AddJsonFile("appsettings.json", true, true) + .AddJsonFile($"appsettings.{hostingContext.HostingEnvironment.EnvironmentName}.json", true, true) + .AddJsonFile("configuration.json") + .AddEnvironmentVariables(); + }) + .ConfigureServices(s => { + s.AddSingleton(schema); + s.AddOcelot() + .AddSingletonDelegatingHandler(); + }) + .ConfigureLogging((hostingContext, logging) => + { + logging.AddConfiguration(hostingContext.Configuration.GetSection("Logging")); + logging.AddConsole(); + }) + .UseIISIntegration() + .Configure(app => + { + app.UseOcelot().Wait(); + }) + .Build() + .Run(); + } + } +} diff --git a/samples/OcelotGraphQL/README.md b/samples/OcelotGraphQL/README.md new file mode 100644 index 00000000..06ec668a --- /dev/null +++ b/samples/OcelotGraphQL/README.md @@ -0,0 +1,71 @@ +# Ocelot using GraphQL example + +Loads of people keep asking me if Ocelot will every support GraphQL, in my mind Ocelot and GraphQL are two different things that can work together. +I would not try and implement GraphQL in Ocelot instead I would either have Ocelot in front of GraphQL to handle things like authorisation / authentication or I would +bring in the awesome [graphql-dotnet](https://github.com/graphql-dotnet/graphql-dotnet) library and use it in a [DelegatingHandler](http://ocelot.readthedocs.io/en/latest/features/delegatinghandlers.html). This way you could have Ocelot and GraphQL without the extra hop to GraphQL. This same is an example of how to do that. + +## Example + +If you run this project with + +$ dotnet run + +Use postman or something to make the following requests and you can see Ocelot and GraphQL in action together... + +GET http://localhost:5000/graphql?query={ hero(id: 4) { id name } } + +RESPONSE +```json + { + "data": { + "hero": { + "id": 4, + "name": "Tom Pallister" + } + } + } +``` + +POST http://localhost:5000/graphql + +BODY +```json + { hero(id: 4) { id name } } +``` + +RESPONSE +```json + { + "data": { + "hero": { + "id": 4, + "name": "Tom Pallister" + } + } + } +``` + +## Notes + +Please note this project never goes out to another service, it just gets the data for GraphQL in memory. You would need to add the details of your GraphQL server in configuration.json e.g. + +```json +{ + "ReRoutes": [ + { + "DownstreamPathTemplate": "/graphql", + "DownstreamScheme": "http", + "DownstreamHostAndPorts": [ + { + "Host": "yourgraphqlhost.com", + "Port": 80 + } + ], + "UpstreamPathTemplate": "/graphql", + "DelegatingHandlers": [ + "GraphQlDelegatingHandler" + ] + } + ] + } +``` \ No newline at end of file diff --git a/samples/OcelotGraphQL/configuration.json b/samples/OcelotGraphQL/configuration.json new file mode 100644 index 00000000..115006f9 --- /dev/null +++ b/samples/OcelotGraphQL/configuration.json @@ -0,0 +1,19 @@ +{ + "ReRoutes": [ + { + "DownstreamPathTemplate": "/", + "DownstreamScheme": "http", + "DownstreamHostAndPorts": [ + { + "Host": "jsonplaceholder.typicode.com", + "Port": 80 + } + ], + "UpstreamPathTemplate": "/graphql", + "DelegatingHandlers": [ + "GraphQlDelegatingHandler" + ] + } + ] + } + \ No newline at end of file