次の方法で共有


RoslynCodeTaskFactory を使って MSBuild インラインラスクを作成する

CodeTaskFactory と同様に、RoslynCodeTaskFactory ではクロス プラットフォームの Roslyn コンパイラを使用して、インライン タスクとして使用するためのメモリ内タスク アセンブリを生成します。 RoslynCodeTaskFactory タスクは、.NET Standard をターゲットとし、.NET Framework や .NET Core のランタイムだけでなく、Linux や macOS などの他のプラットフォームでも機能します。

Note

RoslynCodeTaskFactory は MSBuild 15.8 以降でのみ使用できます。 MSBuild のバージョンは Visual Studio のバージョンに従うので、RoslynCodeTaskFactory は Visual Studio 2017 バージョン 15.8 以降で利用できます。

RoslynCodeTaskFactory を使用したインライン タスクの構造

RoslynCodeTaskFactory インライン タスクは、CodeTaskFactory とまったく同じ方法で宣言されています。唯一の違いは、.NET Standard をターゲットとしていることです。 インライン タスクとそれを格納する UsingTask 要素は通常、.targets ファイルに含められ、必要に応じて他のプロジェクト ファイルにインポートされます。 基本的なインライン タスクの例を次に示します。 このタスクでは何も実行されないことに注意してください。

<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <!-- This simple inline task does nothing. -->
  <UsingTask
    TaskName="DoNothing"
    TaskFactory="RoslynCodeTaskFactory"
    AssemblyFile="$(MSBuildToolsPath)\Microsoft.Build.Tasks.Core.dll" >
    <ParameterGroup />
    <Task>
      <Reference Include="" />
      <Using Namespace="" />
      <Code Type="Fragment" Language="cs">
      </Code>
    </Task>
  </UsingTask>
</Project>

この例の UsingTask 要素には、タスクとそれをコンパイルするインライン タスク ファクトリを記述した 3 つの属性が含まれています。

  • TaskName 属性は、タスクの名前を指定します。この例では DoNothing と指定しています。

  • TaskFactory 属性は、インライン タスク ファクトリを実装するクラスの名前を指定します。

  • AssemblyFile 属性は、インライン タスク ファクトリの場所を指定します。 代わりに、AssemblyName 属性を使用してインライン タスク ファクトリ クラスの完全修飾名を指定することもできます。これは通常、グローバル アセンブリ キャッシュ (GAC: Global Assembly Cache) に配置されます。

DoNothing タスクの残りの要素は空になっていますが、ここではインライン タスクの順序と構造を示すために含めてあります。 具体的な例については、この後の例を参照してください。

  • ParameterGroup 要素は省略可能です。 指定する場合は、タスクのパラメーターを宣言します。 入力パラメーターと出力パラメーターの詳細については、この後の「入力パラメーターと出力パラメーター」を参照してください。

  • Task 要素にはタスクのソース コードを記述して格納します。

  • Reference 要素は、コードで使用する .NET アセンブリへの参照を指定します。 これは、Visual Studio でプロジェクトに参照を追加することに相当します。 Include 属性は参照アセンブリのパスを指定します。

  • Using 要素には、アクセスする名前空間をリストします。 これは、Visual C# の Using ステートメントに似ています。 Namespace 属性は、含める名前空間を指定します。

