Microsoft MVP성태의 닷넷 이야기
.NET Framework: 887. C# - ASP.NET 웹 응용 프로그램의 출력 가로채기 [링크 복사], [링크+제목 복사],
조회: 21829
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# - ASP.NET 웹 응용 프로그램의 출력 가로채기

ASP.NET의 출력을 가로채는 것은 HttpResponse.Filter 속성을 이용해 쉽게 구현할 수 있습니다.

HttpResponse.Filter Property
; https://learn.microsoft.com/en-us/dotnet/api/system.web.httpresponse.filter?view=netframework-4.8

따라서 BeginRequest (또는 Page_Load 등의 원하는) 시점에 다음과 같이 Filter 속성을 대체해 주고,

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Security;
using System.Web.SessionState;

namespace WebApplication2
{
    public class Global : System.Web.HttpApplication
    {

        protected void Application_Start(object sender, EventArgs e)
        {

        }

        protected void Session_Start(object sender, EventArgs e)
        {

        }

        protected void Application_BeginRequest(object sender, EventArgs e)
        {
            HttpContext context = HttpContext.Current;
            context.Response.Filter = new FilterStream(context.Response.Filter, context.Response.ContentEncoding);
        }

        protected void Application_AuthenticateRequest(object sender, EventArgs e)
        {

        }

        protected void Application_Error(object sender, EventArgs e)
        {

        }

        protected void Session_End(object sender, EventArgs e)
        {

        }

        protected void Application_End(object sender, EventArgs e)
        {

        }
    }
}

출력을 가로챈 클래스의 요건은 Stream 클래스만 상속받으면 됩니다.

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Web;

namespace WebApplication2
{
    public class FilterStream : Stream
    {
        private readonly Stream _orginalStream;
        private readonly Encoding _encoding;

        public FilterStream(Stream originalStream, Encoding encoding)
        {
            _orginalStream = originalStream;
            _encoding = encoding;
        }

        public override bool CanRead => _orginalStream.CanRead;

        public override bool CanSeek => _orginalStream.CanSeek;

        public override bool CanWrite => _orginalStream.CanWrite;

        public override long Length => _orginalStream.Length;

        public override long Position { get => _orginalStream.Position; set => _orginalStream.Position = value; }

        public override void Flush()
        {
            _orginalStream.Flush();
        }

        public override int Read(byte[] buffer, int offset, int count)
        {
            return _orginalStream.Read(buffer, offset, count);
        }

        public override long Seek(long offset, SeekOrigin origin)
        {
            return _orginalStream.Seek(offset, origin);
        }

        public override void SetLength(long value)
        {
            _orginalStream.SetLength(value);
        }

        public override void Write(byte[] buffer, int offset, int count)
        {
            _orginalStream.Write(buffer, offset, count);
        }
    }
}

중요한 것은 Write 메서드인데, 입력으로 받는 buffer를 _orginalStream.Write 메서드에 어떻게 전달하느냐에 따라 추가/삭제/변경 등을 할 수 있습니다. 예를 들어 다음과 같이 구현하면,

public override void Write(byte[] buffer, int offset, int count)
{
    _orginalStream.Write(buffer, offset, count);

    byte[] buf = _encoding.GetBytes("TEST");
    _orginalStream.Write(buf, 0, buf.Length);
}

Write 메서드가 발생할 때마다, 즉 ASP.NET의 내부 동작에 의해 출력되는 HTTP chunk마다 마지막에 "TEST"라는 문자열이 추가됩니다. 따라서, 다음과 같이 (대개의 경우) 한 번의 chunk로 출력되는 작은 웹 페이지는,

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default.aspx.cs" Inherits="WebApplication2.Default" %>

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
        <div>
            test is good
        </div>
    </form>
</body>
</html>

출력 시 다음과 같이 "TEST" 문자열이 마지막에 붙는 식으로 동작하게 됩니다.

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head><title>

</title></head>
<body>
    <form method="post" action="./" id="form1">
<div class="aspNetHidden">
<input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwULLTE2MTY2ODcyMjlkZLcQMOQaGGJkAZFI3/Vw49ifxpRjSfnundrgHmYnfO7H" />
</div>

<div class="aspNetHidden">

    <input type="hidden" name="__VIEWSTATEGENERATOR" id="__VIEWSTATEGENERATOR" value="CA0B0334" />
</div>
        <div>
            test is good
        </div>
    </form>
</body>
</html>
TEST

