Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

Visual Studio 확장 도구 AttachToW3WP - w3wp.exe에 대한 디버거 연결을 자동화

지난번 글을 통해서, global.asax.cs 등의 초기 코드에 대한 디버깅이 얼마나 불편한 것인지 설명을 했습니다.

IIS 7.5 - Global.asax.cs처럼 초기에 실행되는 코드에 Breakpoint를 잡는 방법
; https://www.sysnet.pe.kr/2/0/1521

IIS 8.0/8.5 - Global.asax.cs처럼 초기에 실행되는 코드에 Breakpoint를 잡는 방법
; https://www.sysnet.pe.kr/2/0/1522

자, 그럼 이 문제를 Visual Studio 확장을 이용해 개선해 보는 것은 어떨까요? 목표는 다음과 같이 "TOOLS" / "AttachToW3WP" 메뉴를 추가하고 단지 그 버튼을 누르는 것만으로 global.asax.cs의 BP에 걸리도록 하는 것입니다.

attach_to_w3wp_1.png



의외로 방법이 그리 어렵진 않습니다. 우선, 다음의 글에 설명된 대로 "Menu Command" 유형의 VSP 프로젝트를 만들고,

Visual Studio 확장(VSIX) 만드는 방법
; https://www.sysnet.pe.kr/2/0/1515

기본 생성된 코드에서 "AttachToW3WP" 메뉴를 선택하면 실행될 MenuItemCallback의 기능을 정의해 주면 됩니다.

우선, iisreset 기능 먼저 넣어야 겠지요. 하지만 잘 생각해 보면 iisreset까지는 필요없습니다. 우리가 원하는 것은 w3wp.exe의 초기화된 상태이기 때문에 단순히 recycle 명령어만 수행해 주면 되기 때문입니다. 따라서 다음과 같이 코딩할 수 있습니다.

private void MenuItemCallback(object sender, EventArgs e)
{
    // w3wp.exe를 recycle 시킨다.
    // recycle로 새롭게 생성된 프로세스를 attach시키면 되기 때문에 recycle 전에 실행되어 있는 w3wp.exe PID 목록을 반환받는다.
    List<int> currentWorkerProcess = RecycleAppPool();
    if (currentWorkerProcess == null)
    {
        return;
    }

    // currentWorkerProcess 목록을 제외한 w3wp.exe 프로세스를 디버거로 연결한다.
    AttachToW3wp(currentWorkerProcess);
}

recycle 대상이 되는 AppPool은 프로젝트 속성에서 구할 수 있습니다.

private List<int> RecycleAppPool()
{
    // 프로젝트 속성에서 recycle 대상이 되는 apppool의 metabase 경로를 구하고,
    string metabasePath = GetProjectItemValue("WebApplication.AspnetCompilerIISMetabasePath");

    if (string.IsNullOrEmpty(metabasePath) == true)
    {
        ShowMessage("Startup project is not a web project type or \"Use Local IIS Web server\" is not checked.");
        return null;
    }

    // 해당 metabase의 apppool을 recycle 시킨다.
    return StartAppPool(metabasePath);
}

string GetProjectItemValue(string propName)
{
    string uniqueName = null;

    try
    {
        // 솔루션에서 Startup 프로젝트로 지정된 프로젝트의 이름을 구하고,
        uniqueName = (string)((object[])(_applicationObject.Solution.SolutionBuild.StartupProjects))[0];
    }
    catch (NullReferenceException)
    {
        return string.Empty;
    }

    // Startup 프로젝트 객체를 반환받은 후,
    Project prj = GetProject(_applicationObject.Solution, uniqueName);
    if (prj == null)
    {
        return string.Empty;
    }

    // IIS Metabase 경로 속성을 구한다.
    foreach (var prop in prj.Properties)
    {
        Property property = prop as Property;
        if (property.Name == propName)
        {
            return property.Value as string;
        }
    }

    return string.Empty;
}

