Esto es compatible con VS 2010. Pero no 2008. Consulte la opción UseHardLinksIfPossible para Copiar en _CopyFilesMarkedCopyLocal.
Véase también http://social.msdn.microsoft.com/Forums/en/tfsbuild/thread/9382a3d8-4632-4826-ad15-d5e845080981, http://msdn.microsoft.com/en-us/library/ms171466(v=VS.90).aspx para contexto.
Reemplazar el objetivo _CopyFilesMarkedCopyLocal. Agregamos algo como esto a los archivos csproj en la parte inferior: <Import Project="..\..\..\..\..\\CommonBuild\TW.Override.Microsoft.Common.targets" />
(Esto se agrega automáticamente al archivo con una tarea nant en cada compilación nant completa, por lo que cada proyecto se beneficia).
Nuestro archivo de nuevos objetivos es:
<Project ToolsVersion="3.5" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<UsingTask TaskName="CopyWithHardlinkOption" AssemblyFile="..\lib\TWBuildOptimization\TW.Hardlinker.dll" />
<!--
============================================================
_CopyFilesMarkedCopyLocal
Overridden in order to allow hardlinking with our custom Copy Task.
Hardlinking is a major performance improvement. Sometimes 50% of the time compared to copying.
Copy references that are marked as "CopyLocal" and their dependencies, including .pdbs, .xmls and satellites.
============================================================
-->
<Target
Name="_CopyFilesMarkedCopyLocal">
<CopyWithHardlinkOption
SourceFiles="@(ReferenceCopyLocalPaths)"
DestinationFiles="@(ReferenceCopyLocalPaths->'$(OutDir)%(DestinationSubDirectory)%(Filename)%(Extension)')"
SkipUnchangedFiles="true"
UseHardlinksIfPossible="true"
OverwriteReadOnlyFiles="$(OverwriteReadOnlyFiles)">
<Output TaskParameter="DestinationFiles" ItemName="FileWritesShareable"/>
</CopyWithHardlinkOption>
</Target>
</Project>
ver la opción de msbuild 4 Copia UseHardlinksIfPossible. Transporté eso a 3.5 mediante la descompilación y la reimplementación. La lógica relevante en CopyFileWithLogging fue:
// The port from 4.0's task that allows hardlinking
bool hardlinkSucceeded = false;
if (UseHardlinksIfPossible)
{
if (File.Exists(destinationFile))
{
FileUtilities.DeleteNoThrow(destinationFile);
}
if (!TwNativeMethods.CreateHardLink(destinationFile, sourceFile, IntPtr.Zero))
{
var win32Exception = new Win32Exception(Marshal.GetLastWin32Error());
Log.LogMessage(MessageImportance.High, "Hardlinking had a problem {0}, will retry copying. {1}", new object[] {win32Exception.Message, win32Exception});
}
hardlinkSucceeded = true;
}
if (!hardlinkSucceeded)
{
Log.LogMessageFromResources(MessageImportance.Normal, "Copy.FileComment", new object[] { sourceFile, destinationFile });
Log.LogMessageFromResources(MessageImportance.Low, "Shared.ExecCommand", new object[0]);
Log.LogCommandLine(MessageImportance.Low, "copy /y \"" + sourceFile + "\" \"" + destinationFile + "\"");
File.Copy(sourceFile, destinationFile, true);
}
// end port
también tuvo que añadir los siguientes:
// decompiled from 4.0
internal static class TwNativeMethods
{
[DllImport("kernel32.dll", CharSet=CharSet.Unicode, SetLastError=true)]
internal static extern bool CreateHardLink(string newFileName, string exitingFileName, IntPtr securityAttributes);
}
// decompiled from 4.0
internal static class FileUtilities
{
internal static void DeleteNoThrow(string path)
{
try
{
File.Delete(path);
}
catch (Exception exception)
{
if (ExceptionHandling.NotExpectedException(exception))
{
throw;
}
}
}
}
// decompiled from 4.0
internal static class ExceptionHandling
{
// Methods
internal static bool IsCriticalException(Exception e)
{
return (((e is StackOverflowException) || (e is OutOfMemoryException)) || ((e is ExecutionEngineException) || (e is AccessViolationException)));
}
internal static bool NotExpectedException(Exception e)
{
return (((!(e is UnauthorizedAccessException) && !(e is ArgumentNullException)) && (!(e is PathTooLongException) && !(e is DirectoryNotFoundException))) && ((!(e is NotSupportedException) && !(e is ArgumentException)) && (!(e is SecurityException) && !(e is IOException))));
}
internal static bool NotExpectedReflectionException(Exception e)
{
return ((((!(e is TypeLoadException) && !(e is MethodAccessException)) && (!(e is MissingMethodException) && !(e is MemberAccessException))) && ((!(e is BadImageFormatException) && !(e is ReflectionTypeLoadException)) && (!(e is CustomAttributeFormatException) && !(e is TargetParameterCountException)))) && (((!(e is InvalidCastException) && !(e is AmbiguousMatchException)) && (!(e is InvalidFilterCriteriaException) && !(e is TargetException))) && (!(e is MissingFieldException) && NotExpectedException(e))));
}
}
Por qué no estableció un DEVPATH a toda la papelera de carpetas en su lugar. Entonces no necesita hacer nada, excepto apagar copylocal. http://blogs.msdn.com/b/junfeng/archive/2005/12/13/503059.aspx – adrianm