Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 15개 있습니다.)
디버깅 기술: 22. VS.NET SP1 + .NET Framework 소스 코드 디버깅
; https://www.sysnet.pe.kr/2/0/623

개발 환경 구성: 112. Visual Studio 2010 - .NET Framework 소스 코드 디버깅
; https://www.sysnet.pe.kr/2/0/1009

디버깅 기술: 47. .NET Reflector를 이용한 "소스 코드가 없는" 어셈블리 디버깅
; https://www.sysnet.pe.kr/2/0/1201

개발 환경 구성: 143. Visual Studio 2010 - .NET Framework 소스 코드 디버깅 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1243

개발 환경 구성: 297. 소스 코드가 없는 닷넷 어셈블리를 디버깅할 때 지역 변숫값을 확인하는 방법
; https://www.sysnet.pe.kr/2/0/11036

개발 환경 구성: 391. (GitHub 등과 직접 연동해) 소스 코드 디버깅을 쉽게 해 주는 SourceLink
; https://www.sysnet.pe.kr/2/0/11630

VS.NET IDE: 126. 디컴파일된 소스에 탐색을 사용하도록 설정(Enable navigation to decompiled sources)
; https://www.sysnet.pe.kr/2/0/11689

VS.NET IDE: 145. NuGet + Github 라이브러리 디버깅 관련 옵션 3가지 - "Enable Just My Code" / "Enable Source Link support" / "Suppress JIT optimization on module load (Managed only)"
; https://www.sysnet.pe.kr/2/0/12200

VS.NET IDE: 149. ("Binary was not built with debug information" 상태로) 소스 코드 디버깅이 안되는 경우
; https://www.sysnet.pe.kr/2/0/12278

개발 환경 구성: 500. (PDB 연결이 없는) DLL의 소스 코드 디버깅을 dotPeek 도구로 해결하는 방법
; https://www.sysnet.pe.kr/2/0/12281

VS.NET IDE: 153. 닷넷 응용 프로그램에서의 "My Code" 범위와 "Enable Just My Code"의 역할
; https://www.sysnet.pe.kr/2/0/12401

VS.NET IDE: 170. Visual Studio에서 .NET Core/5+ 역어셈블 소스코드 확인하는 방법
; https://www.sysnet.pe.kr/2/0/12880

VS.NET IDE: 177. 비주얼 스튜디오 2022를 이용한 (소스 코드가 없는) 닷넷 모듈 디버깅 - "외부 원본(External Sources)"
; https://www.sysnet.pe.kr/2/0/13109

VS.NET IDE: 180. Visual Studio - 닷넷 소스 코드 디버깅 중 "Decompile source code"가 동작하는 않는 문제
; https://www.sysnet.pe.kr/2/0/13247

VS.NET IDE: 189. Visual Studio - 닷넷 소스코드 디컴파일 찾기가 안 될 때
; https://www.sysnet.pe.kr/2/0/13554




비주얼 스튜디오 2022를 이용한 (소스 코드가 없는) 닷넷 모듈 디버깅 - "외부 원본(External Sources)"

Visual Studio 2022에 추가된 "외부 원본(External Sources)" 기능을 아시나요? ^^

Debugging External Sources with Visual Studio
; https://devblogs.microsoft.com/visualstudio/debugging-external-sources-with-visual-studio/

그러니까 Source Link,

(GitHub 등과 직접 연동해) 소스 코드 디버깅을 쉽게 해 주는 SourceLink
; https://www.sysnet.pe.kr/2/0/11630

또는 (이제는 잘 사용하지 않겠지만) Source Server가 연동되면서,

TFS Team Build + Source Server = 소스 코드 디버깅
; https://www.sysnet.pe.kr/2/0/600

PDB 파일이 있다면 소스 코드 연동이 자동으로 되는 기능입니다. 마침 근래에 예제로 만들었던 KoreaBusInfo가,

C# - 버스 노선 및 위치 정보 조회 API 사용을 위한 기초 라이브러리
; https://www.sysnet.pe.kr/2/0/12780

SourceLink도 있고 nuget에 배포돼 있으니 예제로 적당할 듯합니다.

nuget - KoreaBusInfo 
; https://www.nuget.org/packages/KoreaBusInfo/

자, 그럼 한번 실습해 볼까요? ^^ 이를 위해 간단하게 다음의 코드를 만들고,

// Install-Package KoreaBusInfo

using KoreaBusInfo.Seoul;

