mirror of
				https://github.com/nsnail/Ocelot.git
				synced 2025-10-26 20:49:24 +08:00 
			
		
		
		
	Added first authentiction code..we have a test that makes sure we are unauthenticed but i havent been able to get authenticated to work yet due to identity server usual madness when calling with their SDK!
This commit is contained in:
		| @@ -1,7 +1,6 @@ | |||||||
| echo ------------------------- | echo ------------------------- | ||||||
|  |  | ||||||
| echo Building Ocelot | echo Building Ocelot | ||||||
|  |  | ||||||
| dotnet restore src/Ocelot | dotnet restore src/Ocelot | ||||||
| dotnet restore src/Ocelot.Library | dotnet restore src/Ocelot.Library | ||||||
| dotnet build src/Ocelot | dotnet build src/Ocelot | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Configuration.Yaml | namespace Ocelot.Library.Infrastructure.Configuration.Yaml | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Configuration.Yaml | namespace Ocelot.Library.Infrastructure.Configuration.Yaml | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Configuration.Yaml | namespace Ocelot.Library.Infrastructure.Configuration.Yaml | ||||||
| { | { | ||||||
|     public class DownstreamTemplateAlreadyUsedError : Error |     public class DownstreamTemplateAlreadyUsedError : Error | ||||||
|     { |     { | ||||||
|         public DownstreamTemplateAlreadyUsedError(string message) : base(message) |         public DownstreamTemplateAlreadyUsedError(string message) : base(message, OcelotErrorCode.DownstreamTemplateAlreadyUsedError) | ||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ using System.Collections.Generic; | |||||||
| using System.Linq; | using System.Linq; | ||||||
| using Microsoft.Extensions.Options; | using Microsoft.Extensions.Options; | ||||||
| using Ocelot.Library.Infrastructure.Configuration; | using Ocelot.Library.Infrastructure.Configuration; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| using Ocelot.Library.Infrastructure.UrlMatcher; | using Ocelot.Library.Infrastructure.UrlMatcher; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,13 +2,14 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.DownstreamRouteFinder | namespace Ocelot.Library.Infrastructure.DownstreamRouteFinder | ||||||
| { | { | ||||||
|     public class UnableToFindDownstreamRouteError : Error |     public class UnableToFindDownstreamRouteError : Error | ||||||
|     { |     { | ||||||
|         public UnableToFindDownstreamRouteError() : base("UnableToFindDownstreamRouteError") |         public UnableToFindDownstreamRouteError() : base("UnableToFindDownstreamRouteError", OcelotErrorCode.UnableToFindDownstreamRouteError) | ||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								src/Ocelot.Library/Infrastructure/Errors/Error.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/Ocelot.Library/Infrastructure/Errors/Error.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | namespace Ocelot.Library.Infrastructure.Errors | ||||||
|  | { | ||||||
|  |     public abstract class Error  | ||||||
|  |     { | ||||||
|  |         protected Error(string message, OcelotErrorCode code) | ||||||
|  |         { | ||||||
|  |             Message = message; | ||||||
|  |             Code = code; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public string Message { get; private set; } | ||||||
|  |         public OcelotErrorCode Code { get; private set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								src/Ocelot.Library/Infrastructure/Errors/OcelotErrorCode.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/Ocelot.Library/Infrastructure/Errors/OcelotErrorCode.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | namespace Ocelot.Library.Infrastructure.Errors | ||||||
|  | { | ||||||
|  |     public enum OcelotErrorCode | ||||||
|  |     { | ||||||
|  |         UnauthenticatedError,  | ||||||
|  |         UnknownError, | ||||||
|  |         DownstreamTemplateAlreadyUsedError, | ||||||
|  |         UnableToFindDownstreamRouteError, | ||||||
|  |         CannotAddDataError, | ||||||
|  |         CannotFindDataError, | ||||||
|  |         UnableToCompleteRequestError | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,80 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using Microsoft.AspNetCore.Builder; | ||||||
|  | using Microsoft.AspNetCore.Http; | ||||||
|  | using Ocelot.Library.Infrastructure.Configuration; | ||||||
|  | using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Repository; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
|  | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
|  | { | ||||||
|  |     public class AuthenticationMiddleware : OcelotMiddleware | ||||||
|  |     { | ||||||
|  |         private readonly RequestDelegate _next; | ||||||
|  |         private RequestDelegate _authenticationNext; | ||||||
|  |         private readonly IScopedRequestDataRepository _scopedRequestDataRepository; | ||||||
|  |         private readonly IApplicationBuilder _app; | ||||||
|  |  | ||||||
|  |         public AuthenticationMiddleware(RequestDelegate next, IApplicationBuilder app, | ||||||
|  |             IScopedRequestDataRepository scopedRequestDataRepository)  | ||||||
|  |             : base(scopedRequestDataRepository) | ||||||
|  |         { | ||||||
|  |             _next = next; | ||||||
|  |             _scopedRequestDataRepository = scopedRequestDataRepository; | ||||||
|  |             _app = app; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public async Task Invoke(HttpContext context) | ||||||
|  |         { | ||||||
|  |             var downstreamRoute = _scopedRequestDataRepository.Get<DownstreamRoute>("DownstreamRoute"); | ||||||
|  |  | ||||||
|  |             if (downstreamRoute.IsError) | ||||||
|  |             { | ||||||
|  |                 SetPipelineError(downstreamRoute.Errors); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (IsAuthenticatedRoute(downstreamRoute.Data.ReRoute)) | ||||||
|  |             { | ||||||
|  |                 //todo - build auth pipeline and then call normal pipeline if all good? | ||||||
|  |                 //create new app builder | ||||||
|  |                 var builder = _app.New(); | ||||||
|  |                 //set up any options for the authentication | ||||||
|  |                 var jwtBearerOptions = new JwtBearerOptions | ||||||
|  |                 { | ||||||
|  |                     AutomaticAuthenticate = true, | ||||||
|  |                     AutomaticChallenge = true, | ||||||
|  |                     RequireHttpsMetadata = false, | ||||||
|  |                 }; | ||||||
|  |                 //set the authentication middleware | ||||||
|  |                 builder.UseJwtBearerAuthentication(jwtBearerOptions); | ||||||
|  |                 //use mvc so we hit the catch all authorised controller | ||||||
|  |                 builder.UseMvc(); | ||||||
|  |                 //then build it | ||||||
|  |                 _authenticationNext = builder.Build(); | ||||||
|  |                 //then call it | ||||||
|  |                 await _authenticationNext(context); | ||||||
|  |                 //check if the user is authenticated | ||||||
|  |                 if (context.User.Identity.IsAuthenticated) | ||||||
|  |                 { | ||||||
|  |                     await _next.Invoke(context); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 {    | ||||||
|  |                     SetPipelineError(new List<Error> {new UnauthenticatedError($"Request for authenticated route {context.Request.Path} by {context.User.Identity.Name} was unauthenticated")}); | ||||||
|  |                 }       | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 await _next.Invoke(context); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static bool IsAuthenticatedRoute(ReRoute reRoute) | ||||||
|  |         { | ||||||
|  |             return reRoute.IsAuthenticated; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,12 +1,12 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class AuthenticationMiddlewareMiddlewareExtensions |     public static class AuthenticationMiddlewareMiddlewareExtensions | ||||||
|     { |     { | ||||||
|         public static IApplicationBuilder UseAuthenticationMiddleware(this IApplicationBuilder builder) |         public static IApplicationBuilder UseAuthenticationMiddleware(this IApplicationBuilder builder) | ||||||
|         { |         { | ||||||
|             return builder.UseMiddleware<AuthenticationMiddleware>(); |             return builder.UseMiddleware<AuthenticationMiddleware>(builder); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -3,7 +3,7 @@ using Microsoft.AspNetCore.Http; | |||||||
| using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public class DownstreamRouteFinderMiddleware : OcelotMiddleware |     public class DownstreamRouteFinderMiddleware : OcelotMiddleware | ||||||
|     { |     { | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class DownstreamRouteFinderMiddlewareExtensions |     public static class DownstreamRouteFinderMiddlewareExtensions | ||||||
|     { |     { | ||||||
| @@ -4,7 +4,7 @@ using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | |||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.UrlTemplateReplacer; | using Ocelot.Library.Infrastructure.UrlTemplateReplacer; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public class DownstreamUrlCreatorMiddleware : OcelotMiddleware |     public class DownstreamUrlCreatorMiddleware : OcelotMiddleware | ||||||
|     { |     { | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class DownstreamUrlCreatorMiddlewareExtensions |     public static class DownstreamUrlCreatorMiddlewareExtensions | ||||||
|     { |     { | ||||||
| @@ -3,7 +3,7 @@ using Microsoft.AspNetCore.Http; | |||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public class HttpRequestBuilderMiddleware : OcelotMiddleware |     public class HttpRequestBuilderMiddleware : OcelotMiddleware | ||||||
|     { |     { | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class HttpRequestBuilderMiddlewareExtensions |     public static class HttpRequestBuilderMiddlewareExtensions | ||||||
|     { |     { | ||||||
| @@ -4,7 +4,7 @@ using Ocelot.Library.Infrastructure.Repository; | |||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| using Ocelot.Library.Infrastructure.Requester; | using Ocelot.Library.Infrastructure.Requester; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public class HttpRequesterMiddleware : OcelotMiddleware |     public class HttpRequesterMiddleware : OcelotMiddleware | ||||||
|     { |     { | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class HttpRequesterMiddlewareExtensions |     public static class HttpRequesterMiddlewareExtensions | ||||||
|     { |     { | ||||||
| @@ -4,22 +4,25 @@ using Microsoft.AspNetCore.Http; | |||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.Responder; | using Ocelot.Library.Infrastructure.Responder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public class HttpResponderMiddleware : OcelotMiddleware |     public class HttpResponderMiddleware : OcelotMiddleware | ||||||
|     { |     { | ||||||
|         private readonly RequestDelegate _next; |         private readonly RequestDelegate _next; | ||||||
|         private readonly IHttpResponder _responder; |         private readonly IHttpResponder _responder; | ||||||
|         private readonly IScopedRequestDataRepository _scopedRequestDataRepository; |         private readonly IScopedRequestDataRepository _scopedRequestDataRepository; | ||||||
|  |         private readonly IErrorsToHttpStatusCodeMapper _codeMapper; | ||||||
| 
 | 
 | ||||||
|         public HttpResponderMiddleware(RequestDelegate next,  |         public HttpResponderMiddleware(RequestDelegate next,  | ||||||
|             IHttpResponder responder, |             IHttpResponder responder, | ||||||
|             IScopedRequestDataRepository scopedRequestDataRepository) |             IScopedRequestDataRepository scopedRequestDataRepository,  | ||||||
|  |             IErrorsToHttpStatusCodeMapper codeMapper) | ||||||
|             :base(scopedRequestDataRepository) |             :base(scopedRequestDataRepository) | ||||||
|         { |         { | ||||||
|             _next = next; |             _next = next; | ||||||
|             _responder = responder; |             _responder = responder; | ||||||
|             _scopedRequestDataRepository = scopedRequestDataRepository; |             _scopedRequestDataRepository = scopedRequestDataRepository; | ||||||
|  |             _codeMapper = codeMapper; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public async Task Invoke(HttpContext context) |         public async Task Invoke(HttpContext context) | ||||||
| @@ -28,11 +31,19 @@ namespace Ocelot.Library.Middleware | |||||||
| 
 | 
 | ||||||
|             if (PipelineError()) |             if (PipelineError()) | ||||||
|             { |             { | ||||||
|                 //todo obviously this needs to be better...prob look at response errors |  | ||||||
|                 // and make a decision i guess |  | ||||||
|                 var errors = GetPipelineErrors(); |                 var errors = GetPipelineErrors(); | ||||||
|                  |                  | ||||||
|                 await _responder.CreateNotFoundResponse(context); |                 var statusCode = _codeMapper.Map(errors); | ||||||
|  | 
 | ||||||
|  |                 if (!statusCode.IsError) | ||||||
|  |                 { | ||||||
|  |                     await _responder.CreateErrorResponse(context, statusCode.Data); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     await _responder.CreateErrorResponse(context, 500); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public static class HttpResponderMiddlewareExtensions |     public static class HttpResponderMiddlewareExtensions | ||||||
|     { |     { | ||||||
| @@ -1,8 +1,9 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| 
 | 
 | ||||||
| namespace Ocelot.Library.Middleware | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
| { | { | ||||||
|     public abstract class OcelotMiddleware |     public abstract class OcelotMiddleware | ||||||
|     { |     { | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  |  | ||||||
|  | namespace Ocelot.Library.Infrastructure.Middleware | ||||||
|  | { | ||||||
|  |     public class UnauthenticatedError : Error | ||||||
|  |     { | ||||||
|  |         public UnauthenticatedError(string message) : base(message, OcelotErrorCode.UnauthenticatedError) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,10 +1,11 @@ | |||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Repository | namespace Ocelot.Library.Infrastructure.Repository | ||||||
| { | { | ||||||
|     public class CannotAddDataError : Error |     public class CannotAddDataError : Error | ||||||
|     { |     { | ||||||
|         public CannotAddDataError(string message) : base(message) |         public CannotAddDataError(string message) : base(message, OcelotErrorCode.CannotAddDataError) | ||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Repository | namespace Ocelot.Library.Infrastructure.Repository | ||||||
| { | { | ||||||
|     public class CannotFindDataError : Error |     public class CannotFindDataError : Error | ||||||
|     { |     { | ||||||
|         public CannotFindDataError(string message) : base(message) |         public CannotFindDataError(string message) : base(message, OcelotErrorCode.CannotFindDataError) | ||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using Microsoft.AspNetCore.Http; | using Microsoft.AspNetCore.Http; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Repository | namespace Ocelot.Library.Infrastructure.Repository | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Net.Http; | using System.Net.Http; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System; | using System; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Requester | namespace Ocelot.Library.Infrastructure.Requester | ||||||
| @@ -6,7 +7,7 @@ namespace Ocelot.Library.Infrastructure.Requester | |||||||
|     public class UnableToCompleteRequestError : Error |     public class UnableToCompleteRequestError : Error | ||||||
|     { |     { | ||||||
|         public UnableToCompleteRequestError(Exception exception)  |         public UnableToCompleteRequestError(Exception exception)  | ||||||
|             : base($"Error making http request, exception: {exception.Message}") |             : base($"Error making http request, exception: {exception.Message}", OcelotErrorCode.UnableToCompleteRequestError) | ||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
|  | namespace Ocelot.Library.Infrastructure.Responder | ||||||
|  | { | ||||||
|  |     public class ErrorsToHttpStatusCodeMapper : IErrorsToHttpStatusCodeMapper | ||||||
|  |     { | ||||||
|  |         public Response<int> Map(List<Error> errors) | ||||||
|  |         { | ||||||
|  |             if (errors.Any(e => e.Code == OcelotErrorCode.UnauthenticatedError)) | ||||||
|  |             { | ||||||
|  |                 return new OkResponse<int>(401); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return new OkResponse<int>(404); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System.Net; | using System.Net.Http; | ||||||
| using System.Net.Http; |  | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Microsoft.AspNetCore.Http; | using Microsoft.AspNetCore.Http; | ||||||
|  |  | ||||||
| @@ -23,11 +22,11 @@ namespace Ocelot.Library.Infrastructure.Responder | |||||||
|             return context; |             return context; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public async Task<HttpContext> CreateNotFoundResponse(HttpContext context) |         public async Task<HttpContext> CreateErrorResponse(HttpContext context, int statusCode) | ||||||
|         { |         { | ||||||
|             context.Response.OnStarting(x => |             context.Response.OnStarting(x => | ||||||
|             { |             { | ||||||
|                 context.Response.StatusCode = (int)HttpStatusCode.NotFound; |                 context.Response.StatusCode = statusCode; | ||||||
|                 return Task.CompletedTask; |                 return Task.CompletedTask; | ||||||
|             }, context); |             }, context); | ||||||
|             return context; |             return context; | ||||||
|   | |||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  |  | ||||||
|  | namespace Ocelot.Library.Infrastructure.Responder | ||||||
|  | { | ||||||
|  |     public interface IErrorsToHttpStatusCodeMapper | ||||||
|  |     { | ||||||
|  |         Response<int> Map(List<Error> errors); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -7,7 +7,6 @@ namespace Ocelot.Library.Infrastructure.Responder | |||||||
|     public interface IHttpResponder |     public interface IHttpResponder | ||||||
|     { |     { | ||||||
|         Task<HttpContext> CreateResponse(HttpContext context, HttpResponseMessage response); |         Task<HttpContext> CreateResponse(HttpContext context, HttpResponseMessage response); | ||||||
|         Task<HttpContext> CreateNotFoundResponse(HttpContext context); |         Task<HttpContext> CreateErrorResponse(HttpContext context, int statusCode); | ||||||
|  |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +0,0 @@ | |||||||
| namespace Ocelot.Library.Infrastructure.Responses |  | ||||||
| { |  | ||||||
|     public abstract class Error  |  | ||||||
|     { |  | ||||||
|         public Error(string message) |  | ||||||
|         { |  | ||||||
|             Message = message; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public string Message { get; private set; } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Responses | namespace Ocelot.Library.Infrastructure.Responses | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Responses | namespace Ocelot.Library.Infrastructure.Responses | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Responses | namespace Ocelot.Library.Infrastructure.Responses | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  |  | ||||||
| namespace Ocelot.Library.Infrastructure.Responses | namespace Ocelot.Library.Infrastructure.Responses | ||||||
| { | { | ||||||
|   | |||||||
| @@ -1,48 +0,0 @@ | |||||||
| namespace Ocelot.Library.Middleware |  | ||||||
| { |  | ||||||
|     using System.Threading.Tasks; |  | ||||||
|     using Infrastructure.Configuration; |  | ||||||
|     using Infrastructure.DownstreamRouteFinder; |  | ||||||
|     using Infrastructure.Repository; |  | ||||||
|     using Microsoft.AspNetCore.Http; |  | ||||||
|  |  | ||||||
|     public class AuthenticationMiddleware : OcelotMiddleware |  | ||||||
|     { |  | ||||||
|         private readonly RequestDelegate _next; |  | ||||||
|         private readonly IScopedRequestDataRepository _scopedRequestDataRepository; |  | ||||||
|  |  | ||||||
|         public AuthenticationMiddleware(RequestDelegate next,  |  | ||||||
|             IScopedRequestDataRepository scopedRequestDataRepository)  |  | ||||||
|             : base(scopedRequestDataRepository) |  | ||||||
|         { |  | ||||||
|             _next = next; |  | ||||||
|             _scopedRequestDataRepository = scopedRequestDataRepository; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public async Task Invoke(HttpContext context) |  | ||||||
|         { |  | ||||||
|             var downstreamRoute = _scopedRequestDataRepository.Get<DownstreamRoute>("DownstreamRoute"); |  | ||||||
|  |  | ||||||
|             if (downstreamRoute.IsError) |  | ||||||
|             { |  | ||||||
|                 SetPipelineError(downstreamRoute.Errors); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (IsAuthenticatedRoute(downstreamRoute.Data.ReRoute)) |  | ||||||
|             { |  | ||||||
|                 //todo - build auth pipeline and then call normal pipeline if all good? |  | ||||||
|                 await _next.Invoke(context); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 await _next.Invoke(context); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private static bool IsAuthenticatedRoute(ReRoute reRoute) |  | ||||||
|         { |  | ||||||
|             return reRoute.IsAuthenticated; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										22
									
								
								src/Ocelot/Controllers/HomeController.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/Ocelot/Controllers/HomeController.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | using Microsoft.AspNetCore.Authorization; | ||||||
|  | using Microsoft.AspNetCore.Mvc; | ||||||
|  |  | ||||||
|  | namespace Ocelot.Controllers | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// This controller is a catch all for requests so that if we build it into a pipeline | ||||||
|  |     /// the requests get authorised. | ||||||
|  |     /// </summary> | ||||||
|  |     public class HomeController : Controller | ||||||
|  |     { | ||||||
|  |         //[Authorize] | ||||||
|  |         [Route("{*url}")] | ||||||
|  |         public void Index() | ||||||
|  |         { | ||||||
|  |             if (true == true) | ||||||
|  |             { | ||||||
|  |                  | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -10,13 +10,13 @@ using Microsoft.Extensions.Options; | |||||||
| using Ocelot.Library.Infrastructure.Configuration; | using Ocelot.Library.Infrastructure.Configuration; | ||||||
| using Ocelot.Library.Infrastructure.Configuration.Yaml; | using Ocelot.Library.Infrastructure.Configuration.Yaml; | ||||||
| using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| using Ocelot.Library.Infrastructure.Requester; | using Ocelot.Library.Infrastructure.Requester; | ||||||
| using Ocelot.Library.Infrastructure.Responder; | using Ocelot.Library.Infrastructure.Responder; | ||||||
| using Ocelot.Library.Infrastructure.UrlMatcher; | using Ocelot.Library.Infrastructure.UrlMatcher; | ||||||
| using Ocelot.Library.Infrastructure.UrlTemplateReplacer; | using Ocelot.Library.Infrastructure.UrlTemplateReplacer; | ||||||
| using Ocelot.Library.Middleware; |  | ||||||
|  |  | ||||||
| namespace Ocelot | namespace Ocelot | ||||||
| { | { | ||||||
| @@ -40,8 +40,10 @@ namespace Ocelot | |||||||
|         public void ConfigureServices(IServiceCollection services) |         public void ConfigureServices(IServiceCollection services) | ||||||
|         { |         { | ||||||
|             services.AddOptions(); |             services.AddOptions(); | ||||||
|  |             services.AddMvc(); | ||||||
|              |              | ||||||
|             services.Configure<YamlConfiguration>(Configuration); |             services.Configure<YamlConfiguration>(Configuration); | ||||||
|  |             services.AddAuthentication(); | ||||||
|  |  | ||||||
|             // Add framework services. |             // Add framework services. | ||||||
|             services.AddSingleton<IOcelotConfiguration, OcelotConfiguration>(); |             services.AddSingleton<IOcelotConfiguration, OcelotConfiguration>(); | ||||||
| @@ -52,6 +54,7 @@ namespace Ocelot | |||||||
|             services.AddSingleton<IHttpRequester, HttpClientHttpRequester>(); |             services.AddSingleton<IHttpRequester, HttpClientHttpRequester>(); | ||||||
|             services.AddSingleton<IHttpResponder, HttpContextResponder>(); |             services.AddSingleton<IHttpResponder, HttpContextResponder>(); | ||||||
|             services.AddSingleton<IRequestBuilder, HttpRequestBuilder>(); |             services.AddSingleton<IRequestBuilder, HttpRequestBuilder>(); | ||||||
|  |             services.AddSingleton<IErrorsToHttpStatusCodeMapper, ErrorsToHttpStatusCodeMapper>(); | ||||||
|  |  | ||||||
|             // see this for why we register this as singleton http://stackoverflow.com/questions/37371264/invalidoperationexception-unable-to-resolve-service-for-type-microsoft-aspnetc |             // see this for why we register this as singleton http://stackoverflow.com/questions/37371264/invalidoperationexception-unable-to-resolve-service-for-type-microsoft-aspnetc | ||||||
|             services.AddSingleton<IHttpContextAccessor, HttpContextAccessor>(); |             services.AddSingleton<IHttpContextAccessor, HttpContextAccessor>(); | ||||||
|   | |||||||
							
								
								
									
										202
									
								
								test/Ocelot.AcceptanceTests/AuthenticationTests.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								test/Ocelot.AcceptanceTests/AuthenticationTests.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.IO; | ||||||
|  | using System.Net; | ||||||
|  | using System.Net.Http; | ||||||
|  | using System.Text.Encodings.Web; | ||||||
|  | using IdentityServer4.Models; | ||||||
|  | using IdentityServer4.Services.InMemory; | ||||||
|  | using Microsoft.AspNetCore.Builder; | ||||||
|  | using Microsoft.AspNetCore.Hosting; | ||||||
|  | using Microsoft.AspNetCore.Http; | ||||||
|  | using Microsoft.AspNetCore.TestHost; | ||||||
|  | using Microsoft.Extensions.DependencyInjection; | ||||||
|  | using Ocelot.Library.Infrastructure.Configuration.Yaml; | ||||||
|  | using Shouldly; | ||||||
|  | using TestStack.BDDfy; | ||||||
|  | using Xunit; | ||||||
|  | using YamlDotNet.Serialization; | ||||||
|  |  | ||||||
|  | namespace Ocelot.AcceptanceTests | ||||||
|  | { | ||||||
|  |     public class AuthenticationTests : IDisposable | ||||||
|  |     { | ||||||
|  |         private TestServer _server; | ||||||
|  |         private HttpClient _client; | ||||||
|  |         private HttpResponseMessage _response; | ||||||
|  |         private readonly string _configurationPath; | ||||||
|  |         private StringContent _postContent; | ||||||
|  |         private IWebHost _builder; | ||||||
|  |  | ||||||
|  |         // Sadly we need to change this when we update the netcoreapp version to make the test update the config correctly | ||||||
|  |         private double _netCoreAppVersion = 1.4; | ||||||
|  |         private HttpClient _idServerClient; | ||||||
|  |         private TestServer _idServer; | ||||||
|  |  | ||||||
|  |         public AuthenticationTests() | ||||||
|  |         { | ||||||
|  |             _configurationPath = $"./bin/Debug/netcoreapp{_netCoreAppVersion}/configuration.yaml"; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public void should_return_401_using_jwt() | ||||||
|  |         { | ||||||
|  |             this.Given(x => x.GivenThereIsAnIdentityServerOn("http://localhost:51888")) | ||||||
|  |                 .And(x => x.GivenThereIsAServiceRunningOn("http://localhost:51876", 201, string.Empty)) | ||||||
|  |                 .And(x => x.GivenThereIsAConfiguration(new YamlConfiguration | ||||||
|  |                 { | ||||||
|  |                     ReRoutes = new List<YamlReRoute> | ||||||
|  |                     { | ||||||
|  |                         new YamlReRoute | ||||||
|  |                         { | ||||||
|  |                             DownstreamTemplate = "http://localhost:51876/", | ||||||
|  |                             UpstreamTemplate = "/", | ||||||
|  |                             UpstreamHttpMethod = "Post", | ||||||
|  |                             Authentication = "JwtBearerAuthentication" | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 })) | ||||||
|  |                 .And(x => x.GivenTheApiGatewayIsRunning()) | ||||||
|  |                 .And(x => x.GivenThePostHasContent("postContent")) | ||||||
|  |                 .When(x => x.WhenIPostUrlOnTheApiGateway("/")) | ||||||
|  |                 .Then(x => x.ThenTheStatusCodeShouldBe(HttpStatusCode.Unauthorized)) | ||||||
|  |                 .BDDfy(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenThePostHasContent(string postcontent) | ||||||
|  |         { | ||||||
|  |             _postContent = new StringContent(postcontent); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// This is annoying cos it should be in the constructor but we need to set up the yaml file before calling startup so its a step. | ||||||
|  |         /// </summary> | ||||||
|  |         private void GivenTheApiGatewayIsRunning() | ||||||
|  |         { | ||||||
|  |             _server = new TestServer(new WebHostBuilder() | ||||||
|  |                 .UseStartup<Startup>()); | ||||||
|  |  | ||||||
|  |             _client = _server.CreateClient(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenThereIsAConfiguration(YamlConfiguration yamlConfiguration) | ||||||
|  |         { | ||||||
|  |             var serializer = new Serializer(); | ||||||
|  |  | ||||||
|  |             if (File.Exists(_configurationPath)) | ||||||
|  |             { | ||||||
|  |                 File.Delete(_configurationPath); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             using (TextWriter writer = File.CreateText(_configurationPath)) | ||||||
|  |             { | ||||||
|  |                 serializer.Serialize(writer, yamlConfiguration); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenThereIsAServiceRunningOn(string url, int statusCode, string responseBody) | ||||||
|  |         { | ||||||
|  |             _builder = new WebHostBuilder() | ||||||
|  |                 .UseUrls(url) | ||||||
|  |                 .UseKestrel() | ||||||
|  |                 .UseContentRoot(Directory.GetCurrentDirectory()) | ||||||
|  |                 .UseIISIntegration() | ||||||
|  |                 .UseUrls(url) | ||||||
|  |                 .Configure(app => | ||||||
|  |                 { | ||||||
|  |                     app.Run(async context => | ||||||
|  |                     { | ||||||
|  |                         context.Response.StatusCode = statusCode; | ||||||
|  |                         await context.Response.WriteAsync(responseBody); | ||||||
|  |                     }); | ||||||
|  |                 }) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             _builder.Start(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenThereIsAnIdentityServerOn(string url) | ||||||
|  |         { | ||||||
|  |             var builder = new WebHostBuilder() | ||||||
|  |                 .UseUrls(url) | ||||||
|  |                 .UseKestrel() | ||||||
|  |                 .UseContentRoot(Directory.GetCurrentDirectory()) | ||||||
|  |                 .UseIISIntegration() | ||||||
|  |                 .UseUrls(url) | ||||||
|  |                 .ConfigureServices(services => | ||||||
|  |                 { | ||||||
|  |                     services.AddDeveloperIdentityServer() | ||||||
|  |                     .AddInMemoryClients(new List<Client> { | ||||||
|  |                     new Client | ||||||
|  |                     { | ||||||
|  |                         ClientId = "test", | ||||||
|  |                         AllowedGrantTypes = GrantTypes.ResourceOwnerPassword, | ||||||
|  |                         ClientSecrets = new List<Secret> {  new Secret("test".Sha256()) }, | ||||||
|  |                         AllowedScopes = new List<string> { "api1" }, | ||||||
|  |                         AllowAccessToAllScopes = true, | ||||||
|  |                         AccessTokenType = AccessTokenType.Jwt, | ||||||
|  |                         Enabled = true | ||||||
|  |  | ||||||
|  |                     } }) | ||||||
|  |                     .AddInMemoryScopes(new List<Scope> { new Scope | ||||||
|  |                     { | ||||||
|  |                         Name = "api1", | ||||||
|  |                         Description = "My API", | ||||||
|  |                         Enabled = true | ||||||
|  |  | ||||||
|  |                     }}) | ||||||
|  |                     .AddInMemoryUsers(new List<InMemoryUser> { new InMemoryUser | ||||||
|  |                     { | ||||||
|  |                         Username = "test", Password = "test", Enabled = true, Subject = "asdads" | ||||||
|  |                     }}); | ||||||
|  |                 }) | ||||||
|  |                 .Configure(app => | ||||||
|  |                 { | ||||||
|  |                     app.UseIdentityServer(); | ||||||
|  |                 }); | ||||||
|  |                  | ||||||
|  |             _idServer = new TestServer(builder); | ||||||
|  |             _idServerClient = _idServer.CreateClient(); | ||||||
|  |  | ||||||
|  |             var response = _idServerClient.GetAsync($"{url}/.well-known/openid-configuration").Result; | ||||||
|  |             response.EnsureSuccessStatusCode(); | ||||||
|  |             var content = response.Content.ReadAsStringAsync().Result; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenIHaveAToken(string url) | ||||||
|  |         { | ||||||
|  |             var tokenUrl = $"{url}/connect/token"; | ||||||
|  |             var formData = new List<KeyValuePair<string, string>> | ||||||
|  |             { | ||||||
|  |                 new KeyValuePair<string, string>("client_id", "test"), | ||||||
|  |                 new KeyValuePair<string, string>("client_secret", "test".Sha256()), | ||||||
|  |                 new KeyValuePair<string, string>("scope", "api1"), | ||||||
|  |                 new KeyValuePair<string, string>("username", "test"), | ||||||
|  |                 new KeyValuePair<string, string>("password", "test"), | ||||||
|  |                 new KeyValuePair<string, string>("grant_type", "password") | ||||||
|  |             }; | ||||||
|  |             var content = new FormUrlEncodedContent(formData); | ||||||
|  |             var response = _idServerClient.PostAsync(tokenUrl, content).Result; | ||||||
|  |             var responseContent = response.Content.ReadAsStringAsync().Result; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void WhenIPostUrlOnTheApiGateway(string url) | ||||||
|  |         { | ||||||
|  |             _response = _client.PostAsync(url, _postContent).Result; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void ThenTheStatusCodeShouldBe(HttpStatusCode expectedHttpStatusCode) | ||||||
|  |         { | ||||||
|  |             _response.StatusCode.ShouldBe(expectedHttpStatusCode); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public void Dispose() | ||||||
|  |         { | ||||||
|  |             _idServerClient?.Dispose(); | ||||||
|  |             _idServer?.Dispose(); | ||||||
|  |             _builder?.Dispose(); | ||||||
|  |             _client.Dispose(); | ||||||
|  |             _server.Dispose(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -110,6 +110,7 @@ namespace Ocelot.AcceptanceTests | |||||||
|                 .Then(x => x.ThenTheStatusCodeShouldBe(HttpStatusCode.Created)) |                 .Then(x => x.ThenTheStatusCodeShouldBe(HttpStatusCode.Created)) | ||||||
|                 .BDDfy(); |                 .BDDfy(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private void GivenThePostHasContent(string postcontent) |         private void GivenThePostHasContent(string postcontent) | ||||||
|         { |         { | ||||||
|             _postContent = new StringContent(postcontent); |             _postContent = new StringContent(postcontent); | ||||||
| @@ -184,10 +185,7 @@ namespace Ocelot.AcceptanceTests | |||||||
|  |  | ||||||
|         public void Dispose() |         public void Dispose() | ||||||
|         { |         { | ||||||
|             if (_builder != null) |             _builder?.Dispose(); | ||||||
|             { |  | ||||||
|                 _builder.Dispose(); |  | ||||||
|             } |  | ||||||
|             _client.Dispose(); |             _client.Dispose(); | ||||||
|             _server.Dispose(); |             _server.Dispose(); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -11,31 +11,32 @@ | |||||||
|  |  | ||||||
|   "testRunner": "xunit", |   "testRunner": "xunit", | ||||||
|  |  | ||||||
|   "dependencies": { |     "dependencies": { | ||||||
|     "Microsoft.NETCore.App": { |         "Microsoft.NETCore.App": { | ||||||
|       "version": "1.0.0", |             "version": "1.0.0", | ||||||
|       "type": "platform" |             "type": "platform" | ||||||
|  |         }, | ||||||
|  |         "Microsoft.AspNetCore.Mvc": "1.0.0", | ||||||
|  |         "Microsoft.AspNetCore.Server.IISIntegration": "1.0.0", | ||||||
|  |         "Microsoft.AspNetCore.Server.Kestrel": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Configuration.EnvironmentVariables": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Configuration.FileExtensions": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Configuration.Json": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Logging": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Logging.Console": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Logging.Debug": "1.0.0", | ||||||
|  |         "Microsoft.Extensions.Options.ConfigurationExtensions": "1.0.0", | ||||||
|  |         "Microsoft.AspNetCore.Http": "1.0.0", | ||||||
|  |         "Ocelot.Library": "1.0.0-*", | ||||||
|  |         "xunit": "2.1.0", | ||||||
|  |         "dotnet-test-xunit": "2.2.0-preview2-build1029", | ||||||
|  |         "Shouldly": "2.8.0", | ||||||
|  |         "Ocelot": "1.0.0-*", | ||||||
|  |         "Microsoft.AspNetCore.TestHost": "1.0.0", | ||||||
|  |         "TestStack.BDDfy": "4.3.1", | ||||||
|  |         "YamlDotNet": "3.9.0", | ||||||
|  |         "IdentityServer4": "1.0.0-rc2" | ||||||
|     }, |     }, | ||||||
|     "Microsoft.AspNetCore.Mvc": "1.0.0", |  | ||||||
|     "Microsoft.AspNetCore.Server.IISIntegration": "1.0.0", |  | ||||||
|     "Microsoft.AspNetCore.Server.Kestrel": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Configuration.EnvironmentVariables": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Configuration.FileExtensions": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Configuration.Json": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Logging": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Logging.Console": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Logging.Debug": "1.0.0", |  | ||||||
|     "Microsoft.Extensions.Options.ConfigurationExtensions": "1.0.0", |  | ||||||
|     "Microsoft.AspNetCore.Http": "1.0.0", |  | ||||||
|     "Ocelot.Library": "1.0.0-*", |  | ||||||
|     "xunit": "2.1.0", |  | ||||||
|     "dotnet-test-xunit": "2.2.0-preview2-build1029", |  | ||||||
|     "Shouldly": "2.8.0", |  | ||||||
|     "Ocelot": "1.0.0-*", |  | ||||||
|     "Microsoft.AspNetCore.TestHost": "1.0.0", |  | ||||||
|     "TestStack.BDDfy": "4.3.1", |  | ||||||
|     "YamlDotNet": "3.9.0" |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   "frameworks": { |   "frameworks": { | ||||||
|     "netcoreapp1.4": { |     "netcoreapp1.4": { | ||||||
|   | |||||||
| @@ -7,10 +7,10 @@ using Microsoft.AspNetCore.TestHost; | |||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
| using Moq; | using Moq; | ||||||
| using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | using Ocelot.Library.Infrastructure.DownstreamRouteFinder; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| using Ocelot.Library.Infrastructure.UrlMatcher; | using Ocelot.Library.Infrastructure.UrlMatcher; | ||||||
| using Ocelot.Library.Middleware; |  | ||||||
| using TestStack.BDDfy; | using TestStack.BDDfy; | ||||||
| using Xunit; | using Xunit; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| namespace Ocelot.UnitTests.Middleware | using Ocelot.Library.Infrastructure.Middleware; | ||||||
|  |  | ||||||
|  | namespace Ocelot.UnitTests.Middleware | ||||||
| { | { | ||||||
|     using System; |     using System; | ||||||
|     using System.Collections.Generic; |     using System.Collections.Generic; | ||||||
| @@ -9,7 +11,6 @@ | |||||||
|     using Library.Infrastructure.Repository; |     using Library.Infrastructure.Repository; | ||||||
|     using Library.Infrastructure.Responses; |     using Library.Infrastructure.Responses; | ||||||
|     using Library.Infrastructure.UrlMatcher; |     using Library.Infrastructure.UrlMatcher; | ||||||
|     using Library.Middleware; |  | ||||||
|     using Microsoft.AspNetCore.Hosting; |     using Microsoft.AspNetCore.Hosting; | ||||||
|     using Microsoft.AspNetCore.TestHost; |     using Microsoft.AspNetCore.TestHost; | ||||||
|     using Microsoft.Extensions.DependencyInjection; |     using Microsoft.Extensions.DependencyInjection; | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| namespace Ocelot.UnitTests.Middleware | using Ocelot.Library.Infrastructure.Middleware; | ||||||
|  |  | ||||||
|  | namespace Ocelot.UnitTests.Middleware | ||||||
| { | { | ||||||
|     using System; |     using System; | ||||||
|     using System.Collections.Generic; |     using System.Collections.Generic; | ||||||
| @@ -10,7 +12,6 @@ | |||||||
|     using Library.Infrastructure.Responses; |     using Library.Infrastructure.Responses; | ||||||
|     using Library.Infrastructure.UrlMatcher; |     using Library.Infrastructure.UrlMatcher; | ||||||
|     using Library.Infrastructure.UrlTemplateReplacer; |     using Library.Infrastructure.UrlTemplateReplacer; | ||||||
|     using Library.Middleware; |  | ||||||
|     using Microsoft.AspNetCore.Hosting; |     using Microsoft.AspNetCore.Hosting; | ||||||
|     using Microsoft.AspNetCore.TestHost; |     using Microsoft.AspNetCore.TestHost; | ||||||
|     using Microsoft.Extensions.DependencyInjection; |     using Microsoft.Extensions.DependencyInjection; | ||||||
|   | |||||||
| @@ -7,10 +7,10 @@ using Microsoft.AspNetCore.Http; | |||||||
| using Microsoft.AspNetCore.TestHost; | using Microsoft.AspNetCore.TestHost; | ||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
| using Moq; | using Moq; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| using Ocelot.Library.Middleware; |  | ||||||
| using TestStack.BDDfy; | using TestStack.BDDfy; | ||||||
| using Xunit; | using Xunit; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,11 +6,11 @@ using Microsoft.AspNetCore.Hosting; | |||||||
| using Microsoft.AspNetCore.TestHost; | using Microsoft.AspNetCore.TestHost; | ||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
| using Moq; | using Moq; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.RequestBuilder; | using Ocelot.Library.Infrastructure.RequestBuilder; | ||||||
| using Ocelot.Library.Infrastructure.Requester; | using Ocelot.Library.Infrastructure.Requester; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| using Ocelot.Library.Middleware; |  | ||||||
| using TestStack.BDDfy; | using TestStack.BDDfy; | ||||||
| using Xunit; | using Xunit; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,10 +6,10 @@ using Microsoft.AspNetCore.Http; | |||||||
| using Microsoft.AspNetCore.TestHost; | using Microsoft.AspNetCore.TestHost; | ||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
| using Moq; | using Moq; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
| using Ocelot.Library.Infrastructure.Repository; | using Ocelot.Library.Infrastructure.Repository; | ||||||
| using Ocelot.Library.Infrastructure.Responder; | using Ocelot.Library.Infrastructure.Responder; | ||||||
| using Ocelot.Library.Infrastructure.Responses; | using Ocelot.Library.Infrastructure.Responses; | ||||||
| using Ocelot.Library.Middleware; |  | ||||||
| using TestStack.BDDfy; | using TestStack.BDDfy; | ||||||
| using Xunit; | using Xunit; | ||||||
|  |  | ||||||
| @@ -19,6 +19,7 @@ namespace Ocelot.UnitTests.Middleware | |||||||
|     { |     { | ||||||
|         private readonly Mock<IHttpResponder> _responder; |         private readonly Mock<IHttpResponder> _responder; | ||||||
|         private readonly Mock<IScopedRequestDataRepository> _scopedRepository; |         private readonly Mock<IScopedRequestDataRepository> _scopedRepository; | ||||||
|  |         private readonly Mock<IErrorsToHttpStatusCodeMapper> _codeMapper; | ||||||
|         private readonly string _url; |         private readonly string _url; | ||||||
|         private readonly TestServer _server; |         private readonly TestServer _server; | ||||||
|         private readonly HttpClient _client; |         private readonly HttpClient _client; | ||||||
| @@ -30,10 +31,12 @@ namespace Ocelot.UnitTests.Middleware | |||||||
|             _url = "http://localhost:51879"; |             _url = "http://localhost:51879"; | ||||||
|             _responder = new Mock<IHttpResponder>(); |             _responder = new Mock<IHttpResponder>(); | ||||||
|             _scopedRepository = new Mock<IScopedRequestDataRepository>(); |             _scopedRepository = new Mock<IScopedRequestDataRepository>(); | ||||||
|  |             _codeMapper = new Mock<IErrorsToHttpStatusCodeMapper>(); | ||||||
|  |  | ||||||
|             var builder = new WebHostBuilder() |             var builder = new WebHostBuilder() | ||||||
|               .ConfigureServices(x => |               .ConfigureServices(x => | ||||||
|               { |               { | ||||||
|  |                   x.AddSingleton(_codeMapper.Object); | ||||||
|                   x.AddSingleton(_responder.Object); |                   x.AddSingleton(_responder.Object); | ||||||
|                   x.AddSingleton(_scopedRepository.Object); |                   x.AddSingleton(_scopedRepository.Object); | ||||||
|               }) |               }) | ||||||
|   | |||||||
| @@ -0,0 +1,72 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.IO; | ||||||
|  | using System.Net.Http; | ||||||
|  | using Microsoft.AspNetCore.Http; | ||||||
|  | using Ocelot.Library.Infrastructure.Errors; | ||||||
|  | using Ocelot.Library.Infrastructure.Middleware; | ||||||
|  | using Ocelot.Library.Infrastructure.Responder; | ||||||
|  | using Ocelot.Library.Infrastructure.Responses; | ||||||
|  | using Shouldly; | ||||||
|  | using TestStack.BDDfy; | ||||||
|  | using Xunit; | ||||||
|  |  | ||||||
|  | namespace Ocelot.UnitTests.Responder | ||||||
|  | { | ||||||
|  |     public class ErrorsToHttpStatusCodeMapperTests | ||||||
|  |     { | ||||||
|  |         private readonly IErrorsToHttpStatusCodeMapper _codeMapper; | ||||||
|  |         private Response<int> _result; | ||||||
|  |         private List<Error> _errors; | ||||||
|  |  | ||||||
|  |         public ErrorsToHttpStatusCodeMapperTests() | ||||||
|  |         { | ||||||
|  |             _codeMapper = new ErrorsToHttpStatusCodeMapper(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public void should_create_unauthenticated_response_code() | ||||||
|  |         { | ||||||
|  |             this.Given(x => x.GivenThereAreErrors(new List<Error> | ||||||
|  |                 { | ||||||
|  |                     new UnauthenticatedError("no matter") | ||||||
|  |                 })) | ||||||
|  |                 .When(x => x.WhenIGetErrorStatusCode()) | ||||||
|  |                 .Then(x => x.ThenTheResponseIsStatusCodeIs(401)) | ||||||
|  |                 .BDDfy(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public void should_create_not_found_response_response_code() | ||||||
|  |         { | ||||||
|  |             this.Given(x => x.GivenThereAreErrors(new List<Error> | ||||||
|  |                 { | ||||||
|  |                     new AnyError() | ||||||
|  |                 })) | ||||||
|  |                 .When(x => x.WhenIGetErrorStatusCode()) | ||||||
|  |                 .Then(x => x.ThenTheResponseIsStatusCodeIs(404)) | ||||||
|  |                 .BDDfy(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         class AnyError : Error | ||||||
|  |         { | ||||||
|  |             public AnyError() : base("blahh", OcelotErrorCode.UnknownError) | ||||||
|  |             { | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void GivenThereAreErrors(List<Error> errors) | ||||||
|  |         { | ||||||
|  |             _errors = errors; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void WhenIGetErrorStatusCode() | ||||||
|  |         { | ||||||
|  |             _result = _codeMapper.Map(_errors); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private void ThenTheResponseIsStatusCodeIs(int expectedCode) | ||||||
|  |         { | ||||||
|  |             _result.Data.ShouldBe(expectedCode); | ||||||
|  |         }     | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| using Microsoft.AspNetCore.Http; |  | ||||||
| using Xunit; |  | ||||||
|  |  | ||||||
| namespace Ocelot.UnitTests.Responder |  | ||||||
| { |  | ||||||
|     public class ResponderTests |  | ||||||
|     { |  | ||||||
|         [Fact] |  | ||||||
|         public void should_do_something() |  | ||||||
|         { |  | ||||||
|  |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
		Reference in New Issue
	
	Block a user
	 TomPallister
					TomPallister