일단, recycle을 시켰으면 이제 새롭게 생성된 w3wp.exe 프로세스에 Visual Studio 디버거를 연결하면 됩니다.

private void AttachToW3wp(List<int> currentWorkerProcess)
{
    // 로컬 PC에 실행된 모든 프로세스를 열람하면서,
    foreach (var proc in _applicationObject.Debugger.LocalProcesses)
    {
        EnvDTE90.Process3 process = proc as EnvDTE90.Process3;
        if (process == null)
        {
            continue;
        }

        // 혹시 recycle 중이지만 아직 살아있는 이전 w3wp.exe가 있다면 넘어가고,
        if (currentWorkerProcess.Contains(process.ProcessID) == true)
        {
            continue;
        }

        // 새롭게 실행된 프로세스 중에서 w3wp.exe 실행 파일에 대해서만,
        if (process.Name.IndexOf("W3WP.EXE", 0, StringComparison.OrdinalIgnoreCase) != -1)
        {
            bool attached = false;
            try
            {
                // 디버거를 연결한다.
                process.Attach();
                attached = true;
            }
            catch
            {
            }

            if (attached == true)
            {
                // 디버거를 연결 후, 프로젝트의 IIS URL 속성을 가져와서,
                string startupUrl = GetProjectItemValue("WebApplication.IISUrl");
                if (string.IsNullOrEmpty(startupUrl) == false)
                {
                    // 웹 브라우저를 실행시켜 w3wp.exe가 ASP.NET 처리를 시작하게 만든다.
                    RunInternetExplorer(startupUrl);
                }
            }

            break;
        }
    }
}

사실 위의 코드는 상당히 간략화된 유형인데, 좀 더 유연한 동작을 하려면 대상 프로젝트의 .NET Framework 버전에 따라 디버거 엔진을 골라줘야 합니다.

Visual Studio 2013 기준으로 현재 지원되는 Debug Engine의 이름으로는 다음과 같은 항목들이 있습니다.

  • Silverlight
  • T-SQL
  • IntelliTrace
  • Native
  • Trace
  • Managed
  • Managed (v3.5, v3.0, v2.0)
  • Workflow
  • Managed/Native
  • Script
  • GPU - Software Emulator
  • Managed (v4.5, v4.0)

그리고, EnvDTE.Project의 Properties 컬렉션에 담긴 속성으로 구할 수 있는 값들은 대략 이렇습니다.

