Question

I have an MSBuild file that builds */.sln files (builds all .sln files that exist).

The build uses the Build target, so if no changes were made to input files, no project should be built again.

I would like to execute some custom target only if a project actually gets built again.

How can this be done?

Both AfterBuild and AfterCompile are always called, no matter if the compile/build actually takes place.

Was it helpful?

Solution

Basically you want the same behaviour as the PostBuildEvent for instance, so I looked up how Microsoft.Common.Targets does it (this file always provides a nice insight in to how msbuild is supposed to be used). Here's the solution:

<PropertyGroup>
  <RunPostBuildEvent>OnOutputUpdated</RunPostBuildEvent>
</PropertyGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<Target Name="RunWhenBuild" AfterTargets="CoreBuild"       
        Condition="'$(_AssemblyTimestampBeforeCompile)'!='$(_AssemblyTimestampAfterCompile)'">
  <Message Importance="high" Text="RunWhenBuild!!"/>
</Target>

And this is what goes on: when there is a property named RunPostBuildEvent with a value of OnOutputUpdated the CoreBuild target's dependncies will eventually record the timestamp of the output file before and after the build. And if they are equal the output was not build. So all that is left is getting your target to run after CoreBuild and checking on these timestamps.

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top