[Из песочницы] Расширение процесса сборки с помощью MSBuild
В данной статье речь пойдет о том, как расширить процесс сборки проекта с помощью MSBuild.
MSBuild устроен таким образом, что сборка проекта разбита на несколько этапов.
Target — это некоторый этап (событие), происходящее во время сборки проекта. Можно использовать стандартные таргеты, либо определять собственные.
Task — это некоторая задача, которая может выполняться на определенном этапе. Можно использовать стандартные таски или создавать собственные.
Цитата из документации о таргетах (https://docs.microsoft.com/en-us/visualstudio/msbuild/msbuild-targets):
Targets group tasks together in a particular order and allow the build process to be factored into smaller units.
For example, one target may delete all files in the output directory to prepare for the build, while another
compiles the inputs for the project and places them in the empty directory.
Для работы MSBuild Microsoft определил ряд стандартных таргетов (в файлах Microsoft.Common.targets, Microsoft.CSharp.targets и т.д.). Довольно тяжело найти информацию о том, какие таргеты уже определены, но опытным путём было выявлено, что есть следующие (упорядочены, могут быть указаны не все):
- BeforeRebuild
- Clean
- BeforeBuild
- BuildOnlySettings
- PrepareForBuild
- PreBuildEvent
- ResolveReferences
- PrepareResources
- ResolveKeySource
- Compile
- UnmanagedUnregistration
- GenerateSerializationAssemblies
- CreateSatelliteAssemblies
- GenerateManifests
- GetTargetPath
- PrepareForRun
- UnmanagedRegistration
- IncrementalClean
- PostBuildEvent
- AfterBuild
- AfterRebuild
Таргеты BeforeBuild и AfterBuild специально созданы для переопределения и их можно использовать. Остальные таргеты из списка не рекомендую использовать, чтобы ничего не сломалось.
Для примеров необходимо:
- Установленная среда разработки Visual Studio
- Создать проект типа Console Application с именем MSBuildExample
- Открыть папку проекта и найти там файл MSBuildExample.csproj
- Открыть файл MSBuildExample.csproj в блокноте или другом редакторе
Во всех примерах этой статьи понадобится редактировать файл MSBuildExample.csproj. Каждый пример подразумевает удаление кода предыдущего примера и добавление нового. Код необходимо добавлять в конец файла .csproj до последней строчки, содержащей закрывающий тег Project.
Внимание! В файле .csproj регистр букв важен.
Для запуска примера необходимо запускать build в среде разработки Visual Studio. Для некоторых примеров потребуется выбирать solution конфигурацию.
Результат будет выводиться в окно Output в Visual Studio (внизу). Если его нет, то откройте его через пункты меню View => Output.
Для примеров будем использовать таск Message, который будет выводить информацию в окно Output в Visual Studio. Как говорилось ранее есть стандартные таргеты BeforeBuild и AfterBuild, воспользуемся ими. Про подготовку читать в разделе Подготовка окружения для примеров.
Результат выполнения (лишнее исключено):
…
BeforeBuild event
…
AfterBuild event
…
Как видно, был выполнен task Message, который вывел указанный нами текст в момент BeforeBuild и AfterBuild в окно Output в Visual Studio.
При определении таргета с одним и тем же именем он перезаписывается!
Результат выполнения (лишнее исключено):
…
Second message
…
Вывело только второе сообщение, потому что использовали таргеты с одним именем и он был перезаписан вторым значением.
Если таргетов BeforeBuild и AfterBuild недостаточно или нужно, чтобы таски выполнялись на другом этапе жизненного цикла сборки, то можно определить собственный таргет. Для этих целей есть параметры BeforeTargets и AfterTargets.
Результат выполнения (лишнее исключено):
…
MyCustomBeforeTarget event
BeforeBuild event
MyCustomAfterTarget event
…
Было определено два собственных таргета — MyCustomBeforeTarget и MyCustomAfterTarget.
Таргет MyCustomBeforeTarget выполняется до таргета BeforeBuild, потому что мы указали:
BeforeTargets="BeforeBuild"
Таргет MyCustomAfterTarget выполняется после таргета BeforeBuild, потому что мы указали:
AfterTargets="BeforeBuild"
В данной статье не рассматривается как можно писать собственные таски, но прежде чем писать таск, ознакомьтесь со списком тасков, предоставляемых Microsoft.
Рассмотрим несколько примеров использования тасков и макросов.
Код примера:A Boolean expression that the MSBuild engine uses to determine whether this task will be executed.
Если будет выбрана solution конфигурация Debug, то результат будет выглядеть так (лишнее исключено):
Если будет выбрана solution конфигурация Release, то результат будет выглядеть так (лишнее исключено):…
Current configuration is Debug
…
Информацию о макросе $(Configuration) и других макросах можете найти в разделе переменные и макросы в .csproj.…
Current configuration is Release
…
Информацию о синтаксисе условий можно прочитать там https://docs.microsoft.com/en-us/visualstudio/msbuild/msbuild-conditions
Код примера:
Current configuration is $(Configuration)
Для определения собственной переменной используется элемент PropertyGroup.
Используем для этого таск Error и уже знакомый нам параметр Condition.
Код примера:
Результат:
В условии Exists используется относительный путь от папки, в которой находится файл .csproj. Для обращения к папке выше текущей использовать '…/'. Если нужно обратиться к вложенной папке, то использовать формат '[DirectoryName]/App.Debug.config'.
Код примера:
Свойство SourceFiles — массив файлов, которые необходимо скачать. Указывать без кавычек, через точку с запятой.
Свойство DestinationFiles — массив файлов куда будут копироваться файлы. Указывать без кавычек, через точку с запятой.
Подробнее о макросе $(OutputPath) читать в разделе переменные и макросы в .csproj.
В файле .csproj можно использовать ряд стандартных макросов, их список можно найти здесь https://msdn.microsoft.com/en-us/library/c02as0cs.aspx и здесь https://msdn.microsoft.com/en-us/library/bb629394.aspx. Рассмотрим некоторые полезные макросы:
- $(MSBuildToolsPath) — указывает на путь к папке MSBuild. Например, C:\Program Files (x86)\MSBuild\14.0\Bin. Данный макрос при комбинировании пути использовать со слешем. Например, $(MSBuildToolsPath)\Microsoft.Web.Publishing.Tasks.dll. Иначе он может некорректно формировать путь и выдавать ошибку, что файл не найден.
- $(OutputPath) — относительный путь к выходной папке. Например, bin\Stage. Данный макрос использовать со слешем, например, $(OutputPath)\$(TargetFileName).config.
- $(TargetFileName) — имя выходного файла вместе с расширением. Например, MSBuildExample.exe. Расширение и формат имени выходного файла может отличаться от различных типов проектов. С помощью этого макроса можно безопасно определить какое будет имя у файла конфига. Может быть полезно для трасформаций конфигов.
- $(Configuration) — имя текущей конфигурации. Например, Release, Debug
- $(IntermediateOutputPath) — путь к папке obj. Например, obj\Stage.
Для определения собственных параметров использовать PropertyGroup. Пример определения собственной переменной можно найти в разделе таски в MSBuild.