WebApplication.AspNetDebugging: True
WebApplication.AspnetCompilerIISMetabasePath: /LM/W3SVC/10/ROOT/
OutputTypeEx: 2
TargetFrameworkMoniker: .NETFramework,Version=v4.0
ComVisible: False
EnableSecurityDebugging: True
OptionCompare: 0
StartupObject: 
WebApplication.SSLEnabled: False
WebApplication.UseIIS: True
ManifestCertificateThumbprint: 
Trademark: 
Title: WebApplication1
WebApplication.StartExternalUrl: 
WebApplication.IISUrl: http://localhost:8034
AssemblyOriginatorKeyFileType: 1
FileName: WebApplication1.csproj
WebServer: (no eval)
AssemblyOriginatorKeyMode: 0
AssemblyKeyContainerName: 
WebApplication.WindowsAuthenticationEnabled: (no eval)
WebApplication.SecureUrl: 
WebApplication.DevelopmentServerCommandLine: 
WebApplication.SQLDebugging: False
WebApplication.StartPageUrl: 
WebApplication.DefaultServerDirectoryListing: True
WebApplication.DevelopmentServerVPath: /
WebApplication.DevelopmentServerPort: 43746
ProjectType: 0
ReferencePath: 
WebApplication.IsUsingIISExpress: False
PreBuildEvent: 
WebApplication.AnonymousAuthenticationEnabled: (no eval)
WebApplication.SilverlightDebugging: False
WebApplication.StartCmdLineArguments: 
Copyright: Copyright ⓒ  2013
ApplicationIcon: 
WebApplication.CurrentDebugUrl: http://localhost:8034/About.aspx
ExcludedPermissions: 
RunPostBuildEvent: 1
DefaultTargetSchema: 1
RootNamespace: WebApplication1
WebApplication.IsUsingCustomServer: False
ManifestTimestampUrl: 
ManifestKeyFile: 
DebugSecurityZoneURL: 
Product: WebApplication1
PostBuildEvent: 
OptionStrict: 0
DefaultHTMLPageLayout: 1
DelaySign: False
OutputType: 2
WebApplication.StartWorkingDirectory: 
WebApplication.DebugStartAction: 0
NeutralResourcesLanguage: 
OptionExplicit: 1
OutputFileName: WebApplication1.dll
ServerExtensionsVersion: (no eval)
WebApplication.NonSecureUrl: 
WebApplication.ToString: Microsoft.VisualStudio.Web.Application.WAProjectExtender
AssemblyGuid: 7ea9b99e-ec77-460c-afe9-a2857f8e4e8a
GenerateManifests: False
AssemblyVersion: 1.0.0.0
Win32ResourceFile: 
Description: 
URL: file:///d:\WebApplication1\WebApplication1\
DefaultClientScript: 0
WebApplication.NativeDebugging: False
TargetFramework: 262144
SignManifests: False
OfflineURL: (no eval)
WebServerVersion: (no eval)
Publish: (no eval)
AssemblyType: 0
FullPath: d:\WebApplication1\WebApplication1\
WebAccessMethod: (no eval)
WebApplication.UseIISExpress: False
WebApplication.BrowseURL: http://localhost:8034
WebApplication.NTLMAuthentication: False
WebApplication.OverrideIISAppRootUrl: False
WebApplication.OpenedURL: file:///d:/WebApplication1/WebApplication1/
AssemblyKeyProviderName: 
TypeComplianceDiagnostics: False
Company: 
ActiveFileSharePath: (no eval)
AssemblyOriginatorKeyFile: 
WebApplication.StartWebServerOnDebug: True
WebApplication.AutoAssignPort: True
ApplicationManifest: DefaultManifest
AssemblyFileVersion: 1.0.0.0
AspnetVersion: (no eval)
FileSharePath: (no eval)
AssemblyName: WebApplication1
WebApplication.EditAndContinue: True
WebApplication.ServerDirectoryListing: True
LocalPath: d:\WebApplication1\WebApplication1\
DefaultNamespace: WebApplication1
LinkRepair: False
WebApplication.StartExternalProgram: 
WebApplication.IISAppRootUrl: 
TargetZone: 
SignAssembly: False

이 중에서 TargetFramework의 값이 262144로 되어 있는 것이 재미있는데요. 이에 대해서는 다음의 글에서 해석 방법을 알려주고 있습니다.

HOWTO: Get the target .NET Framework of a Visual Studio 2008 project from a Visual Studio add-in or macro
; http://www.mztools.com/articles/2008/MZ2008015.aspx

즉, 262144 == 0x40000이므로 .NET 4.0 버전의 Framework인 것입니다.

참고로, 전체 소스코드는 다음의 경로에 공개되어 있으니 원하시는 분은 다운로드해도 됩니다.

AttachToW3WP 
; https://github.com/stjeong/AttachToW3WP

자, 이제 빌드하고 vsix 파일을 시스템에 설치하면 "TOOLS"에 "AttachToW3WP" 메뉴가 생성되고 이를 클릭하면 다음과 같이 웹브라우저가 실행되면서 global.asax.cs 내부에 BP가 잡히는 것을 확인할 수 있습니다.

attach_to_w3wp_2.png

역시 Visual Studio Gallery에 올려져 있기 때문에 Visual Studio 2012/2013 사용자들은 Extension Manager를 이용해 바로 검색해서 설치할 수 있습니다.

Visual Studio Gallery - AttachToW3WP
; http://visualstudiogallery.msdn.microsoft.com/14b2a959-446f-406c-bcf0-abe87fc529e7




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/27/2021]

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

