첨부 실행 코드는 나눔고딕코딩 폰트를 사용합니다.
본 블로그는 광고를 포함하고 있습니다.
광고 클릭에서 발생하는 수익금은 모두 블로그 콘텐츠 향상을 위해 쓰여집니다.

728x90
반응형
728x170

TestProject.zip
다운로드

▶ Handlers/SecureRequestHandler.cs

using System;
using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;

namespace TestProject.Handlers
{
    /// <summary>
    /// 보안 요청 핸들러
    /// </summary>
    public class SecureRequestHandler : DelegatingHandler
    {
        //////////////////////////////////////////////////////////////////////////////////////////////////// Method
        ////////////////////////////////////////////////////////////////////////////////////////// Protected

        #region 송신하기 (비동기) - SendAsync(request, cancellationToken)

        /// <summary>
        /// 송신하기 (비동기)
        /// </summary>
        /// <param name="httpRequestMessage">HTTP 요청 메시지</param>
        /// <param name="cancellationToken">취소 토큰</param>
        /// <returns>HTTP 응답 메시지 태스크</returns>
        protected override Task<HttpResponseMessage> SendAsync
        (
            HttpRequestMessage httpRequestMessage,
            CancellationToken  cancellationToken
        )
        {
            if(httpRequestMessage.RequestUri.Scheme == Uri.UriSchemeHttp)
            {
                UriBuilder uriBuilder = new UriBuilder(httpRequestMessage.RequestUri)
                {
                    Scheme = Uri.UriSchemeHttps
                };

                httpRequestMessage.RequestUri = uriBuilder.Uri;
            }

            return base.SendAsync(httpRequestMessage, cancellationToken);
        }

        #endregion
    }
}

 

728x90

 

▶ Handlers/RequestDataHandler.cs

using Microsoft.Extensions.Logging;
using System;
using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;

namespace TestProject.Handlers
{
    /// <summary>
    /// 데이터 요청 핸들러
    /// </summary>
    public class RequestDataHandler : DelegatingHandler
    {
        //////////////////////////////////////////////////////////////////////////////////////////////////// Field
        ////////////////////////////////////////////////////////////////////////////////////////// Private

        #region Field

        /// <summary>
        /// 로그 기록기
        /// </summary>
        private readonly ILogger<RequestDataHandler> logger;

        /// <summary>
        /// 요청 소스 헤더명
        /// </summary>
        private const string RequestSourceHeaderName = "Request-Source";

        /// <summary>
        /// 요청 소스
        /// </summary>
        private const string RequestSource = "TestProject";

        /// <summary>
        /// 요청 ID 헤더명
        /// </summary>
        private const string RequestIDHeaderName = "Request-Identifier";

        #endregion

        //////////////////////////////////////////////////////////////////////////////////////////////////// Constructor
        ////////////////////////////////////////////////////////////////////////////////////////// Public

        #region 생성자 - RequestDataHandler(logger)

        /// <summary>
        /// 생성자
        /// </summary>
        /// <param name="logger">로그 기록기</param>
        public RequestDataHandler(ILogger<RequestDataHandler> logger)
        {
            this.logger = logger;
        }

        #endregion

        //////////////////////////////////////////////////////////////////////////////////////////////////// Method
        ////////////////////////////////////////////////////////////////////////////////////////// Protected

        #region 송신하기 (비동기) - SendAsync(httpRequestMessage, cancellationToken)

        /// <summary>
        /// 송신하기 (비동기)
        /// </summary>
        /// <param name="httpRequestMessage">HTTP 요청 메시지</param>
        /// <param name="cancellationToken">취소 토큰</param>
        /// <returns>HTTP 응답 메시지 태스크</returns>
        protected override Task<HttpResponseMessage> SendAsync
        (
            HttpRequestMessage httpRequestMessage,
            CancellationToken  cancellationToken
        )
        {
            Guid identifierGUID = Guid.NewGuid();

            this.logger.LogInformation("Starting request {Identifier}", identifierGUID);

            httpRequestMessage.Headers.Add(RequestSourceHeaderName, RequestSource            );
            httpRequestMessage.Headers.Add(RequestIDHeaderName    , identifierGUID.ToString());

            return base.SendAsync(httpRequestMessage, cancellationToken);
        }

        #endregion
    }
}

 

300x250

 

▶ Startup.cs

using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;

using TestProject.Handlers;

namespace TestProject
{
    /// <summary>
    /// 시작
    /// </summary>
    public class Startup
    {
        //////////////////////////////////////////////////////////////////////////////////////////////////// Property
        ////////////////////////////////////////////////////////////////////////////////////////// Public

        #region 구성 - Configuration

        /// <summary>
        /// 구성
        /// </summary>
        public IConfiguration Configuration { get; }