namespace ConsoleApp1
{
    public class Program
    {
        public static void Main(string[] args)
        {
            string busNumber = "103";
            string key = "...[인증키(Encoding)]...";

            BusInfo sbi = new BusInfo(key);
            sbi.GetBusRoute(busNumber);
        }
    }
}

빌드한 후, 첫 번째 코드에 BP를 걸고 F5 디버깅을 시작합니다. 그럼 Solution Explorer에 다음과 같이 "External Sources" 노드가 보입니다.

vs_extdep_1.png

아니... 아무것도 없군요.

원본 정보를 포함하는 기호가 로드되지 않았습니다. (No symbols containing source information loaded.)

^^ 위에처럼 아무것도 안 보일 수도 있고, 어떤 분은 잘 보일 수도 있습니다. 그 이유는 아래에서 설명합니다.

이것을 해결하는 방법은 2가지입니다.

첫 번째로, 직접 로드할 것을 지시하면 됩니다. 위의 화면을 보면 "External Sources" 하위의 "Modules without sources"에 "KoreaBusInfo.dll"이 있습니다. 그것을 우클릭해 "Load Symbols" 메뉴를 선택하면 "PDB" 파일을 로드하면서 "External Sources" 하위에 위치하게 됩니다. (F5 디버깅 시마다 매번 해야 합니다.)

두 번째로, "Debug" / "Options"의 "Enable Just My Code" 옵션이 켜져 있기 때문에 저런 식으로 수동으로 직접 로드하게 된 것입니다. 그것을 끄고 다시 디버깅을 시작하면, 이번에는 자동으로 "External Sources" 하위에 KoreaBusInfo.dll이 위치하게 됩니다.

일단, 위와 같은 방식을 통하면 다음과 같이 Extenal Sources에 KoreaBusInfo.dll이 위치하게 됩니다.

vs_extdep_2.png

이제 원하는 C# 소스 코드 파일을 더블 클릭하면 github와 연동해 소스 코드 파일이 로드됩니다.




물론 BP를 연동한 디버깅도 가능합니다. 그런데, 디버깅의 경우에는 반드시 "Enable Just My Code" 옵션을 꺼야 가능합니다. (그러니까, 결국 위에서 2가지 방법을 제시했지만 현실적으로는 2번째 방법만이 해답입니다.)

만약 그 옵션을 끄지 않고 소스 코드 파일에 BP를 지정하면, BP에 경고 표시가 뜨고 마우스 커서를 올리면 다음과 같은 메시지를 보게 됩니다.

The breakpoint will not currently be hit. Breakpoints cannot be set in code that is optimized when the debugger option 'Just My Code' is enabled.


따라서 "Enable Just My Code" 옵션을 끄면 다음과 같이 BP도 잘 걸리고, Watch 창에 추가한 변숫값도 잘 확인이 됩니다.

vs_extdep_3.png




비록 nuget으로부터 참조하지는 않았지만 닷넷의 BCL 역시 nuget에 오픈소스로 공개돼 있고 SourceLink 연동이 돼 있는 상태입니다. 단지 PDB 파일만 없는 경우인데요, 따라서 pdb 파일만 해결되면 위에서 설명한 KoreaBusInfo 패키지와 동일하게 디버깅이 가능합니다.

물론 마이크로소프트는 닷넷의 BCL에 대한 심벌 파일을 제공하고 있는데요, 이를 위해 해야 할 일은 단순히 디버깅 중에 "Debug" / "Options" 창에서 "Symbols" 범주를 선택하면 보이는 "Load all symbols" 버튼을 누르는 것입니다.

그럼 비주얼 스튜디오는 가능한 모든 방법을 동원해 현재 프로세스에 로드된 모듈들의 pdb 심벌 파일을 찾게 되고, 결국 아래와 같이 "External Sources"에 pdb 연동이 된 닷넷의 모듈들을 함께 연동해 줍니다.

vs_extdep_4.png

보는 바와 같이 대부분의 .NET BCL은 "External Sources"에 올라와 소스 파일을 볼 수 있지만, 그 와중에 PDB 파일을 찾을 수 없었던 "Microsoft.Extensions.DotNetDeltaApplier.dll" 등의 모듈은 여전히 "Modules without sources"에 남아 있는 것을 볼 수 있습니다.