여기서, chunk 단위라는 점을 주의해야 하는데, 이 때문에 일부 내용을 변경해야 하는 식이라면 chunk에 걸쳐서 출력이 되는 것도 감안하거나 아예 로직의 단순함을 위해 그런 부분은 포기하는 선택을 해야 합니다. 게다가, 이미 인코딩이 끝난 byte [] 형식이기 때문에 변경해야 하는 문자열이 ascii가 아니라면 인코딩 방식에 따라 꽤나 복잡한 작업이 될 수 있습니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/9/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  [52]  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12718정성태7/19/202116305개발 환경 구성: 581. Windows에서 WSL로 파일 복사 시 root 소유권으로 적용되는 문제파일 다운로드1
12717정성태7/18/202117307Windows: 195. robocopy에서 파일의 ADS(Alternate Data Stream) 정보 복사를 제외하는 방법
12716정성태7/17/202117243개발 환경 구성: 580. msbuild의 Exec Task에 robocopy를 사용하는 방법파일 다운로드1
12715정성태7/17/202124675오류 유형: 736. Windows - MySQL zip 파일 버전의 "mysqld --skip-grant-tables" 실행 시 비정상 종료 [1]
12714정성태7/16/202117772오류 유형: 735. VCRUNTIME140.dll, MSVCP140.dll, VCRUNTIME140.dll, VCRUNTIME140_1.dll이 없어 exe 실행이 안 되는 경우
12713정성태7/16/202120965.NET Framework: 1077. C# - 동기 방식이면서 비동기 규약을 따르게 만드는 Task.FromResult파일 다운로드1
12712정성태7/15/202119154개발 환경 구성: 579. Azure - 리눅스 호스팅의 Site Extension 제작 방법
12711정성태7/15/202118070개발 환경 구성: 578. Azure - Java Web App Service를 위한 Site Extension 제작 방법
12710정성태7/15/202122659개발 환경 구성: 577. MQTT - emqx.io 서비스 소개
12709정성태7/14/202117876Linux: 42. 실행 중인 docker 컨테이너에 대한 구동 시점의 docker run 명령어를 확인하는 방법
12708정성태7/14/202122558Linux: 41. 리눅스 환경에서 디스크 용량 부족 시 원인 분석 방법
12707정성태7/14/202188128오류 유형: 734. MySQL - Authentication method 'caching_sha2_password' not supported by any of the available plugins.
12706정성태7/14/202119624.NET Framework: 1076. C# - AsyncLocal 기능을 CallContext만으로 구현하는 방법 [2]파일 다운로드1
12705정성태7/13/202120738VS.NET IDE: 168. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 - 두 번째 이야기
12704정성태7/12/202118831개발 환경 구성: 576. Azure VM의 서비스를 Azure Web App Service에서만 접근하도록 NSG 설정을 제한하는 방법
12703정성태7/11/202123411개발 환경 구성: 575. Azure VM에 (ICMP) ping을 허용하는 방법
12702정성태7/11/202120489오류 유형: 733. TaskScheduler에 등록된 wacs.exe의 Let's Encrypt 인증서 업데이트 문제
12701정성태7/9/202120263.NET Framework: 1075. C# - ThreadPool의 스레드는 반환 시 ThreadStatic과 AsyncLocal 값이 초기화 될까요?파일 다운로드1
12700정성태7/8/202120967.NET Framework: 1074. RuntimeType의 메모리 누수? [1]
12699정성태7/8/202119459VS.NET IDE: 167. Visual Studio 디버깅 중 GC Heap 상태를 보여주는 "Show Diagnostic Tools" 메뉴 사용법
12698정성태7/7/202123090오류 유형: 732. Windows 11 업데이트 시 3% 또는 0%에서 다운로드가 멈춘 경우
12697정성태7/7/202117230개발 환경 구성: 574. Windows 11 (Insider Preview) 설치하는 방법
12696정성태7/6/202119380VC++: 146. 운영체제의 스레드 문맥 교환(Context Switch)을 유사하게 구현하는 방법파일 다운로드2
12695정성태7/3/202119646VC++: 145. C 언어의 setjmp/longjmp 기능을 Thread Context를 이용해 유사하게 구현하는 방법파일 다운로드1
12694정성태7/2/202120555Java: 24. Azure - Spring Boot 앱을 Java SE(Embedded Web Server)로 호스팅 시 로그 파일 남기는 방법 [1]
12693정성태6/30/202117606오류 유형: 731. Azure Web App Site Extension - Failed to install web app extension [...]. {1}
... 46  47  48  49  50  51  [52]  53  54  55  56  57  58  59  60  ...