        #endregion

        //////////////////////////////////////////////////////////////////////////////////////////////////// Constructor
        ////////////////////////////////////////////////////////////////////////////////////////// Public

        #region 생성자 - Startup(configuration)

        /// <summary>
        /// 생성자
        /// </summary>
        /// <param name="configuration">구성</param>
        public Startup(IConfiguration configuration)
        {
            Configuration = configuration;
        }

        #endregion

        //////////////////////////////////////////////////////////////////////////////////////////////////// Method
        ////////////////////////////////////////////////////////////////////////////////////////// Public

        #region 서비스 컬렉션 구성하기 - ConfigureServices(services)

        /// <summary>
        /// 서비스 컬렉션 구성하기
        /// </summary>
        /// <param name="services">서비스 컬렉션</param>
        public void ConfigureServices(IServiceCollection services)
        {
            services.AddTransient<SecureRequestHandler>();
            services.AddTransient<RequestDataHandler>();

            services.AddHttpClient("test")
                // 이 핸들러는 외부에 있으며 요청 중에 먼저 호출되고 응답 중에 마지막으로 호출된다.
                .AddHttpMessageHandler<SecureRequestHandler>()
                // 이 핸들러는 전송되는 요청에 가장 가까운 내부에 있다.
                .AddHttpMessageHandler<RequestDataHandler>();

            services.AddControllersWithViews();
        }

        #endregion
        #region 구성하기 - Configure(app, environment)

        /// <summary>
        /// 구성하기
        /// </summary>
        /// <param name="app">애플리케이션 빌더</param>
        /// <param name="environment">웹 호스트 환경</param>
        public void Configure(IApplicationBuilder app, IWebHostEnvironment environment)
        {
            if(environment.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }
            else
            {
                app.UseExceptionHandler("/Home/Error");

                app.UseHsts();
            }

            app.UseHttpsRedirection();

            app.UseStaticFiles();

            app.UseRouting();

            app.UseAuthorization();

            app.UseEndpoints
            (
                endpoints =>
                {
                    endpoints.MapControllerRoute
                    (
                        name    : "default",
                        pattern : "{controller=Home}/{action=Index}/{id?}"
                    );
                }
            );
        }

        #endregion
    }
}

※ 위임 처리기를 사용하는 코드는 없으며 설정하는 코드만 있는 예제이다.

728x90
반응형
그리드형(광고전용)

'C# > ASP.NET MVC' 카테고리의 다른 글

[C#/ASP.NET MVC] PollyServiceCollectionExtensions 클래스 : AddPolicyRegistry 확장 메소드를 사용해 Polly 레지스트리에 정책 추가하기  (0) 2020.11.08
[C#/ASP.NET MVC] PollyHttpClientBuilderExtensions 클래스 : AddTransientHttpErrorPolicy 확장 메소드를 사용해 Polly 정책 중첩하기  (0) 2020.11.08
[C#/ASP.NET MVC] PollyHttpClientBuilderExtensions 클래스 : AddPolicyHandler 확장 메소드를 사용해 동적으로 정책 선택하기  (0) 2020.11.08
[C#/ASP.NET MVC] PollyHttpClientBuilderExtensions 클래스 : AddTransientHttpErrorPolicy 확장 메소드를 사용해 일시적인 오류 처리하기  (0) 2020.11.08
[C#/ASP.NET MVC] HttpClientBuilderExtensions 클래스 : AddHttpMessageHandler 확장 메소드를 사용해 위임 처리기 사용하기  (0) 2020.11.07
[C#/ASP.NET MVC] HttpClientBuilderExtensions 클래스 : AddHttpMessageHandler 확장 메소드를 사용해 위임 처리기 사용하기  (0) 2020.11.07
[C#/ASP.NET MVC] DelegatingHandler 클래스 : 데이터 요청 핸들러 사용하기  (0) 2020.11.07
[C#/ASP.NET MVC] DelegatingHandler 클래스 : 헤더 검증 핸들러 사용하기  (0) 2020.11.07
[C#/ASP.NET MVC] DelegatingHandler 클래스 : 보안 요청 핸들러 사용하기  (0) 2020.11.07
[C#/ASP.NET MVC] HttpServiceCollectionExtensions 클래스 : AddHttpContextAccessor 확장 메소드를 사용해 IHttpContextAccessor 서비스 디폴트 구현 기능 추가하기  (0) 2020.11.07
[C#/ASP.NET MVC] HttpClientFactoryServiceCollectionExtensions 클래스 : AddHttpClient 확장 메소드를 사용해 HttpClient 관련 객체 의존성 주입 사용하기  (0) 2020.11.07
Posted by icodebroker

댓글을 달아 주세요