그럼 nuget에서 참조한 패키지가 pdb 파일을 제공하지 않는다면 어떻게 될까요? 일례로 GosperCombination 패키지는 pdb 파일 없이 dll 파일만 제공하고 SoruceLink 등의 연동은 하나도 안 돼 있습니다.

이걸로 간단하게 예제를 만들어 볼까요? ^^

// Install-Package GosperCombination

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Gosper;

namespace ConsoleApp1
{
    public class Program
    {
        public static void Main(string[] args)
        {
            string[] items = new string[] { "ant", "bug", "cat", "dog", "elk" };

            int choose = 3;
            Combination c = new Combination(items, choose);

            foreach (var elems in c.Successor())
            {
                Console.WriteLine(elems);
            }
        }
    }
}

당연히 위와 같은 프로젝트는 디버깅을 시작했을 때 "External Sources" 노드에 GosperCombination.dll이 "Modules without sources"에 위치하게 됩니다.

비록 github 등의 외부 Source Server와 연동은 안 되지만, 이런 경우 전통적인 Reflection을 통한 PDB 생성을 시켜 연동하는 것은 가능하므로, 해당 모듈을 마우스 우클릭해 나오는 메뉴에서 "Decompile Source to Symbol File"을 선택하면,

vs_extdep_5.png

해당 모듈은 다시 "External Sources"로 올라가고,

vs_extdep_6.png

Reflection 수준으로 풀어낸 소스 코드와 연동해 디버깅이 가능합니다.

vs_extdep_7.png

어떤가요? 이 정도면 제법 훌륭하죠? ^^ 그동안 작성했던 dll만 있는 모듈에 대한 소스 코드 연동 글들이,

Visual Studio 2010 - .NET Framework 소스 코드 디버깅
; https://www.sysnet.pe.kr/2/0/1009

Visual Studio 2010 - .NET Framework 소스 코드 디버깅 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1243

디컴파일된 소스에 탐색을 사용하도록 설정(Enable navigation to decompiled sources)
; https://www.sysnet.pe.kr/2/0/11689

.NET Reflector를 이용한 "소스 코드가 없는" 어셈블리 디버깅
; https://www.sysnet.pe.kr/2/0/1201

이제 "External Sources" 하나로 통합이 되면서 매우 쉽게 사용할 수 있게 된 것입니다. ^^




본문에서 KoreaBusInfo 패키지의 경우 "Load Symbols" 메뉴를 선택하면 "External Sources"에 (자동으로) 추가된다고 했는데요, 만약 그렇지 않고 PDB 파일을 지정하라는 파일 열기 대화창이 뜨는 경우가 있을 것입니다.

자동으로 PDB 파일에 대한 경로를 찾지 못한 것인데요, 그럴 때는 어쩔 수 없이 사용자가 직접 PDB 파일을 찾아서 지정해야 합니다. 가령 nuget으로부터 참조한 패키지의 경우 대부분은 pdb 파일도 함께 배포하기 마련인데요, KoreaBusInfo도 역시 다음과 같은 nuget 경로에,

// 기본 nuget 패키지 위치: %userprofile%\.nuget\packages
%USERPROFILE%\.nuget\packages\koreabusinfo\1.0.2\lib\netstandard2.1

// 또는 "NUGET_PACKAGES" 환경 변수를 재정의했다면,
// 예를 들어 c:\temp_root인 경우,
c:\temp_root\koreabusinfo\1.0.2\lib\netstandard2.1

KoreaBusInfo.pdb 파일이 있을 텐데, 그것을 지정하면 됩니다. 그렇게 하면, "Debug" / Options"의 창에서 "Debugging" / "Symbols" 범주를 보면 해당 디렉터리가 "Symbol file (.pdb) locations" 목록에 추가돼 이후 "Enable Just My Code" 옵션만 꺼져 있다면 자동으로 "External Sources"에 디버깅할 때마다 위치하게 됩니다.




참고로, "Debugging External Sources with Visual Studio" 글에는 보다 편리한 "External Sources" 연동을 위한 옵션 설정을 설명합니다. 편의를 위해 "Debug" / "Options" 창에서 "Symbols" 범주의 "Microsoft Symbol Server", "NuGet.org Symbol Server" 옵션을 켜두는 것인데... 저 같은 경우에는 예전부터 설정해 두고 쓰던 옵션입니다. ^^

그런데, 이게 좀 불편할 수 있는데요, 디버깅을 하면 해당 Symbol Server 목록으로부터 pdb 파일을 찾으려고 애쓰기 때문에 cache돼 있지 않은 pdb 파일을 요구하는 최초 EXE 디버깅 시 비주얼 스튜디오의 디버깅 세션 진입이 매우 오래 걸릴 수 있습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/27/2022]

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

