RpcHttpRouter.cs 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. using JsonRPC4.Common.Tools;
  2. using JsonRPC4.Router.Abstractions;
  3. using JsonRPC4.Router.Defaults;
  4. using JsonRPC4.Router.Utilities;
  5. using Microsoft.AspNetCore.Http;
  6. using Microsoft.AspNetCore.Routing;
  7. using Microsoft.Extensions.DependencyInjection;
  8. using Microsoft.Extensions.Logging;
  9. using System;
  10. using System.Collections.Generic;
  11. using System.IO;
  12. using System.IO.Compression;
  13. using System.Linq;
  14. using System.Threading.Tasks;
  15. namespace JsonRPC4.Router
  16. {
  17. public class RpcHttpRouter : IRouter
  18. {
  19. private static readonly char[] encodingSeperators = new char[2]
  20. {
  21. ',',
  22. ' '
  23. };
  24. private IRpcMethodProvider methodProvider
  25. {
  26. get;
  27. }
  28. public RpcHttpRouter(IRpcMethodProvider methodProvider)
  29. {
  30. this.methodProvider = methodProvider;
  31. }
  32. public VirtualPathData GetVirtualPath(VirtualPathContext context)
  33. {
  34. return null;
  35. }
  36. public async Task RouteAsync(RouteContext context)
  37. {
  38. ILogger<RpcHttpRouter> logger = context.HttpContext.RequestServices.GetService<ILogger<RpcHttpRouter>>();
  39. try
  40. {
  41. RpcPath requestPath;
  42. if (!context.HttpContext.Request.Path.HasValue)
  43. {
  44. requestPath = null;
  45. goto IL_00d3;
  46. }
  47. if (RpcPath.TryParse(MemoryExtensions.AsSpan(context.HttpContext.Request.Path.Value), out requestPath))
  48. {
  49. goto IL_00d3;
  50. }
  51. logger?.LogInformation("Could not parse the path '" + context.HttpContext.Request.Path.Value + "' for the request into an rpc path. Skipping rpc router middleware.");
  52. goto end_IL_0035;
  53. IL_00d3:
  54. logger?.LogInformation($"Rpc request with route '{requestPath}' started.");
  55. IRpcRequestHandler requestHandler = context.HttpContext.RequestServices.GetRequiredService<IRpcRequestHandler>();
  56. IRouteContext routeContext = DefaultRouteContext.FromHttpContext(context.HttpContext, methodProvider);
  57. Stream writableStream = BuildWritableResponseStream(context.HttpContext);
  58. using (MemoryStream requestBody = new MemoryStream())
  59. {
  60. await context.HttpContext.Request.Body.CopyToAsync(requestBody);
  61. requestBody.Position = 0L;
  62. if (!(await requestHandler.HandleRequestAsync(requestPath, requestBody, routeContext, writableStream)))
  63. {
  64. context.HttpContext.Response.StatusCode = 204;
  65. context.MarkAsHandled();
  66. return;
  67. }
  68. }
  69. context.MarkAsHandled();
  70. logger?.LogInformation("Rpc request complete");
  71. end_IL_0035:;
  72. }
  73. catch (Exception ex)
  74. {
  75. string message = "Unknown exception occurred when trying to process Rpc request. Marking route unhandled";
  76. logger?.LogException(ex, message);
  77. context.MarkAsHandled();
  78. }
  79. }
  80. private Stream BuildWritableResponseStream(HttpContext httpContext)
  81. {
  82. httpContext.Response.ContentType = "application/json";
  83. string text = httpContext.Request.Headers["Accept-Encoding"];
  84. if (!string.IsNullOrWhiteSpace(text))
  85. {
  86. IStreamCompressor service = httpContext.RequestServices.GetService<IStreamCompressor>();
  87. if (service != null)
  88. {
  89. string[] array = text.Split(encodingSeperators, StringSplitOptions.RemoveEmptyEntries);
  90. foreach (string text2 in array)
  91. {
  92. if (service.TryGetCompressionStream(httpContext.Response.Body, text2, CompressionMode.Compress, out Stream compressedStream))
  93. {
  94. httpContext.Response.Headers.Add("Content-Encoding", text2);
  95. return compressedStream;
  96. }
  97. }
  98. }
  99. }
  100. return httpContext.Response.Body;
  101. }
  102. }
  103. }