Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 5개 있습니다.)
.NET Framework: 903. .NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션
; https://www.sysnet.pe.kr/2/0/12210

.NET Framework: 928. .NET Framework의 Strong-named 어셈블리 바인딩 (2) - 런타임에 바인딩 리디렉션
; https://www.sysnet.pe.kr/2/0/12271

.NET Framework: 929. (StrongName의 버전 구분이 필요 없는) .NET Core 어셈블리 바인딩 규칙
; https://www.sysnet.pe.kr/2/0/12272

.NET Framework: 930. 개발자를 위한 닷넷 어셈블리 바인딩 - DEVPATH 환경 변수
; https://www.sysnet.pe.kr/2/0/12276

개발 환경 구성: 498. DEVPATH 환경 변수의 사용 예 - .NET Reflector의 (PDB 연결이 없는) DLL의 소스 코드 디버깅
; https://www.sysnet.pe.kr/2/0/12277




.NET Framework의 Strong-named 어셈블리 바인딩 (2) - 런타임에 바인딩 리디렉션

지난 글에서,

.NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션
; https://www.sysnet.pe.kr/2/0/12210

어셈블리의 버전 불일치에 대한 해결책을 app.config을 이용해 우회했는데요. 이것을 런타임에 AppDomain.CurrentDomain.AssemblyResolve 이벤트를 이용해 개발자가 직접 제어하는 것도 가능합니다.

AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;

private static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
    AssemblyName asmName = new AssemblyName(args.Name);

    string path = Path.GetDirectoryName(typeof(Program).Assembly.Location);
    string dllPath = Path.Combine(path, $"{asmName.Name}.dll");

    Console.WriteLine("Resolving path: " + dllPath);
    return Assembly.LoadFile(dllPath);
}

AssemblyResolve 이벤트는 대개 특정 어셈블리를 로딩하지 못했을 때 개발자가 직접 로드하는 식으로 구현하는데요, 그 구현의 특성상 버전을 무시한 로딩을 하는 것도 가능합니다. 예를 들어, 아래의 코드는 "Version=1.0.0.0", "Version=1.5.0.0"의 ClassLibrary1.dll 요청에 대해 동일하게 2.0.0.0 파일로 처리를 합니다.

// ClassLibrary1, Version=2.0.0.0
using System;

namespace ClassLibrary1
{
    public class Class1
    {
        public static int Version = 5;

        public void Test()
        {
            Console.WriteLine(ClassLibrary1.Class1.Version ++);
        }
    }
}

// ConsoleApp1
using System;
using System.IO;
using System.Reflection;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;

            // new ClassLibrary1.Class1().Test();
            LoadAtRuntime("Version=1.0.0.0");
            LoadAtRuntime("Version=1.5.0.0");
        }

        private static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
        {
            AssemblyName asmName = new AssemblyName(args.Name);

            string path = Path.GetDirectoryName(typeof(Program).Assembly.Location);
            string dllPath = Path.Combine(path, $"{asmName.Name}.dll");

            Console.WriteLine("Resolving path: " + dllPath);
            return Assembly.LoadFile(dllPath);
        }

        private static void LoadAtRuntime(string versionText)
        {
            string clName = $"ClassLibrary1, {versionText}, Culture=neutral, PublicKeyToken=0086c02b325d69fe";
            Assembly asm = Assembly.Load(clName);
            Console.WriteLine(asm.FullName);

            object objInstance = Activator.CreateInstance(asm.GetType("ClassLibrary1.Class1"));
            dynamic clInst = objInstance;
            clInst.Test();
        }
    }
}

실행해 보면, 동일한 DLL로 처리되었기 때문에 ClassLibrary1의 static 변수의 값이 바뀌는 것을 확인할 수 있습니다.

Resolving path: C:\temp\ConsoleApp1\bin\Debug\ClassLibrary1.dll
ClassLibrary1, Version=2.0.0.0, Culture=neutral, PublicKeyToken=0086c02b325d69fe
5
Resolving path: C:\temp\ConsoleApp1\bin\Debug\ClassLibrary1.dll
ClassLibrary1, Version=2.0.0.0, Culture=neutral, PublicKeyToken=0086c02b325d69fe
6




유의할 점이 있다면, Assembly 정보를 byte []로 로드해 반환하는 경우,

