Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers.

ASP.NET Core 쪽을 웹을 통해 조금씩 접하는 것이 못내 아쉬워서 누군가 책을 낸 것이 있다면 정독해야겠다 싶은 차에... 마침 딱 원하던 책이 나왔습니다. ^^

ASP.NET & Core를 다루는 기술 : 웹 응용프로그램 제작 기술의 집합체 
; http://www.yes24.com/24/Goods/32692615?Acode=101

ASP.NET Core만을 원했지만, 기존의 ASP.NET WebForms/MVC를 모두 다루고 있어서 우선 그 부분부터 읽게 되었는데요. 196페이지에 <asp:Substitution /> 컨트롤을 다루면서 다음과 같은 흥미로운 설명을 하고 있습니다.

IIS Express가 아닌 마이크로소프트 애저의 웹 앱과 같은 곳에 업로드해 실제 서버(IIS 웹 서버)에서 실행하면 정상적으로 동작한다...

실제로 다음과 같은 코드로 웹 페이지를 만들어 Substitution 컨트롤을 OutputCache와 함께 사용해 보면,

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default.aspx.cs" Inherits="WebApplication1.Default" %>
<%@ OutputCache Duration="60" VaryByParam="None" %>
<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
    <div>
        <asp:Literal ID="Literal1" runat="server"></asp:Literal>
        <asp:Substitution ID="subs" runat="server" MethodName="GetContents" />
    </div>
    </form>
</body>
</html>

using System;
using System.Text;
using System.Web;

namespace WebApplication1
{
    public partial class Default : System.Web.UI.Page
    {
        protected void Page_Load(object sender, EventArgs e)
        {
        }

        public static string GetContents(HttpContext context)
        {
            return DateTime.Now.ToString();
        }
    }
}

실행 시 다음과 같은 오류 페이지가 뜹니다.

Server Error in '/' Application.

Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers. ?Either a native module in the pipeline has modified an HTTP_DATA_CHUNK structure associated with a managed post cache substitution callback, or a managed filter has modified the response. 
Description: An unhandled exception occurred during the execution of the current web request. Please review the stack trace for more information about the error and where it originated in the code. 

Exception Details: System.InvalidOperationException: Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers. ?Either a native module in the pipeline has modified an HTTP_DATA_CHUNK structure associated with a managed post cache substitution callback, or a managed filter has modified the response.

Source Error: 

An unhandled exception was generated during the execution of the current web request. Information regarding the origin and location of the exception can be identified using the exception stack trace below. 

Stack Trace: 


[InvalidOperationException: Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers.  Either a native module in the pipeline has modified an HTTP_DATA_CHUNK structure associated with a managed post cache substitution callback, or a managed filter has modified the response.]
   System.Web.HttpWriter.GetIntegratedSnapshot(Boolean& hasSubstBlocks, IIS7WorkerRequest wr) +3438653
   System.Web.HttpResponse.GetSnapshot() +122
   System.Web.Caching.OutputCacheModule.OnLeave(Object source, EventArgs eventArgs) +9859908
   System.Web.SyncEventExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute() +141
   System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously) +69

Version Information:?Microsoft .NET Framework Version:4.0.30319; ASP.NET Version:4.6.1586.0 

그런데, 책의 설명과는 달리 제 컴퓨터에 설치된 IIS에서 실행해 봐도 여전히 저 오류가 뜹니다. 오호~~~ 책의 설명에서는 "실제 서버(IIS 웹 서버)"에서 실행하면 된다고 했는데 어찌 된 일인지 여전히 안됩니다.

문제 분석을 위해 오류 메시지를 보겠습니다.

Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers.

캐시 부분 적용은, response 버퍼를 변경하는 IIS 통합 파이프라인 모드의 HttpModule과는 호환되지 않는다는 것입니다. 그렇다면, 이것은 IIS Express와 IIS 실제 서버의 문제가 아닙니다. 분명히, 출력 버퍼를 변경하는 HttpModule이 있다는 것을 의미합니다.

이를 위해 현재 IIS (Express) 프로세스에 올라온 HttpModule을 확인해 보았습니다.

protected void Page_Load(object sender, EventArgs e)
{
    HttpApplication httpApps = HttpContext.Current.ApplicationInstance;
    StringBuilder sb = new StringBuilder();

    sb.Append(DateTime.Now.ToString() + "<br />");

    foreach (var module in httpApps.Modules)
    {
        sb.Append(module + "<br />\r\n");
    }

    Literal1.Text = sb.ToString();
}

출력 결과는 다음과 같습니다.

