Microsoft MVP성태의 닷넷 이야기
닷넷: 2266. C# - (Reflection 없이) DLL AssemblyFileVersion 구하는 방법 [링크 복사], [링크+제목 복사],
조회: 8518
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 3개 있습니다.)
.NET Framework: 181. AssemblyVersion, AssemblyFileVersion, AssemblyInformationalVersion
; https://www.sysnet.pe.kr/2/0/897

닷넷: 2266. C# - (Reflection 없이) DLL AssemblyFileVersion 구하는 방법
; https://www.sysnet.pe.kr/2/0/13651

닷넷: 2267. C# - Linux 환경에서 (Reflection 없이) DLL AssemblyFileVersion 구하는 방법
; https://www.sysnet.pe.kr/2/0/13652




C# - (Reflection 없이) DLL AssemblyFileVersion 구하는 방법

(AssmeblyVersion이 아닌) AssemblyFileVersion은 Windows의 PE(Portable Executables) 포맷에 따라 Resource Section에 저장된 정보입니다.

따라서, 이 정보를 읽는 방법은 2가지인데요, 1) Reflection으로 Attribute를 읽거나 2) Win32 API를 이용해 접근하는 방법이 있습니다.

우선, Reflection을 사용해 볼까요? .NET Core/5+부터 AppDomain에 대한 생성이 제한되면서 Reflection을 위해 함부로 로딩할 수 없게 되었습니다. 이로 인해, Assembly로써 DLL을 Load하는 방식이 아닌, 파일로부터 직접 메타데이터 정보를 읽어 오는 방식을 새롭게 제공하고 있는데, 바로 System.Reflection.MetadataLoadContext 패키지가 그것입니다.

예를 들어, Newtonsoft.Json 패키지의 Newtonsoft.Json.dll을 대상으로 MetadataLoadContext를 이용해 File Version을 가져오는 것은 대충 다음과 같이 코딩할 수 있습니다.

using System.Reflection;

namespace ConsoleApp2;

// Install-Package System.Reflection.MetadataLoadContext
internal class Program
{
    static void Main(string[] args)
    {
        string dllPath = Path.Combine(".", "DLLs", "net6.0", "Newtonsoft.Json.dll");

        if (File.Exists(dllPath) == false)
        {
            Console.WriteLine("DLL not found");
            return;
        }

        var coreAssemblies = new List<string>();
        var resolver = new PathAssemblyResolver(coreAssemblies);
        using var mlc = new MetadataLoadContext(resolver);

        var asm = mlc.LoadFromAssemblyPath(dllPath);
        Console.WriteLine(GetFileVersion(asm));
    }

    static readonly Version _emptyVersion = new Version();

    private static Version GetFileVersion(Assembly asm)
    {
        foreach (CustomAttributeData data in asm.CustomAttributes)
        {
            try
            {
                if (data.AttributeType.FullName == typeof(AssemblyFileVersionAttribute).FullName)
                {
                    if (data.ConstructorArguments.Count == 1)
                    {
                        string versionText = data.ConstructorArguments[0].Value as string ?? "";
                        return Version.Parse(versionText);
                    }
                }
            }
            catch
            {
            }
        }

        return _emptyVersion;
    }
}

그런데, 실제로 저렇게만 작성하면 실행 시 이런 예외가 발생합니다.

Unhandled exception. System.IO.FileNotFoundException: Could not find core assembly. Either specify a valid core assembly name in the MetadataLoadContext constructor or provide a MetadataAssemblyResolver that can load the core assembly.
   at System.Reflection.TypeLoading.CoreTypes..ctor(MetadataLoadContext loader, String coreAssemblyName)
   at System.Reflection.MetadataLoadContext..ctor(MetadataAssemblyResolver resolver, String coreAssemblyName)
   at ConsoleApp2.Program.Main(String[] args)

왜냐하면, Newtonsoft.Json.dll은 다음의 많은 DLL을 참조하고 있기 때문에,

// System.Collections, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Collections.Concurrent, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.ComponentModel.Primitives, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.ComponentModel.TypeConverter, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Data.Common, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Diagnostics.TraceSource, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Linq, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Linq.Expressions, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.ObjectModel, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Reflection.Emit.ILGeneration, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Reflection.Emit.Lightweight, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Reflection.Primitives, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Runtime, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Runtime.InteropServices, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Runtime.Numerics, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Runtime.Serialization.Formatters, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Runtime.Serialization.Primitives, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Text.Encoding.Extensions, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Text.RegularExpressions, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Threading, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Xml.ReaderWriter, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
// System.Xml.XDocument, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a

(대체로 모든 DLL들이 필요하진 않지만) 이에 대한 위치도 알려야 하기 때문입니다. 쉬운 상황이라면, 가령 대상 DLL이 현재 실행 중인 응용 프로그램의 런타임과 같다면 간단하게 coreAssemblies 목록을 이렇게 전달할 수 있습니다.

string runtimeDir = RuntimeEnvironment.GetRuntimeDirectory();

var coreAssemblies = Directory.GetFiles(runtimeDir, "*.dll").ToList();

var resolver = new PathAssemblyResolver(coreAssemblies);
using var mlc = new MetadataLoadContext(resolver);

// ...[생략]...