private static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
    AssemblyName asmName = new AssemblyName(args.Name);

    string path = Path.GetDirectoryName(typeof(Program).Assembly.Location);
    string dllPath = Path.Combine(path, $"{asmName.Name}.dll");

    Console.WriteLine("Resolving path: " + dllPath);

    byte[] buf = File.ReadAllBytes(dllPath);
    return Assembly.Load(buf);
}

이를 구분할 수 있는 파일 Identity가 없는 Assembly이기 때문에 매번 반환하는 Assembly를 다르게 취급한다는 점입니다. 실제로 위와 같이 File.ReadAllBytes + Assembly.Load로 처리하는 경우 다음과 같이 "Version"의 값이 개별적으로 초기화되는 것을 볼 수 있습니다.

Resolving path: C:\temp\ConsoleApp1\bin\Debug\ClassLibrary1.dll
ClassLibrary1, Version=2.0.0.0, Culture=neutral, PublicKeyToken=0086c02b325d69fe
5
Resolving path: C:\temp\ConsoleApp1\bin\Debug\ClassLibrary1.dll
ClassLibrary1, Version=2.0.0.0, Culture=neutral, PublicKeyToken=0086c02b325d69fe
5

따라서, byte[]로 다뤄야 하는 어셈블리가 있다면 AssemblyLoad 이벤트에서는 기존에 로드된 어셈블리를 찾아보는 작업을 추가해야 합니다.

private static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{
    AssemblyName asmName = new AssemblyName(args.Name);

    Assembly everLoaded = EverLoaded(asmName);
    if (everLoaded != null)
    {
        return everLoaded;
    }

    string path = Path.GetDirectoryName(typeof(Program).Assembly.Location);
    string dllPath = Path.Combine(path, $"{asmName.Name}.dll");

    Console.WriteLine("Resolving path: " + dllPath);

    byte[] buf = File.ReadAllBytes(dllPath);
    return Assembly.Load(buf);
}

static Assembly EverLoaded(AssemblyName asmName)
{
    foreach (Assembly asm in AppDomain.CurrentDomain.GetAssemblies())
    {
        AssemblyName targetName = asm.GetName();
        if (targetName.Name == asmName.Name &&
            targetName.GetPublicKeyToken().SequenceEqual(asmName.GetPublicKeyToken()) == true)
        {
            return asm;
        }
    }

    return null;
}

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




이 외에도 다음의 글을 보면,

How the Runtime Locates Assemblies
; https://learn.microsoft.com/en-us/dotnet/framework/deployment/how-the-runtime-locates-assemblies

바인딩에 관한 여러 가지 방법이 나옵니다. 그중에서 게시자 정책 파일(Publisher Policy File)의 경우,

How the Runtime Locates Assemblies - Publisher Policy File
; https://learn.microsoft.com/en-us/dotnet/framework/deployment/how-the-runtime-locates-assemblies#publisher-policy-file

Introduction to Publisher Policy File
; https://www.c-sharpcorner.com/UploadFile/satisharveti/introduction-to-publisher-policy-file/

해당 어셈블리를 사용하는 측에서 바인딩을 변경하는 것이 아니고, 어셈블리를 제공하는 측에서 바인딩을 변경하는 전용 어셈블리를 함께 배포하는 식입니다. 예를 들어, 기존의 app.config에서 했던 것처럼 바인딩을 우회하는 config 파일을 만든 후,

<?xml version="1.0" encoding="utf-8"?>
<configuration>
    <runtime>
        <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
            <dependentAssembly>
                <assemblyIdentity name="ClassLibrary1" publicKeyToken="0086c02b325d69fe" culture="neutral" />
                <bindingRedirect oldVersion="0.0.0.0-2.0.0.0" newVersion="2.0.0.0" />
            </dependentAssembly>
    </runtime>
</configuration>

al.exe를 이용해 다음과 같은 식으로 config의 내용을 담은 DLL로 변환합니다.

Al.exe /link:ClassLibrary1.dll.config /out:policy.1.0.ClassLibrary1.dll /keyfile:..\..\my.key /v:1.0.0.0

