Microsoft MVP성태의 닷넷 이야기
VS.NET IDE: 55. XML/XSLT로 구현하는 매크로 확장 [링크 복사], [링크+제목 복사],
조회: 40248
글쓴 사람
정성태 (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

... 61  62  63  64  [65]  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12314정성태9/7/202019941오류 유형: 645. IIS HTTPERR - Timer_MinBytesPerSecond, Timer_ConnectionIdle 로그
12313정성태9/6/202019428개발 환경 구성: 509. Logstash - 사용자 정의 grok 패턴 추가를 이용한 IIS 로그 처리
12312정성태9/5/202026406개발 환경 구성: 508. Logstash 기본 사용법 [2]
12311정성태9/4/202018937.NET Framework: 937. C# - 간단하게 만들어 보는 리눅스의 nc(netcat), json_pp 프로그램 [1]
12310정성태9/3/202018239오류 유형: 644. Windows could not start the Elasticsearch 7.9.0 (elasticsearch-service-x64) service on Local Computer.
12309정성태9/3/202017470개발 환경 구성: 507. Elasticsearch 6.6부터 기본 추가된 한글 형태소 분석기 노리(nori) 사용법
12308정성태9/2/202019315개발 환경 구성: 506. Windows - 단일 머신에서 단일 바이너리로 여러 개의 ElasticSearch 노드를 실행하는 방법
12307정성태9/2/202020187오류 유형: 643. curl - json_parse_exception / Invalid UTF-8 start byte
12306정성태9/1/202017451오류 유형: 642. SQL Server 시작 오류 - error code 10013
12305정성태9/1/202019282Windows: 172. "Administered port exclusions"이 아닌 포트 범위 항목을 삭제하는 방법
12304정성태8/31/202017702개발 환경 구성: 505. 윈도우 - (네트워크 어댑터의 우선순위로 인한) 열거되는 IP 주소 순서를 조정하는 방법
12303정성태8/30/202018026개발 환경 구성: 504. ETW - 닷넷 프레임워크 기반의 응용 프로그램을 위한 명령행 도구 etrace 소개
12302정성태8/30/202018168.NET Framework: 936. C# - ETW 관련 Win32 API 사용 예제 코드 (5) - Private Logger파일 다운로드1
12301정성태8/30/202017775오류 유형: 641. error MSB4044: The "Fody.WeavingTask" task was not given a value for the required parameter "IntermediateDir".
12300정성태8/29/202018082.NET Framework: 935. C# - ETW 관련 Win32 API 사용 예제 코드 (4) CLR ETW Consumer파일 다운로드1
12299정성태8/27/202018821.NET Framework: 934. C# - ETW 관련 Win32 API 사용 예제 코드 (3) ETW Consumer 구현파일 다운로드1
12298정성태8/27/202018303오류 유형: 640. livekd - Could not resolve symbols for ntoskrnl.exe: MmPfnDatabase
12297정성태8/25/202017789개발 환경 구성: 503. SHA256 테스트 인증서 생성 방법
12296정성태8/24/202018898.NET Framework: 933. C# - ETW 관련 Win32 API 사용 예제 코드 (2) NT Kernel Logger파일 다운로드1
12295정성태8/24/202017776오류 유형: 639. Bitvise - Address is already in use; bind() in ListeningSocket::StartListening() failed: Windows error 10013: An attempt was made to access a socket ,,,
12293정성태8/24/202018731Windows: 171. "Administered port exclusions" 설명
12292정성태8/20/202021742.NET Framework: 932. C# - ETW 관련 Win32 API 사용 예제 코드 (1)파일 다운로드2
12291정성태8/15/202018713오류 유형: 638. error 1297: Device driver does not install on any devices, use primitive driver if this is intended.
12290정성태8/11/202019890.NET Framework: 931. C# - IP 주소에 따른 국가별 위치 확인 [8]파일 다운로드1
12289정성태8/6/202016844개발 환경 구성: 502. Portainer에 윈도우 컨테이너를 등록하는 방법
12288정성태8/5/202015999오류 유형: 637. WCF - The protocol 'net.tcp' does not have an implementation of HostedTransportConfiguration type registered.
... 61  62  63  64  [65]  66  67  68  69  70  71  72  73  74  75  ...