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

.NET Core/5+에서 동적 컴파일한 C# 코드를 (Breakpoint도 활용하며) 디버깅하는 방법 - #line 지시자

지난 글에서,

.NET Core/5+에서 C# 코드를 동적으로 컴파일/사용하는 방법
; https://www.sysnet.pe.kr/2/0/12809

C# 소스 코드를 동적으로 컴파일 및 사용하는 방법을 다뤘는데요, 사실 이에 대한 디버깅도 가능합니다. 어떻게 할 수 있는지 한번 다뤄볼까요? ^^

지난 글의 소스 코드를 활용해 바꿔볼 텐데요, 우선 C# 코드를 별도의 파일로 빼내야 합니다. 테스트를 위해 "dynamic_code.txt"라는 파일로 동적 코드를 빼내고,

string codeToCompile = File.ReadAllText("dynamic_code.txt");

다음과 같이, 해당 파일은 "Build Action: None", "Copy to Output Directory: Copy if newer" 옵션을 설정해 줍니다. (위의 경우에는 "Build Action" 설정은 필요 없지만, 원래 C# 코드를 담은 파일의 확장자를 .cs로 주는 것이 일반적이기 때문에 그런 때를 위해 명시하는 것이 좋습니다.)

cs_dynamic_compile_debug_1.png

그리고, 디버깅이 가능하려면 debug 모드로 빌드해야 하므로 옵션을 조정하고,

CSharpCompilationOptions options = new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary
                , optimizationLevel: OptimizationLevel.Debug);

동적 빌드된 어셈블리의 바이너리뿐만 아니라 PDB 정보까지도 출력하도록 바꾸고, 사용할 때도 Symbol 정보를 로드하도록 바꿉니다.

using (var symbolsStream = new MemoryStream())
using (var ms = new MemoryStream())
{
    EmitResult result = compilation.Emit(ms, pdbStream: symbolsStream);

    if (result.Success)
    {
        ms.Seek(0, SeekOrigin.Begin);
        symbolsStream?.Seek(0, SeekOrigin.Begin);

        Assembly assembly = AssemblyLoadContext.Default.LoadFromStream(ms, symbolsStream);
        var type = assembly.GetType("MyType");
        var instance = assembly.CreateInstance("MyType");
        var meth = type.GetMember("Print").First() as MethodInfo;
        meth.Invoke(instance, new[] { "World" });
    }
}

혹은 새로운 옵션인 embedded를 사용해,

닷넷 응용 프로그램을 위한 PDB 옵션 - full, pdbonly, portable, embedded
; https://www.sysnet.pe.kr/2/0/12554

다음과 같이 코드를 좀 더 간략하게 만들 수 있습니다.

using (var ms = new MemoryStream())
{
    var emitOptions = new EmitOptions(debugInformationFormat: DebugInformationFormat.Embedded);

    EmitResult result = compilation.Emit(ms,
        options: emitOptions);

    if (result.Success)
    {
        ms.Seek(0, SeekOrigin.Begin);

        Assembly assembly = AssemblyLoadContext.Default.LoadFromStream(ms);
        var type = assembly.GetType("MyType");
        var instance = assembly.CreateInstance("MyType");
        var meth = type.GetMember("Print").First() as MethodInfo;
        meth.Invoke(instance, new[] { "World" });
    }
}

기본적인 준비는 여기까지가 끝입니다.




자, 이제 마지막 남은 것이 하나 있는데요, 바로 비주얼 스튜디오에게 동적으로 컴파일한 MyType.Print 메서드가 불렸을 때 화면에 어떤 소스 코드 파일을 로드해 보여줄 것인지를 결정하는 것입니다.

그리고 바로 그런 역할을 하는 것이 #line 지시자입니다. 따라서 동적 소스 코드를 포함한 dynamic_code.txt 파일에 다음의 정보를 포함하고,

 1 
 2 #line 3 "dynamic_code.txt"
 3 
 4 using System;
 5 using System.Text;
 6 
 7 public class MyType
 8 {
 9     public void Print(object obj)
10     {
11         StringBuilder sb = new StringBuilder();
12         sb.Append(DateTime.Now);
13 
14         Console.WriteLine("Hello: " + obj + " : " + sb.ToString());
15     }
16 }

빌드하면, 이제 C# 컴파일러는 해당 소스 코드가 "dynamic_code.txt" 파일에 있으며 기준점으로 3번째 라인에 매핑시켜야 한다는 것을 알 수 있습니다.

따라서, 이제 위의 소스 코드에 Breakpoint를 걸 수도 있고 실제로 디버그 모드로 실행하면 다음과 같이 BP도 잡히고 심지어 Watch 창에서 변숫값도 확인할 수 있습니다.

cs_dynamic_compile_debug_2.png

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




한 가지 가정을 해볼까요? 만약 제가 Razor 템플릿 개발자이고 다음과 같은 코드 양식을 사용했을 때,

@page
@{
    var name = string.Empty;
    if (Request.HasFormContentType)
    {
        name = Request.Form["name"];
    }
}
<div style="margin-top:30px;">
    <form method="post">
        <div>
            Name: <input name="name" />
        </div>
        <div>
            <input type="submit" />
        </div>
    </form>
