Using SmartAssembly with ClickOnce and MSI
Published 02 January 2013
This page explains how to use SmartAssembly with Microsoft Setup and Deployment projects in Visual Studio (MSI and Microsoft ClickOnce).
If you use SmartAssembly after you have built your project, you will not be able to deploy it using MSI and ClickOnce. This is because SmartAssembly has modified the assembly.
To use MSI or ClickOnce to deploy your assembly, integrate SmartAssembly directly into the build process in MSBuild.
Open your solution in Visual Studio, change the build output path to obj\Release\ (or obj\x86\Release if your platform target is x86) and build the project in Release mode.
You can confirm from the build output window if your project is successfully building to this path.Create your project using the SmartAssembly project settings page as normal.
Make sure that the input assembly path is the \obj\Release folder (or obj\x86\Release if your platform target is x86), and NOT \bin\Release or bin\x86\Release. The location of the destination assembly does not matter.
- Open your C# or VB.NET project with an XML editor (or Notepad).
The project file looks like the following:
<Project DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> <PropertyGroup> ... </PropertyGroup> ... <ItemGroup> ... </ItemGroup> <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> </Project>
Add a few lines after the
<Import Project>
node:<!-- SmartAssembly task --> <UsingTask TaskName="SmartAssembly.MSBuild.Tasks.Build" AssemblyName="SmartAssembly.MSBuild.Tasks, Version=7.0.0.0, Culture=neutral, PublicKeyToken=7f465a1c156d4d57" /> <Target Name="BuildWithSmartAssembly" AfterTargets="CoreCompile" Condition=" '$(Configuration)' == 'Release' "> <SmartAssembly.MSBuild.Tasks.Build ProjectFile="$(ProjectDir)$(AssemblyName).saproj" OverwriteAssembly="True" /> </Target> <!-- /SmartAssembly task -->
The
<UsingTask>
node adds a reference to SmartAssembly.MSBuild.Tasks.dll, which is installed into the GAC.The second XML node tells MSBuild to run SmartAssembly after
CoreCompile
event, when the project is fully built in Release mode.
You must set the following properties of SmartAssembly.MSBuild.Tasks.Build
task:
| Sets the path to the SmartAssembly project file .saproj. |
| Set to True if you want to overwrite the original assembly with the obfuscated one. If this option is not set, or set to False, SmartAssembly uses the destination file name from the ProjectFile project. |
Other options are:
| Specify an input file name, instead of the one from the ProjectFile project. |
| Specify the output file name for the obfuscated assembly. This property is ignored if the OverwriteAssembly property is set to True. |
| Use a specific version number for the assembly. This changes the version for the AssemblyName, not the file version. |
When outputting an MSI, if you use the Project Output option to embed a file into the program files folder of your deployment project, the embedded file might not be protected inside the MSI. For more information, see Visual Studio Deployment Projects include unprotected builds.
If you have multiple projects
- Ensure that your SA project file is stored in the same folder as your C# or VB.NET project. It must have the same name as the target assembly. For example, if your assembly is MyTestAssembly.dll then the SmartAssembly project must be named MyTestAssembly.saproj.
- Create a new file called SmartAssembly.targets in the same folder as your Visual Studio solution file .sln.
Paste the following code in the new file:
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> <UsingTask TaskName="SmartAssembly.MSBuild.Tasks.Build" AssemblyName="SmartAssembly.MSBuild.Tasks, Version=7.0.0.0, Culture=neutral, PublicKeyToken=7f465a1c156d4d57" /> <Target Name="BuildWithSmartAssembly" AfterTargets="CoreCompile" Condition=" '$(Configuration)' == 'Release' "> <SmartAssembly.MSBuild.Tasks.Build ProjectFile="$(ProjectDir)$(AssemblyName).saproj" Input="$(ProjectDir)obj\Release\$(TargetName)$(TargetExt)" OverwriteAssembly="True" /> </Target> </Project>
Edit your C# or VB.NET project with an XML editor (or notepad). Add the following line:
<Import Project="$(SolutionDir)SmartAssembly.targets" />