Этап сборки, запускаемый TeamCity, всегда строится - даже если нет никаких изменений

Проблема: Я настраиваю TeamCity в качестве сервера сборки для проекта ASP.NET MVC. Я использую Powershell с psake для запуска msbuild с нашим файлом .csproj и создания развертываемого пакета. С сервера сборки я могу открыть PowerShell, запустить скрипт и, поскольку нет изменений исходного кода, msbuild не восстанавливает файлы DLL проекта. НО, когда я вызываю тот же самый сценарий из веб-интерфейса TeamCity, msbuild ВСЕГДА перестраивает и регенерирует файлы DLL, даже если нет никаких изменений. Не то, что он должен делать AFAIK.

Я сузил эту проблему до одного шага. Чтобы не усложнять, я настроил свою конфигурацию TeamCity так, чтобы она не использовала какой-либо элемент управления исходным кодом, она запускала единственный этап сборки «powershell», который вызывает мой сценарий PowerShell.

Сценарий PowerShell запускает одну команду:

exec { &$msbuild $ProjectFile /t:Package "/p:PackageLocation=$PackageFile;OutDir=$TempPath;Configuration=$Config;SolutionDir=$BaseDir\Source\" /v:m }

Когда я вызываю скрипт вручную из командной строки PowerShell, я вижу:

CoreCompile:
Skipping target "CoreCompile" because all output files are up-to-date with respect to the input files.

Когда я вызываю тот же самый скрипт через TeamCity, я вижу:

[11:11:26]: CoreCompile:
[11:11:26]:   c:\WINDOWS\Microsoft.NET\Framework\v4.0.30319\Csc.exe /noconfig ...
<SNIP>
[11:11:32]: CopyFilesToOutputDirectory:
[11:11:32]:   Copying file from "obj\Demo\Website.Web.dll" to "d:\deploy\Build\package\Demo\temp\Website.Web.dll".
[11:11:32]:   Website.Web -> d:\deploy\Build\package\Demo\temp\Website.Web.dll
[11:11:32]:   Copying file from "obj\Demo\Website.Web.pdb" to "d:\deploy\Build\package\Demo\temp\Website.Web.pdb".
[11:11:32]: _CopyWebApplicationLegacy:
[11:11:32]:   Copying Web Application Project Files for Website.Web
[11:11:32]:   Copying file from "obj\Demo\Website.Web.dll" to "d:\deploy\Build\package\Demo\temp\_PublishedWebsites\Website.Web\bin\Website.Web.dll".
[11:11:32]:   Copying file from "obj\Demo\Website.Web.pdb" to "d:\deploy\Build\package\Demo\temp\_PublishedWebsites\Website.Web\bin\Website.Web.pdb".
[11:11:32]:   Copying file from "d:\deploy\Build\package\Demo\temp\Website.Data.dll" to "d:\deploy\Build\package\Demo\temp\_PublishedWebsites\Website.Web\bin\Website.Data.dll".
[11:11:32]:   Copying file from "d:\deploy\Build\package\Demo\temp\Website.Data.pdb" to "d:\deploy\Build\package\Demo\temp\_PublishedWebsites\Website.Web\bin\Website.Data.pdb".

Любые идеи почему запуск этого скрипта из TeamCity заставляет msbuild обнаруживать изменения и перестраивать, а запуск того же самого скрипта вручную - нет?

ОБНОВЛЕНИЕ: Думая, что это могло быть вызвано какой-то причудой в TeamCity Powershell runner , Я просто попытался создать командный файл, который передает сценарий в Powershell.exe и вызвал его из командной строки:

C:\WINDOWS\system32\WindowsPowerShell\v1.0\powershell.exe -NonInteractive -File D:\deploy\Build\run-build.ps1 && exit /b %ERRORLEVEL%

, и я получил точно такое же поведение. Если я вызываю этот командный файл из командной строки, msbuild пропускает компиляцию. Если я вызываю это из TeamCity, библиотеки DLL перекомпилируются.

ОБНОВЛЕНИЕ №2: Эврика! Я включил диагностическую отладку в msbuild и нашел причину принудительной перекомпиляции. Это вызвано целью GenerateTargetFrameworkMonikerAttribute . Вот ключевые биты из вывода журнала:

[15:23:28]: Target "GenerateTargetFrameworkMonikerAttribute" in file "c:\WINDOWS\Microsoft.NET\Framework\v4.0.30319\Microsoft.Common.targets" from project "d:\deploy\source\Website.Data\Website.Data.csproj" (target "BeforeCompile" depends on it):
[15:23:28]: Building target "GenerateTargetFrameworkMonikerAttribute" completely.
[15:23:28]: Output file "C:\TeamCity\buildAgent\temp\buildTmp\.NETFramework,Version=v4.0.AssemblyAttributes.cs" does not exist.
[15:23:28]: Using "WriteLinesToFile" task from assembly "Microsoft.Build.Tasks.v4.0, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a".
[15:23:28]: Task "WriteLinesToFile"
[15:23:28]: Done executing task "WriteLinesToFile".
[15:23:28]: Done building target "GenerateTargetFrameworkMonikerAttribute" in project "SMM.Data.csproj".

Похоже, эта цель создает / обновляет файл AssemblyAttributes в каталоге TEMP, как указано в переменной среды TEMP. Очевидно, TeamCity переопределяет переменную среды TEMP и устанавливает для нее значение: C: \ TeamCity \ buildAgent \ temp \ buildTmp , и этот каталог очищается перед каждой сборкой.

Я могу это увидеть, если вызову Get-ChildItem Env: из powershell:

TEMP                           C:\DOCUME~1\ADMINI~1\LOCALS~1\Temp
TMP                            C:\DOCUME~1\ADMINI~1\LOCALS~1\Temp

Но если я вызову его из сценария powershell, вызываемого из TeamCity:

TEMP                           C:\TeamCity\buildAgent\temp\buildTmp            
TMP                            C:\TeamCity\buildAgent\temp\buildTmp   

Ключевым моментом является то, что после повторной регенерации этого файла:

[15:23:28]: Building target "CoreCompile" completely.
[15:23:28]: Input file "C:\TeamCity\buildAgent\temp\buildTmp\.NETFramework,Version=v4.0.AssemblyAttributes.cs" is newer than output file "obj\Demo\SMM.Data.pdb".

И поэтому весь проект перекомпилируется.

Когда я запускаю сценарий из Powershell, временный каталог не изменяется и не очищается, и сборка выполняется должным образом.

Итак, кто-нибудь знает, как я могу либо изменить каталог, в котором создается этот файл AssemblyAttributes, либо указать TeamCity использовать другой каталог TEMP? Я должен верить, что это проблема, с которой столкнулись другие.

Спасибо!

6
задан obliojoe 28 October 2011 в 23:12
поделиться