</div>
<div>
    @if (!string.IsNullOrEmpty(name))
    {
    <p>Hello @name!</p>
    }
</div>

어떻게 저 razor 파일 내에 있는 C# 코드에 BP를 걸게 만들 수 있었을까요? ^^ 그렇습니다. 자동 생성된 코드에서 저 파일을 지정하도록 하면 되는 것입니다.

// sample.cshtml.g.cs

#line 3 "sample.cshtml"  // 위의 파일 이름이 sample.cshtml라고 가정
using System;
using System.Text;

// razor 파일을 C#으로 다루는 코드




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

[연관 글]






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

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

비밀번호

댓글 작성자
 




... 76  77  78  79  80  81  82  [83]  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11862정성태4/7/201920266개발 환경 구성: 437. .NET EXE의 ASLR 기능을 끄는 방법
11861정성태4/6/201919633디버깅 기술: 126. windbg - .NET x86 CLR2/CLR4 EXE의 EntryPoint
11860정성태4/5/201923555오류 유형: 527. Visual C++ 컴파일 오류 - error C2220: warning treated as error - no 'object' file generated
11859정성태4/4/201920827디버깅 기술: 125. WinDbg로 EXE의 EntryPoint에서 BP 거는 방법
11858정성태3/27/201921682VC++: 129. EXE를 LoadLibrary로 로딩해 PE 헤더에 있는 EntryPoint를 직접 호출하는 방법파일 다운로드1
11857정성태3/26/201919558VC++: 128. strncpy 사용 시 주의 사항(Linux / Windows)
11856정성태3/25/201919788VS.NET IDE: 134. 마이크로소프트의 CoreCLR 프로파일러 리눅스 예제를 Visual Studio F5 원격 디버깅하는 방법 [1]파일 다운로드1
11855정성태3/25/201921981개발 환경 구성: 436. 페이스북 HTTPS 인증을 localhost에서 테스트하는 방법
11854정성태3/25/201917683VS.NET IDE: 133. IIS Express로 호스팅하는 사이트를 https로 접근하는 방법
11853정성태3/24/201920437개발 환경 구성: 435. 존재하지 않는 IP 주소에 대한 Dns.GetHostByAddress/gethostbyaddr/GetNameInfoW 실행이 느리다면? - 두 번째 이야기 [1]
11852정성태3/20/201919636개발 환경 구성: 434. 존재하지 않는 IP 주소에 대한 Dns.GetHostByAddress/gethostbyaddr/GetNameInfoW 실행이 느리다면?파일 다운로드1
11851정성태3/19/201923391Linux: 8. C# - 리눅스 환경에서 DllImport 대신 라이브러리 동적 로드 처리 [2]
11850정성태3/18/201922477.NET Framework: 813. C# async 메서드에서 out/ref/in 유형의 인자를 사용하지 못하는 이유
11849정성태3/18/201921790.NET Framework: 812. pscp.exe 기능을 C#으로 제어하는 방법파일 다운로드1
11848정성태3/17/201918561스크립트: 14. 윈도우 CMD - 파일이 변경된 경우 파일명을 변경해 복사하고 싶다면?
11847정성태3/17/201923024Linux: 7. 리눅스 C/C++ - 공유 라이브러리 동적 로딩 후 export 함수 사용 방법파일 다운로드1
11846정성태3/15/201921676Linux: 6. getenv, setenv가 언어/운영체제마다 호환이 안 되는 문제
11845정성태3/15/201921780Linux: 5. Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency) [3]
11844정성태3/14/201923113개발 환경 구성: 434. Visual Studio 2019 - 리눅스 프로젝트를 이용한 공유/실행(so/out) 프로그램 개발 환경 설정 [1]파일 다운로드1
11843정성태3/14/201918045기타: 75. MSDN 웹 사이트를 기본으로 영문 페이지로 열고 싶다면?
11842정성태3/13/201916419개발 환경 구성: 433. 마이크로소프트의 CoreCLR 프로파일러 예제를 Visual Studio CMake로 빌드하는 방법 [1]파일 다운로드1
11841정성태3/13/201916704VS.NET IDE: 132. Visual Studio 2019 - CMake의 컴파일러를 기본 g++에서 clang++로 변경
11840정성태3/13/201918336오류 유형: 526. 윈도우 10 Ubuntu App 환경에서는 USB 외장 하드 접근 불가
11839정성태3/12/201922332디버깅 기술: 124. .NET Core 웹 앱을 호스팅하는 Azure App Services의 프로세스 메모리 덤프 및 windbg 분석 개요 [3]
11838정성태3/7/201925925.NET Framework: 811. (번역글) .NET Internals Cookbook Part 1 - Exceptions, filters and corrupted processes [1]파일 다운로드1
11837정성태3/6/201939847기타: 74. 도서: 시작하세요! C# 7.3 프로그래밍 [10]
... 76  77  78  79  80  81  82  [83]  84  85  86  87  88  89  90  ...