하지만, 만약 특정 .NET 런타임의 DLL을 참조하고 있다면 상황이 매우 불편해집니다. 왜냐하면, 대상 DLL이 의존하는 닷넷 런타임을 미리 알고 있어야 하는데, 만약 그렇지 않다면 해당 DLL을 Reflection으로 열어 보기 전에 판단할 수 없기 때문입니다. 결국 닭이 먼저냐, 계란이 먼저냐 하는 상황이 발생합니다.




다행히 두 번째 방법을 사용하면 이런 문제를 회피할 수 있습니다. 대신 약간 현란한 Win32 API를 사용해야 하는데, unsafe 구문 사용 여부에 따라 2가지 버전으로 나뉠 수 있습니다.

우선, 아래는 unsafe를 사용하지 않았을 때의 방법입니다.

using System.Runtime.InteropServices;

namespace ConsoleApp1;

internal class Program
{
    [DllImport("version.dll", CharSet = CharSet.Auto, SetLastError = true)]
    static extern uint GetFileVersionInfoSize(string lptstrFilename, IntPtr lpdwHandle);

    [DllImport("version.dll", CharSet = CharSet.Auto, SetLastError = true)]
    static extern bool GetFileVersionInfo(string lptstrFilename, uint dwHandle, uint dwLen, IntPtr lpData);

    [DllImport("version.dll", CharSet = CharSet.Auto, SetLastError = true, EntryPoint = "VerQueryValue")]
    static extern unsafe bool VerQueryValue(IntPtr pBlock, string lpSubBlock, out VS_FIXEDFILEINFO* fileInfo, out uint puLen);

    static void Main(string[] args)
    {
        string path = @"Newtonsoft.Json.dll";

        uint size = GetFileVersionInfoSize(path, IntPtr.Zero);

        IntPtr data = Marshal.AllocHGlobal((int)size);
        bool result = GetFileVersionInfo(path, 0, size, data);
        uint puLen;

        if (result == true)
        {
            IntPtr pBuffer = IntPtr.Zero;
            if (VerQueryValue(data, "\\", out pBuffer, out puLen))
            {
                VS_FIXEDFILEINFO fileInfo = Marshal.PtrToStructure<VS_FIXEDFILEINFO>(pBuffer);
                string version = $"{fileInfo.dwFileVersionMS >> 16}.{fileInfo.dwFileVersionMS & 0xffff}.{fileInfo.dwFileVersionLS >> 16}.{fileInfo.dwFileVersionLS & 0xffff}";
                Console.WriteLine($"{version}");
            }
        }

        if (data != IntPtr.Zero)
        {
            Marshal.FreeHGlobal(data);
        }
    }
}

[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct VS_FIXEDFILEINFO
{
    public uint dwSignature;
    public uint dwStrucVersion;
    public uint dwFileVersionMS;
    public uint dwFileVersionLS;
    public uint dwProductVersionMS;
    public uint dwProductVersionLS;
    public uint dwFileFlagsMask;
    public uint dwFileFlags;
    public uint dwFileOS;
    public uint dwFileType;
    public uint dwFileSubtype;
    public uint dwFileDateMS;
    public uint dwFileDateLS;
}

보는 바와 같이 일단 포인터로 받아온 뒤, 그것을 Marshal.PtrToStructure를 이용해 구조체로 값을 채우는 과정을 거치는데요, 물론 이 과정에서 미미한 성능 이슈가 발생합니다.

만약 그런 것조차 허용하고 싶지 않다면 unsafe 버전을 사용할 수 있습니다.

[DllImport("version.dll", CharSet = CharSet.Auto, SetLastError = true, EntryPoint = "VerQueryValue")]
static extern unsafe bool VerQueryValueUnsafe(IntPtr pBlock, string lpSubBlock, out VS_FIXEDFILEINFO* fileInfo, out uint puLen);

// ...[생략]...

{
    VS_FIXEDFILEINFO* pFileInfo;
    if (VerQueryValueUnsafe(data, "\\", out pFileInfo, out puLen))
    {
        string version = $"{pFileInfo->dwFileVersionMS >> 16}.{pFileInfo->dwFileVersionMS & 0xffff}.{pFileInfo->dwFileVersionLS >> 16}.{pFileInfo->dwFileVersionLS & 0xffff}";
        Console.WriteLine($"{version}");
    }
}

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




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







[최초 등록일: ]
[최종 수정일: 6/24/2024]

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

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  69  70  71  [72]  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12134정성태2/5/202020920.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지 [5]파일 다운로드1
12133정성태2/5/202018338디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
12132정성태1/28/202021193.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기) [1]파일 다운로드1
12131정성태1/27/202020161개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/202017467VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202023547.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202017925오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
12127정성태1/25/202017122.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)파일 다운로드1
12126정성태1/25/202018492.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/202015972VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202017750VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202019461웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/202015585.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/202016301VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202018708.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202018911디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202019912개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202018744디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202019388디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202019630디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202021442디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202021517오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/202016705오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202020477디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
12110정성태1/11/202019790디버깅 기술: 154. Patch Guard로 인해 블루 스크린(BSOD)가 발생하는 사례 [5]파일 다운로드1
12109정성태1/10/202016562오류 유형: 588. Driver 프로젝트 빌드 오류 - Inf2Cat error -2: "Inf2Cat, signability test failed."
... 61  62  63  64  65  66  67  68  69  70  71  [72]  73  74  75  ...