비밀번호

댓글 작성자
 



2023-10-31 11시28분
정성태

1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...
NoWriterDateCnt.TitleFile(s)
13376정성태6/21/20236347.NET Framework: 2129. C# - Polly를 이용한 클라이언트 측의 요청 재시도파일 다운로드1
13375정성태6/20/20235881스크립트: 50. Transformers (신경망 언어모델 라이브러리) 강좌 - 2장 코드 실행 결과
13374정성태6/20/20236082오류 유형: 866. 파이썬 - <class 'AttributeError'> module 'flask.json' has no attribute 'JSONEncoder'
13373정성태6/19/20237425오류 유형: 865. 파이썬 - pymssql 설치 관련 오류 정리
13372정성태6/15/20235956개발 환경 구성: 682. SQL Server TLS 통신을 위해 사용되는 키 길이 확인 방법
13371정성태6/15/20236188개발 환경 구성: 681. openssl - 인증서 버전(V1 / V3)
13370정성태6/14/20236348개발 환경 구성: 680. C# - Ubuntu + Microsoft.Data.SqlClient + SQL Server 2008 R2 연결 방법 - TLS 1.2 지원
13369정성태6/13/20235902개발 환경 구성: 679. PyCharm(을 비롯해 JetBrains에 속한 여타) IDE에서 내부 Window들의 탭이 없어진 경우
13368정성태6/13/20236232개발 환경 구성: 678. openssl로 생성한 인증서를 SQL Server의 암호화 인증서로 설정하는 방법
13367정성태6/10/20236496오류 유형: 864. openssl로 만든 pfx 인증서를 Windows Server 2016 이하에서 등록 시 "The password you entered is incorrect" 오류 발생
13366정성태6/10/20236138.NET Framework: 2128. C# - 윈도우 시스템에서 지원하는 암호화 목록(Cipher Suites) 나열파일 다운로드1
13365정성태6/8/20235633오류 유형: 863. MODIFY FILE encountered operating system error 112(failed to retrieve text for this error. Reason: 15105)
13364정성태6/8/20236769.NET Framework: 2127. C# - Ubuntu + Microsoft.Data.SqlClient + SQL Server 2008 R2 연결 방법 [1]
13363정성태6/7/20236343스크립트: 49. 파이썬 - "Transformers (신경망 언어모델 라이브러리) 강좌" - 1장 2절 코드 실행 결과
13362정성태6/1/20236447.NET Framework: 2126. C# - 서버 측의 요청 제어 (Microsoft.AspNetCore.RateLimiting)파일 다운로드1
13361정성태5/31/20236747오류 유형: 862. Facebook - ASP.NET/WebClient 사용 시 graph.facebook.com/me 호출에 대해 403 Forbidden 오류
13360정성태5/31/20235928오류 유형: 861. WSL/docker - failed to start shim: start failed: io.containerd.runc.v2: create new shim socket
13359정성태5/19/20236181오류 유형: 860. Docker Desktop - k8s 초기화 무한 반복한다면?
13358정성태5/17/20236765.NET Framework: 2125. C# - Semantic Kernel의 Semantic Memory 사용 예제 [1]파일 다운로드1
13357정성태5/16/20236629.NET Framework: 2124. C# - Semantic Kernel의 Planner 사용 예제파일 다운로드1
13356정성태5/15/20237012DDK: 10. Device Driver 테스트 설치 관련 오류 (Code 37, Code 31) 및 인증서 관련 정리
13355정성태5/12/20237052.NET Framework: 2123. C# - Semantic Kernel의 ChatGPT 대화 구현 [1]파일 다운로드1
13354정성태5/12/20237059.NET Framework: 2122. C# - "Use Unicode UTF-8 for worldwide language support" 설정을 한 경우, 한글 입력이 '\0' 문자로 처리
13352정성태5/12/20236860.NET Framework: 2121. C# - Semantic Kernel의 대화 문맥 유지파일 다운로드1
13351정성태5/11/20237344VS.NET IDE: 185. Visual Studio - 원격 Docker container 내에 실행 중인 응용 프로그램에 대한 디버깅 [1]
13350정성태5/11/20236549오류 유형: 859. Windows Date and Time - Unable to continue. You do not have permission to perform this task
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...