비밀번호

댓글 작성자
 



2013-11-02 09시28분
[Lyn] 와... 멋집니다 저도 이런거 해보고싶네요
[guest]
2013-11-02 10시19분
넵. 지금 당장 하셔서 공개하시면 됩니당. ^^ Lyn 님도 아이디어 많으실 것 같은데요.
정성태

... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...
NoWriterDateCnt.TitleFile(s)
1303정성태6/26/201227393개발 환경 구성: 152. sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
1302정성태6/25/201229394개발 환경 구성: 151. Azure 웹 사이트에 사용자 도메인 네임 연결하는 방법
1301정성태6/20/201225763오류 유형: 156. KB2667402 윈도우 업데이트 실패 및 마이크로소프트 Answers 웹 사이트 대응
1300정성태6/20/201231759.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 [1]파일 다운로드2
1299정성태6/18/201232876제니퍼 .NET: 21. 제니퍼 닷넷 - Ninject DI 프레임워크의 성능 분석 [2]파일 다운로드2
1298정성태6/14/201234401VS.NET IDE: 72. Visual Studio에서 pfx 파일로 서명한 경우, 암호는 어디에 저장될까? [2]
1297정성태6/12/201231045VC++: 63. 다른 프로세스에 환경 변수 설정하는 방법파일 다운로드1
1296정성태6/5/201227671.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 [4]파일 다운로드1
1295정성태6/5/201225076.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기파일 다운로드1
1294정성태5/27/201248520.NET Framework: 326. 유니코드와 한글 - 유니코드와 닷넷을 이용한 한글 처리 [7]파일 다운로드2
1293정성태5/24/201229772.NET Framework: 325. System.Drawing.Bitmap 데이터를 Parallel.For로 처리하는 방법 [2]파일 다운로드1
1292정성태5/24/201223750.NET Framework: 324. First-chance exception에 대해 조건에 따라 디버거가 멈추게 할 수는 없을까? [1]파일 다운로드1
1291정성태5/23/201230272VC++: 62. 배열 초기화를 위한 기계어 코드 확인 [2]
1290정성태5/18/201235077.NET Framework: 323. 관리자 권한이 필요한 작업을 COM+에 대행 [7]파일 다운로드1
1289정성태5/17/201239237.NET Framework: 322. regsvcs.exe로 어셈블리 등록 시 시스템 변경 사항 [5]파일 다운로드2
1288정성태5/17/201226461.NET Framework: 321. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (3) - Type Library파일 다운로드1
1287정성태5/17/201229294.NET Framework: 320. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (2) - .NET 4.0 + .NET 2.0 [2]
1286정성태5/17/201238216.NET Framework: 319. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (1) - .NET 2.0 + x86/x64/AnyCPU [5]
1285정성태5/16/201233264.NET Framework: 318. gacutil.exe로 어셈블리 등록 시 시스템 변경 사항파일 다운로드1
1284정성태5/15/201225690오류 유형: 155. Windows Phone 연결 상태에서 DRIVER POWER STATE FAILURE 블루 스크린 뜨는 현상
1283정성태5/12/201233306.NET Framework: 317. C# 관점에서의 Observer 패턴 구현 [1]파일 다운로드1
1282정성태5/12/201226104Phone: 6. Windows Phone 7 Silverlight에서 Google Map 사용하는 방법 [3]파일 다운로드1
1281정성태5/9/201233187.NET Framework: 316. WPF/Silverlight의 그래픽 단위와 Anti-aliasing 처리를 이해하자 [1]파일 다운로드1
1280정성태5/9/201226153오류 유형: 154. Could not load type 'System.ServiceModel.Activation.HttpModule' from assembly 'System.ServiceModel, ...'.
1279정성태5/9/201224915.NET Framework: 315. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 [1]파일 다운로드1
1278정성태5/8/201226145오류 유형: 153. Visual Studio 디버깅 - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...