Microsoft MVP성태의 닷넷 이야기
VS.NET IDE: 55. XML/XSLT로 구현하는 매크로 확장 [링크 복사], [링크+제목 복사],
조회: 40307
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 5개 있습니다.)


XML/XSLT로 구현하는 매크로 확장


C#으로 프로그래밍을 하면서 과거 C/C++ 프로그램과 비교했을 때 아쉬운 점이 하나 있더군요. 바로 매크로 확장 기능이었습니다.

예를 들어 설명해 보면, C/C++에서는 다음과 같이 매크로를 정의해서 사용하는 것이 가능합니다.

#define IF_DOIT(variable) if (condition == variable) funcName_##variable();

int condition = GetCondition();

IF_DOIT(1)
IF_DOIT(2)
IF_DOIT(3)
IF_DOIT(4)
IF_DOIT(5)

나름대로 장점이 있지요. 반복되는 잡다한 코드를 프로그래머에게 숨겨주면서 사소한 오류들을 줄여줄 수가 있고, 코드 수정도 상당히 용이합니다. MFC의 Win32 Message 관련한 매크로가 대표적인 예이지요.

그런데, C#에서는 위와 같은 기능이 없습니다. 그래서, 매번 반복적인 코드를 작성해야 할 때는 오타가 나서 컴파일 오류가 날때도 있고, 반복적인 소스코드로 인해 스크롤을 하다 보면 자신이 찾고 싶은 코드가 어디있는지 발견하기도 힘듭니다.

물론, 나름대로 개발도구상에서 Code Snippets 기능을 제공하는 식으로 해결해 보려고는 하지만,,, 그래도 많이 부족하지요.




그럼, 이 문제를 해결해야 할 텐데. 어디... 필요한 방법을 하나씩 찾아볼까요?

우선, 언어적으로 확장하는 것은, 제가 "Anders Hejlsberg"와 같은 위치에 사람이 아닌 다음에야 불가능한 상황이기 때문에 다른 것을 생각해 봐야 했습니다. 음... 어쩔 수 없지요. 그렇다면, 외부 파일을 사용해야 하고, 그 포맷은... 만만한 XML이 좋을 것 같습니다.

XML로 위의 C/C++ 예제를 바꾼다면 다음과 같은 식이 되겠지요.

<IFs>
  <DOIT condition="1" />
  <DOIT condition="2" />
  <DOIT condition="3" />
  <DOIT condition="4" />
  <DOIT condition="5" />
</IFs>

C/C++의 매크로만큼 간결한 것은 아니지만... 뭐 그래도 그 정도는 감수해야지요. ^^ 자,,, 위와 같이 XML을 만들었으면 이제 코드로 변환을 해야할 텐데요. "변환"이니까, 이 상황에서는 당연히 "XSLT"가 나와야 겠지요. 대강 아래와 같은 식으로 만들 수 있습니다.

<?xml version="1.0" encoding="UTF-8" ?>
<stylesheet version="1.0" xmlns="http://www.w3.org/1999/XSL/Transform">
  <output method="text"  encoding="utf-8" indent="yes"></output>

  <template match="IFs">
using System;
using System.Collections.Generic;
using System.Text;

namespace macroTest
{
  public partial class Test
  {
    private void DoIt(int condition)
    {
    <apply-templates select="//DOIT"></apply-templates>
    }
  }
}

  </template>

  <template match="//DOIT">
      if (condition == <value-of select="@condition"/>)
      {
        funcName_<value-of select="@condition"/>();
      }
  </template>

</stylesheet>

XSL 변환 후에는 다음과 같은 식으로 코드가 생성됩니다.

using System;
using System.Collections.Generic;
using System.Text;

namespace macroTest
{
  public partial class Test
  {
    private void DoIt(int condition)
    {
    
      if (condition == 1)
      {
        funcName_1();
      }
      
.........[반복]..........
    }
  }
}      




이것으로 XML과 XSL 파일이 만들어졌습니다. 물론, 아직까지는 그다지 특별한 매력이 없어 보입니다. 매번 XML을 수정할 때마다 XSLT를 수행해주고, 변환된 코드를 다시 복사해서 붙여넣기를 한다는 것은 오히려 절차만 더 복잡하게 만들었을 뿐입니다.

자... 그럼, 이제 이러한 과정을 자동화해야할 텐데요... 뭐가 좋을까요? ^^
저는 다음의 방법을 생각해 내었습니다.

Visual Studio .NET - 2. Custom Tools
; https://docs.microsoft.com/en-us/archive/msdn-magazine/2002/october/visual-studio-net-helps-you-go-from-geek-to-guru

; https://docs.microsoft.com/en-us/visualstudio/extensibility/internals/custom-tools

그래서, 위와 같은 변환을 자동으로 해주는 "XmlCodeGenerator"라는 것을 만들어봤습니다. 코드는 다음과 같이 매우 간단합니다.