OutputCache
Session
WindowsAuthentication
FormsAuthentication
DefaultAuthentication
RoleManager
UrlAuthorization
FileAuthorization
AnonymousIdentification
Profile
UrlMappingsModule
ServiceModel-4.0
UrlRoutingModule-4.0
ScriptModule-4.0
__DynamicModule_Microsoft.VisualStudio.Web.PageInspector.Runtime.Tracing.PageInspectorHttpModule, Microsoft.VisualStudio.Web.PageInspector.Runtime, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a_d7f285f6-3836-41bf-894a-ec59b545f2ba

위의 목록 중 마지막 것을 제외하고는 모두 web.config의 다음 설정으로 비활성화 시킬 수 있습니다.

<?xml version="1.0"?>
<configuration>
  <system.web>
    ...[생략]...
  </system.web>
  <system.codedom>
    ...[생략]...
  </system.codedom>

    <system.webServer>
        <modules>
            <remove name="OutputCache"/>
            <remove name="Session"/>
            <remove name="WindowsAuthentication"/>
            <remove name="FormsAuthentication"/>
            <remove name="DefaultAuthentication"/>
            <remove name="RoleManager"/>
            <remove name="UrlAuthorization"/>
            <remove name="FileAuthorization"/>
            <remove name="AnonymousIdentification"/>
            <remove name="Profile"/>
            <remove name="UrlMappingsModule"/>
            <remove name="ServiceModel-4.0"/>
            <remove name="UrlRoutingModule-4.0"/>
            <remove name="ScriptModule-4.0"/>
        </modules>
    </system.webServer>
</configuration>

물론, 위와 같이 하면 OutputCache HttpModule도 비활성화되기 때문에 .aspx에 "<%@ OutputCache Duration="60" VaryByParam="None" %>" 설정한 것이 동작하지 않습니다. 그렇기 때문에 Substitution 컨트롤을 사용할 의미도 없습니다.

따라서, 문제의 범위를 줄이기 위해 다음과 같이 web.config을 설정하고,

<?xml version="1.0"?>
<configuration>
  <system.web>
    ...[생략]...
  </system.web>
  <system.codedom>
    ...[생략]...
  </system.codedom>

    <system.webServer>
        <modules>
            <!--<remove name="OutputCache"/>-->
            <remove name="Session"/>
            <remove name="WindowsAuthentication"/>
            <remove name="FormsAuthentication"/>
            <remove name="DefaultAuthentication"/>
            <remove name="RoleManager"/>
            <remove name="UrlAuthorization"/>
            <remove name="FileAuthorization"/>
            <remove name="AnonymousIdentification"/>
            <remove name="Profile"/>
            <remove name="UrlMappingsModule"/>
            <remove name="ServiceModel-4.0"/>
            <remove name="UrlRoutingModule-4.0"/>
            <remove name="ScriptModule-4.0"/>
        </modules>
    </system.webServer>
</configuration>

실행하면 이제 단 2개의 HttpModule만 활성화될 텐데, 마찬가지로 "Post cache substitution is not compatible with modules in the IIS integrated pipeline that modify the response buffers." 오류는 여전히 발생합니다.

즉, Response 버퍼를 바꿔 문제가 되는 바로 그 HttpModule이 "__DynamicModule_Microsoft.VisualStudio.Web.PageInspector.Runtime.Tracing.PageInspectorHttpModule ..."인 것입니다.

이 모듈을 활성화시키지 않으려면, web.config에 다음의 설정을 추가해야 합니다.

<configuration>
  <system.web>
    <compilation debug="true" targetFramework="4.6">
          <assemblies>
              <remove assembly="Microsoft.VisualStudio.Web.PageInspector.Loader, Version=1.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" />
          </assemblies>
      </compilation>
        ...[생략]...
  </system.web>
  <system.codedom>
        ...[생략]...
  </system.codedom>

    <system.webServer>
        ...[생략]...
    </system.webServer>
</configuration>

이렇게 하고 실행하면, IIS 또는 Express에 상관없이 OutputCache와 Substitution 컨트롤이 잘 동작합니다. (물론, 그 외의 다른 모듈을 모두 활성화시켜도 됩니다.)

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



문제가 되는 PageInspector는 Visual Studio가 설치되는 경우 다음의 폴더에 있는,

x86 CLR4: C:\Windows\Microsoft.NET\Framework\v4.0.30319\Config
x64 CLR4: C:\Windows\Microsoft.NET\Framework64\v4.0.30319\Config

web.config에 아래와 같은 설정을 추가합니다.

<compilation>
    <assemblies>
        <remove assembly="Microsoft.VisualStudio.Web.PageInspector.Loader, Version=1.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" />
        <add assembly="Microsoft.VisualStudio.Web.PageInspector.Loader, Version=1.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" />
        ...[생략]...
    </assemblies>