이 과정에서 특이하게 어셈블리 서명에 사용된 키 파일을 사용하는데, 따라서 원 저작자를 제외하고는 게시자 정책 파일을 만들 수 없다는 차별점이 있습니다. 그러니까, 특정 어셈블리를 개발한 측에서 그것의 업데이트를 배포할 때 기존 프로그램들이 새롭게 업데이트된 어셈블리를 로드하도록 만들고 싶을 때 사용할 수 있는 방법입니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 10/21/2022]

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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...
NoWriterDateCnt.TitleFile(s)
10949정성태4/28/201619881.NET Framework: 575. SharedDomain과 JIT 컴파일파일 다운로드1
10948정성태4/28/201623826.NET Framework: 574. .NET - 눈으로 확인하는 SharedDomain의 동작 방식 [3]파일 다운로드1
10947정성태4/27/201621695.NET Framework: 573. .NET CLR4 보안 모델 - 4. CLR4 보안 모델에서의 조건부 APTCA 역할파일 다운로드1
10946정성태4/26/201624513VS.NET IDE: 106. Visual Studio 2015 확장 - INI 파일을 위한 사용자 정의 포맷 기능 (Syntax Highlighting)파일 다운로드1
10945정성태4/26/201618280오류 유형: 327. VSIX 프로젝트 빌드 시 The "VsTemplatePaths" task could not be loaded from the assembly 오류 발생
10944정성태4/22/201619516디버깅 기술: 80. windbg - 풀 덤프 파일로부터 텍스트 파일의 내용을 찾는 방법
10943정성태4/22/201624376디버깅 기술: 79. windbg - 풀 덤프 파일로부터 .NET DLL을 추출/저장하는 방법 [1]
10942정성태4/19/201619673디버깅 기술: 78. windbg 사례 - .NET 예외가 발생한 시점의 오류 분석 [1]
10941정성태4/19/201619588오류 유형: 326. Error MSB8020 - The build tools for v120_xp (Platform Toolset = 'v120_xp') cannot be found.
10940정성태4/18/201622856Windows: 116. 프로세스 풀 덤프 시간을 줄여 주는 Process Reflection [3]
10939정성태4/18/201623882.NET Framework: 572. .NET APM 비동기 호출의 Begin...과 End... 조합 [3]파일 다운로드1
10938정성태4/13/201623447오류 유형: 325. 파일 삭제 시 오류 - Error 0x80070091: The directory is not empty.
10937정성태4/13/201631672Windows: 115. UEFI 모드로 윈도우 10 설치 가능한 USB 디스크 만드는 방법
10936정성태4/8/201642357Windows: 114. 삼성 센스 크로노스 7 노트북의 운영체제를 USB 디스크로 새로 설치하는 방법 [3]
10935정성태4/7/201626655웹: 32. Edge에서 Google Docs 문서 편집 시 한영 전환키가 동작 안하는 문제
10934정성태4/5/201625379디버깅 기술: 77. windbg의 콜스택 함수 인자를 쉽게 확인하는 방법 [1]
10933정성태4/5/201630992.NET Framework: 571. C# - 스레드 선호도(Thread Affinity) 지정하는 방법 [8]파일 다운로드1
10932정성태4/4/201623282VC++: 96. C/C++ 식 평가 - printf("%d %d %d\n", a, a++, a);
10931정성태3/31/201623559개발 환경 구성: 283. Hyper-V 내에 구성한 Active Directory 환경의 시간 구성 방법 [3]
10930정성태3/30/201621513.NET Framework: 570. .NET 4.5부터 추가된 CLR Profiler의 실행 시 Rejit 기능
10929정성태3/29/201631626.NET Framework: 569. ServicePointManager.DefaultConnectionLimit의 역할파일 다운로드1
10928정성태3/28/201637338.NET Framework: 568. ODP.NET의 완전한 닷넷 버전 Oracle ODP.NET, Managed Driver [2]파일 다운로드1
10927정성태3/25/201626545.NET Framework: 567. System.Net.ServicePointManager의 DefaultConnectionLimit 속성 설명
10926정성태3/24/201626087.NET Framework: 566. openssl의 PKCS#1 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 [10]파일 다운로드1
10925정성태3/24/201620387.NET Framework: 565. C# - Rabin-Miller 소수 생성 방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 - 두 번째 이야기파일 다운로드1
10924정성태3/22/201621039오류 유형: 324. Visual Studio에서 Azure 클라우드 서비스 생성 시 Failed to initialize the PowerShell host 에러 발생
... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...