protected override byte[] GenerateCode(string inputFileName, string inputFileContent)
{
  string xmlFilePath = inputFileName;
  string xslFilePath = Path.ChangeExtension(xmlFilePath, "xslt");

  XslCompiledTransform xslt = new XslCompiledTransform();
  XsltSettings xst = XsltSettings.Default;
  xslt.Load(xslFilePath, xst, null);

  StringBuilder sb = new StringBuilder();
  StringWriter sw = new StringWriter(sb, CultureInfo.CurrentCulture);

  XmlWriterSettings xws = new XmlWriterSettings();
  xws.ConformanceLevel = ConformanceLevel.Auto;

  XmlReaderSettings xrs = new XmlReaderSettings();

  using (XmlWriter writer = XmlTextWriter.Create(sw, xws))
  using (XmlReader reader = XmlReader.Create(xmlFilePath, xrs))
  {
    XsltArgumentList xal = new XsltArgumentList();
    xslt.Transform(reader, xal, writer);
  }

  return System.Text.Encoding.UTF8.GetBytes(sb.ToString());
}

위와 같이 만들어진 "XmlCodeGenerator"를 레지스트리에 등록하고 난 후, 원하는 소스를 생성하고 싶다면 다음과 같은 단계를 거치면 됩니다.

  1. 솔루션에 XML 파일을 추가한다. (예를 들어, test.xml)
  2. XML 파일과 동일한 이름으로 확장자만 xslt인 파일을 만든다. (예를 들어, test.xslt)
  3. test.xml 파일을 선택하고 속성창의 "Custom Tool"에 "XmlCodeGenerator"라고 입력한다.

결과적으로 정리해 보면, 다음과 같은 그림이 됩니다.

csharp_extend_macro_1.png

보시는 것처럼, 지정된 XmlCodeGenerator 덕분에 "Test.xml" 하위 노드에 "Test.cs" 파일이 자동으로 생성되었습니다. 이뿐만이 아닙니다. 이후에 "Test.xml"을 수정하고 "저장"을 하게 되면 자동으로 "Test.cs" 파일이 재생성됩니다. 게다가, 소스 제어에 참여하고 있는 경우에는 자동으로 "Test.cs" 파일을 체크아웃시키고 변경 내용이 반영됩니다. 이 정도면... 마음에 드실까요? ^^




물론, 위에서 제작해 본 CodeGenerator가 C/C++에서 제공하는 매크로 기능을 100% 완벽하게 대체할 수는 없지만, 어느 정도는 반복적인 작업들에 대해서 훌륭하게 그 자리를 메꿔줄 수도 있습니다. 또한, 여러분이 상상력을 발휘하시는 만큼... 나름대로의 확장을 거쳐 더욱 훌륭한 기능을 발휘할 수도 있습니다.

예를 들어, 다음과 같은 XML을 정의해 볼수도 있겠지요.

<BasicDac ConnectionString="...">
  <Table Name="Board" />
  <Table Name="Attachment" />
  <Table Name="Memo" />
</BasicDac>

위와 같은 XML에 대한 XSLT에서는, DB 관련 메타 정보를 반환해주는 "XSLT 확장 함수"를 정의해서 기본적인 CRUD 함수를 생성하는 코드를 생성하도록 만들수도 있습니다.

이런 식으로 확장을 해나가다 보면,,, 나중에는 XML과 XSLT를 프로젝트마다 재활용해서 사용할 수도 있을 것입니다. 음... 그렇다면 어느 날엔가는 C# 코드가 아닌 XML 프로젝트가 나올지도 모르겠군요. ^^




[첨부 파일 설명]

  • macroTest.zip: Test.xml/Test.xslt 파일을 가지고 있는 예제 솔루션 파일
  • CodeGenerator_Release.zip: "XmlCodeGenerator"를 빌드한 바이너리 파일들을 포함.

XmlCodeGenerator.dll은 Visual Studio 2005/2008(beta2) 및 Vista x64, Windows 2003 x86에서 동작하는 것을 확인했습니다.

또한, Plug-in이기 때문에 관련 레지스트리에 등록되어야 하는데, 이러한 작업을 간단히 할 수 있도록 압축 파일에 "InstallCodeGenerator.bat" / "UninstallCodeGenerator.bat" 파일을 추가해 두었습니다. (물론, 이러한 배치 파일을 사용하지 않고, Visual Studio .NET - 2. Custom Tools에 소개한 내용대로 직접 레지스트리에 등록해 주는 것도 가능합니다.)

그 외에, Vista UAC 모드에서는 "Run as administrator"를 이용해서 띄운 명령행 창에서 실행시켜 줘야 합니다.

모든 준비가 끝난 후에는, "macroTest.sln" 파일을 VS.NET 2005/2008에서 불러들이고 "Test.xml" 파일의 내용을 변경한 후 저장해 보시면 잘 동작이 되고 있는지 확인해 보실 수 있습니다.



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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2007-11-05 08시59분
[송기수] ㅋㅋ..Good~

시크리트 아티클 잘봤습니다.
[guest]
2007-11-05 09시17분
^^
kevin25
2007-11-12 10시24분
[Annguk] 새로운 정보를 많이 얻고 갑니다.