</compilation>

이 때문에, 책에서는 "IIS Express가 아닌 마이크로소프트 애저의 웹 앱과 같은 곳에 업로드해 실제 서버(IIS 웹 서버)에서 실행하면 정상적으로 동작한다..."라고 오해를 하게 된 것입니다.

실제로, Visual Studio가 설치되지 않은 다른 컴퓨터에서 이 글의 예제를 실행해 보면 다음과 같은 모듈 목록을 볼 수 있습니다.

OutputCache
Session
WindowsAuthentication
FormsAuthentication
DefaultAuthentication
RoleManager
UrlAuthorization
FileAuthorization
AnonymousIdentification
Profile
UrlMappingsModule
UrlRoutingModule-4.0
ScriptModule-4.0
ServiceModel-4.0

그래서 Azure의 웹 앱 환경에서는 잘 동작했던 것이고. ^^





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







[최초 등록일: ]
[최종 수정일: 11/10/2016]

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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...
NoWriterDateCnt.TitleFile(s)
13246정성태2/6/20234084개발 환경 구성: 664. Hyper-V에 설치한 리눅스 VM의 VHD 크기 늘리는 방법 - 두 번째 이야기
13245정성태2/6/20234626.NET Framework: 2093. C# - PEM 파일을 이용한 RSA 개인키/공개키 설정 방법파일 다운로드1
13244정성태2/5/20233983VS.NET IDE: 179. Visual Studio - External Tools에 Shell 내장 명령어 등록
13243정성태2/5/20234851디버깅 기술: 190. windbg - Win32 API 호출 시점에 BP 거는 방법 [1]
13242정성태2/4/20234292디버깅 기술: 189. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.UnauthorizedAccessException
13241정성태2/3/20233820디버깅 기술: 188. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.IO.FileNotFoundException
13240정성태2/1/20233981디버깅 기술: 187. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.Web.HttpException
13239정성태2/1/20233623디버깅 기술: 186. C# - CacheDependency의 숨겨진 예외 - System.Web.HttpException
13238정성태1/31/20235630.NET Framework: 2092. IIS 웹 사이트를 TLS 1.2 또는 TLS 1.3 프로토콜로만 운영하는 방법
13237정성태1/30/20235312.NET Framework: 2091. C# - 웹 사이트가 어떤 버전의 TLS/SSL을 지원하는지 확인하는 방법
13236정성태1/29/20234961개발 환경 구성: 663. openssl을 이용해 인트라넷 IIS 사이트의 SSL 인증서 생성
13235정성태1/29/20234505개발 환경 구성: 662. openssl - 윈도우 환경의 명령행에서 SAN 적용하는 방법
13234정성태1/28/20235541개발 환경 구성: 661. dnSpy를 이용해 소스 코드가 없는 .NET 어셈블리의 코드를 변경하는 방법 [1]
13233정성태1/28/20236894오류 유형: 840. C# - WebClient로 https 호출 시 "The request was aborted: Could not create SSL/TLS secure channel" 예외 발생
13232정성태1/27/20234697스크립트: 43. uwsgi의 --processes와 --threads 옵션
13231정성태1/27/20233611오류 유형: 839. python - TypeError: '...' object is not callable
13230정성태1/26/20234030개발 환경 구성: 660. WSL 2 내부로부터 호스트 측의 네트워크로 UDP 데이터가 1개의 패킷으로만 제한되는 문제
13229정성태1/25/20234966.NET Framework: 2090. C# - UDP Datagram의 최대 크기
13228정성태1/24/20235110.NET Framework: 2089. C# - WMI 논리 디스크가 속한 물리 디스크의 정보를 얻는 방법 [2]파일 다운로드1
13227정성태1/23/20234819개발 환경 구성: 659. Windows - IP MTU 값을 바꿀 수 있을까요? [1]
13226정성태1/23/20234490.NET Framework: 2088. .NET 5부터 지원하는 GetRawSocketOption 사용 시 주의할 점
13225정성태1/21/20233746개발 환경 구성: 658. Windows에서 실행 중인 소켓 서버를 다른 PC 또는 WSL에서 접속할 수 없는 경우
13224정성태1/21/20234089Windows: 221. Windows - Private/Public/Domain이 아닌 네트워크 어댑터 단위로 방화벽을 on/off하는 방법
13223정성태1/20/20234281오류 유형: 838. RDP 연결 오류 - The two computers couldn't connect in the amount of time allotted
13222정성태1/20/20233932개발 환경 구성: 657. WSL - DockerDesktop.vhdx 파일 위치를 옮기는 방법
13221정성태1/19/20234162Linux: 57. C# - 리눅스 프로세스 메모리 정보파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...