Reference 要素と Using 要素は言語に依存しません。 インライン タスクは、サポートされているどの .NET CodeDom 言語 (Visual Basic や Visual C# など) でも記述できます。

Note

Task 要素に含まれる要素はタスク ファクトリによって異なります。この例では、コード タスク ファクトリが格納されています。

コード要素

Task 要素内の最後の子要素は Code 要素です。 Code 要素では、タスクにコンパイルするコードを記述するか参照します。 Code 要素に含める内容は、タスクを記述する方法に応じて異なります。

Language 属性は、コードを記述する言語を指定します。 指定できる値は、cs (C# の場合) または vb (Visual Basic の場合) です。

Type 属性は、Code 要素内のコードの種類を指定します。

  • Type の値が Class の場合、Code 要素には ITask インターフェイスから派生したクラスのコードが含まれます。

  • Type の値が Method の場合、コードは Execute インターフェイスの ITask メソッドのオーバーライドを定義します。

  • Type の値が Fragment の場合、コードは Execute メソッドの内容を定義します。ただし、シグネチャや return ステートメントは含まれません。

コード自体は通常、<![CDATA[ マーカーと ]]> マーカーの間に記述します。 コードは CDATA セクション内に記述するため、"<" や ">" などの予約文字のエスケープを気にする必要はありません。

また、Source 要素の Code 属性を使用して、タスクのコードを含むファイルの場所を指定することもできます。 ソース ファイルのコードの種類は、Type 属性で指定された種類である必要があります。 Source 属性が指定されている場合、Type の既定値は Class です。 Source が指定されていない場合の既定値は Fragment です。

Note

ソース ファイル内のタスク クラスを定義する場合は、クラス名が、対応する UsingTask 要素の TaskName 属性と一致する必要があります。

Hello World

これは RoslynCodeTaskFactory を使用したより堅牢なインライン タスクです。 HelloWorld タスクは、既定のエラー ログ デバイスに "Hello, world!" と表示します。通常、既定のデバイスは、システム コンソールまたは Visual Studio の [出力] ウィンドウです。 この例の Reference 要素は、例を示す目的でのみ含めてあります。

<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <!-- This simple inline task displays "Hello, world!" -->
  <UsingTask
    TaskName="HelloWorld"
    TaskFactory="RoslynCodeTaskFactory"
    AssemblyFile="$(MSBuildToolsPath)\Microsoft.Build.Tasks.Core.dll" >
    <ParameterGroup />
    <Task>
      <Reference Include="System.Xml"/>
      <Using Namespace="System"/>
      <Using Namespace="System.IO"/>
      <Code Type="Fragment" Language="cs">
<![CDATA[
// Display "Hello, world!"
Log.LogError("Hello, world!");
]]>
      </Code>
    </Task>
  </UsingTask>
</Project>

HelloWorld タスクを HelloWorld.targets という名前のファイルに保存すると、次のようにしてプロジェクトから呼び出すことができます。

<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <Import Project="HelloWorld.targets" />
  <Target Name="Hello">
    <HelloWorld />
  </Target>
</Project>

入力パラメーターと出力パラメーター

インライン タスクのパラメーターは、ParameterGroup 要素の子要素です。 どのパラメーターも、それを定義する要素の名前を受け取ります。 次のコードは、Text パラメーターを定義しています。

<ParameterGroup>
    <Text />
</ParameterGroup>

パラメーターには、以下の 1 つ以上の属性を含めることができます。

  • Required は省略可能な属性で、既定値は false です。 true の場合、そのパラメーターは必須で、タスクを呼び出す前に値を指定する必要があります。

  • ParameterType は省略可能な属性で、既定値は System.String です。 System.Convert.ChangeType を使用して文字列との間で変換できる項目または値の完全修飾型に設定できます (つまり、外部タスクとの受け渡しが可能なすべての型)。

  • Output は省略可能な属性で、既定値は false です。 true の場合、そのパラメーターの値を、Execute メソッドから戻る前に指定する必要があります。

たとえば、次のように入力します。

<ParameterGroup>
    <Expression Required="true" />
    <Files ParameterType="Microsoft.Build.Framework.ITaskItem[]" Required="true" />
    <Tally ParameterType="System.Int32" Output="true" />
</ParameterGroup>

この例では、以下の 3 つのパラメーターを定義しています。

  • Expression は、System.String 型の必須の入力パラメーターです。

  • Files は、必須の項目リスト入力パラメーターです。

  • Tally は、System.Int32 型の出力パラメーターです。

Code 要素の Type 属性が Fragment または Method の場合、すべてのパラメーターに対して自動的にプロパティが作成されます。 RoslynCodeTaskFactory では、Code 要素に ClassType 属性が与えられている場合、ParameterGroup を指定する必要がありません。これはソース コードから推論されるためです (これが CodeTaskFactory との違いです)。 それ以外の場合は、タスクのソース コードで明示的にプロパティを宣言する必要があります。プロパティはパラメーター定義と完全に一致している必要があります。

次のインライン タスクは、いくつかのメッセージを記録し、文字列を返します。

<Project xmlns='http://schemas.microsoft.com/developer/msbuild/2003' ToolsVersion="15.0">

    <UsingTask TaskName="MySample"
               TaskFactory="RoslynCodeTaskFactory"
               AssemblyFile="$(MSBuildBinPath)\Microsoft.Build.Tasks.Core.dll">
        <ParameterGroup>
            <Parameter1 ParameterType="System.String" Required="true" />
            <Parameter2 ParameterType="System.String" />
            <Parameter3 ParameterType="System.String" Output="true" />
        </ParameterGroup>
        <Task>
            <Using Namespace="System" />
            <Code Type="Fragment" Language="cs">
              <![CDATA[
              Log.LogMessage(MessageImportance.High, "Hello from an inline task created by Roslyn!");
              Log.LogMessageFromText($"Parameter1: '{Parameter1}'", MessageImportance.High);
              Log.LogMessageFromText($"Parameter2: '{Parameter2}'", MessageImportance.High);
              Parameter3 = "A value from the Roslyn CodeTaskFactory";
            ]]>
            </Code>
        </Task>
    </UsingTask>

    <Target Name="Demo">
      <MySample Parameter1="A value for parameter 1" Parameter2="A value for parameter 2">
          <Output TaskParameter="Parameter3" PropertyName="NewProperty" />
      </MySample>

      <Message Text="NewProperty: '$(NewProperty)'" />
    </Target>
</Project>

これらのインライン タスクは、パスを結合してファイル名を取得できます。

<Project xmlns='http://schemas.microsoft.com/developer/msbuild/2003' ToolsVersion="15.0">

    <UsingTask TaskName="PathCombine"
               TaskFactory="RoslynCodeTaskFactory"
               AssemblyFile="$(MSBuildBinPath)\Microsoft.Build.Tasks.Core.dll">
        <ParameterGroup>
            <Paths ParameterType="System.String[]" Required="true" />
            <Combined ParameterType="System.String" Output="true" />
        </ParameterGroup>
        <Task>
            <Using Namespace="System" />
            <Code Type="Fragment" Language="cs">
            <![CDATA[
            Combined = Path.Combine(Paths);
            ]]>
            </Code>
        </Task>
    </UsingTask>

    <UsingTask TaskName="PathGetFileName"
             TaskFactory="RoslynCodeTaskFactory"
             AssemblyFile="$(MSBuildBinPath)\Microsoft.Build.Tasks.Core.dll">
        <ParameterGroup>
            <Path ParameterType="System.String" Required="true" />
            <FileName ParameterType="System.String" Output="true" />
        </ParameterGroup>
        <Task>
            <Using Namespace="System" />
            <Code Type="Fragment" Language="cs">
            <![CDATA[
            FileName = System.IO.Path.GetFileName(Path);
            ]]>
            </Code>
        </Task>
    </UsingTask>

    <Target Name="Demo">
        <PathCombine Paths="$(Temp);MyFolder;$([System.Guid]::NewGuid()).txt">
            <Output TaskParameter="Combined" PropertyName="MyCombinedPaths" />
        </PathCombine>

        <Message Text="Combined Paths: '$(MyCombinedPaths)'" />

        <PathGetFileName Path="$(MyCombinedPaths)">
            <Output TaskParameter="FileName" PropertyName="MyFileName" />
        </PathGetFileName>

        <Message Text="File name: '$(MyFileName)'" />
    </Target>
</Project>

下位互換性を提供する

RoslynCodeTaskFactory は、MSBuild バージョン 15.8 で初めて使用できるようになりました。 以前の RoslynCodeTaskFactory は使用できなかったけれども CodeTaskFactory は使用できたときのバージョンの Visual Studio と MSBuild をサポートし、ただし同じビルド スクリプトを使用したい場合があるものとします。 次の例のように、$(MSBuildVersion) プロパティを使用する Choose コンストラクトを使用して、RoslynCodeTaskFactory を使用するか、それとも CodeTaskFactory に戻すかを、ビルド時に決定することができます。

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>netcoreapp3.1</TargetFramework>
  </PropertyGroup>

  <Choose>
    <When Condition=" '$(MSBuildVersion.Substring(0,2))' >= 16 Or
    ('$(MSBuildVersion.Substring(0,2))' == 15 And '$(MSBuildVersion.Substring(3,1))' >= 8)">
      <PropertyGroup>
        <TaskFactory>RoslynCodeTaskFactory</TaskFactory>
      </PropertyGroup>
    </When>
    <Otherwise>
      <PropertyGroup>
        <TaskFactory>CodeTaskFactory</TaskFactory>
      </PropertyGroup>
    </Otherwise>
  </Choose>
  
  <UsingTask
    TaskName="HelloWorld"
    TaskFactory="$(TaskFactory)"
    AssemblyFile="$(MSBuildToolsPath)\Microsoft.Build.Tasks.Core.dll">
    <ParameterGroup />
    <Task>
      <Using Namespace="System"/>
      <Using Namespace="System.IO"/>
      <Code Type="Fragment" Language="cs">
        <![CDATA[
         Log.LogError("Using RoslynCodeTaskFactory");
      ]]>
      </Code>
    </Task>
  </UsingTask>

  <Target Name="RunTask" AfterTargets="Build">
    <Message Text="MSBuildVersion: $(MSBuildVersion)"/>
    <Message Text="TaskFactory: $(TaskFactory)"/>
    <HelloWorld />
  </Target>

</Project>