그럼 좋은 글 많이 올려주세요 ~^^
[guest]
2007-11-12 10시46분
^^
kevin25
2008-12-22 02시57분
오호... 이미 이에 대한 글을 쓰신 분이 있네요. ^^;

Using XML/XSLT with the "C Preprocessor"
; http://www.codeproject.com/KB/XML/UsingXML_XSLT.aspx

첫 번째 글을 5월 2일날 쓰다늬... 저보다 6개월을 앞섰군요. ^^; 쩝.
kevin25

... [76]  77  78  79  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
12036정성태10/14/201925431.NET Framework: 866. C# - 고성능이 필요한 환경에서 GC가 발생하지 않는 네이티브 힙 사용파일 다운로드1
12035정성태10/13/201919576개발 환경 구성: 461. C# 8.0의 #nulable 관련 특성을 .NET Framework 프로젝트에서 사용하는 방법 [2]파일 다운로드1
12034정성태10/12/201918893개발 환경 구성: 460. .NET Core 환경에서 (프로젝트가 아닌) C# 코드 파일을 입력으로 컴파일하는 방법 [1]
12033정성태10/11/201923090개발 환경 구성: 459. .NET Framework 프로젝트에서 C# 8.0/9.0 컴파일러를 사용하는 방법
12032정성태10/8/201919215.NET Framework: 865. .NET Core 2.2/3.0 웹 프로젝트를 IIS에서 호스팅(Inproc, out-of-proc)하는 방법 - AspNetCoreModuleV2 소개
12031정성태10/7/201916478오류 유형: 569. Azure Site Extension 업그레이드 시 "System.IO.IOException: There is not enough space on the disk" 예외 발생
12030정성태10/5/201923275.NET Framework: 864. .NET Conf 2019 Korea - "닷넷 17년의 변화 정리 및 닷넷 코어 3.0" 발표 자료 [1]파일 다운로드1
12029정성태9/27/201924107제니퍼 .NET: 29. Jennifersoft provides a trial promotion on its APM solution such as JENNIFER, PHP, and .NET in 2019 and shares the examples of their application.
12028정성태9/26/201919061.NET Framework: 863. C# - Thread.Suspend 호출 시 응용 프로그램 hang 현상을 해결하기 위한 시도파일 다운로드1
12027정성태9/26/201914807오류 유형: 568. Consider app.config remapping of assembly "..." from Version "..." [...] to Version "..." [...] to solve conflict and get rid of warning.
12026정성태9/26/201920231.NET Framework: 862. C# - Active Directory의 LDAP 경로 및 정보 조회
12025정성태9/25/201918537제니퍼 .NET: 28. APM 솔루션 제니퍼, PHP, .NET 무료 사용 프로모션 2019 및 적용 사례 (8) [1]
12024정성태9/20/201920448.NET Framework: 861. HttpClient와 HttpClientHandler의 관계 [2]
12023정성태9/18/201920912.NET Framework: 860. ServicePointManager.DefaultConnectionLimit와 HttpClient의 관계파일 다운로드1
12022정성태9/12/201924860개발 환경 구성: 458. C# 8.0 (Preview) 신규 문법을 위한 개발 환경 구성 [3]
12021정성태9/12/201940660도서: 시작하세요! C# 8.0 프로그래밍 [4]
12020정성태9/11/201923838VC++: 134. SYSTEMTIME 값 기준으로 특정 시간이 지났는지를 판단하는 함수
12019정성태9/11/201917389Linux: 23. .NET Core + 리눅스 환경에서 Environment.CurrentDirectory 접근 시 주의 사항
12018정성태9/11/201916188오류 유형: 567. IIS - Unrecognized attribute 'targetFramework'. Note that attribute names are case-sensitive. (D:\lowSite4\web.config line 11)
12017정성태9/11/201920003오류 유형: 566. 비주얼 스튜디오 - Failed to register URL "http://localhost:6879/" for site "..." application "/". Error description: Access is denied. (0x80070005)
12016정성태9/5/201920012오류 유형: 565. git fetch - warning: 'C:\ProgramData/Git/config' has a dubious owner: '(unknown)'.
12015정성태9/3/201925403개발 환경 구성: 457. 윈도우 응용 프로그램의 Socket 연결 시 time-out 시간 제어
12014정성태9/3/201919157개발 환경 구성: 456. 명령행에서 AWS, Azure 등의 원격 저장소에 파일 관리하는 방법 - cyberduck/duck 소개
12013정성태8/28/201922065개발 환경 구성: 455. 윈도우에서 (테스트) 인증서 파일 만드는 방법 [3]
12012정성태8/28/201926623.NET Framework: 859. C# - HttpListener를 이용한 HTTPS 통신 방법
12011정성태8/27/201926219사물인터넷: 57. C# - Rapsberry Pi Zero W와 PC 간 Bluetooth 통신 예제 코드파일 다운로드1
... [76]  77  78  79  80  81  82  83  84  85  86  87  88  89  90  ...