Being able to use external DLLs in
Dynamics CRM plugins is challenging but doable. Most people who have been doing
development on the Dynamics CRM platform for any period of time know that it is
not enough to simply reference the DLLs in Visual Studio – you must somehow
make them available to run on the server – this can range from mildly annoying
to impossible (if you using CRM Online.)
You
can read the Dynamics CRM Blog article by Michael Scott for
more details, but here is the jist (the same rules apply to 2013/2015):
You may have noticed that Microsoft Dynamics CRM 2011 (and CRM 4 before it)
does not really have support for referencing custom assemblies from a plug-in
assembly. You are limited to .NET Framework assemblies and the public Microsoft Dynamics CRM 2011
Software Development Kit (SDK)assemblies. In Microsoft Visual Studio, your references will look
something like this:
If you wanted to reference an
assembly that you had written, your options were pretty limited:
§ Register the assembly on disk with any referenced
assemblies in the same directory.
§ Register the referenced assembly in the GAC
§ Include the relevant source code in the plug-in
assembly
With the introduction of Solutions in
Microsoft Dynamics CRM 2011, the first two options are not very manageable (as
they require special setup on multiple servers during the installation of the
solution”>PICTURE REMOVED]
If you wanted to reference an
assembly that you had written, your options were pretty limited:
§ Register the assembly on disk with any referenced
assemblies in the same directory.
§ Register the referenced assembly in the GAC
§ Include the relevant source code in the plug-in
assembly
With the introduction of Solutions in
Microsoft Dynamics CRM 2011, the first two options are not very manageable
. In addition,
those options don’t work in Microsoft Dynamics CRM Online, since assemblies must be
registered in the database in CRM Online. If you include source code, as the
last option suggests, then all of the benefits of a referenced assembly are
lost, without gaining any real benefits.
In
the same article it is recommended to use ILMerge. ILMerge is a tool from Microsoft Research that can
combine multiple DLLs into a single DLL – eliminating the problem (in the case
of Dynamics CRM) of having multiple DLL files.
The article goes on to describe the
basics of using ILMerge to create a single plugin DLL. While I have done this
on many projects over the year it has always been more complicated to initially
setup and maintain than I would prefer. It can also be challenging to teach new
developers – many of whom are still getting comfortable with Dynamics CRM
development.
I recently decided to do some
research and figure out a simple and well defined set of step-by-step
instructions to use ILMerge in development. In the next several steps I will
describe the exact process I now use, which doesn’t involve any editing of
MSBUILD files, creation of PowerShell or BAT scripts, etc. It only requires the
use of Nuget and the Visual Studio GUI.
1. Start by creating a new Blank Solution in Visual
Studio (Other Project Types -> Video Studio Solutions.) Name it as you wish,
I used ThinkCrmBlog.PluginDllMerge.
2. Now that you have a new solution add a couple of
class libraries. I’ve named mine ThinkCrmBlog.CrmHelper (this is inline with my
best practice of having a standalone DLL of helper classes that can be used for
CRM development but are not specific to any particular CRM org and can be used
in plugins, workflows, and client side code) and ThinkCrmBlog.DemoPlugin (which
will hold my IPlugin class.data:image/s3,"s3://crabby-images/24b5d/24b5d1f08bb1bf4664267fa8b8a82dce2c0753c6" alt=""
3. So now your solution should look like this picture
(assuming you deleted the Class1.cs files.)data:image/s3,"s3://crabby-images/c1f26/c1f2666a8d82c1248f5ecdf1c54c41c0c02bf63d" alt=""
4. We’ll add the Dynamics CRM libraries using NuGet.data:image/s3,"s3://crabby-images/3d9d8/3d9d82b296371b118f77c508cb7a5657918677a5" alt=""
5. Search for Microsoft.CrmSdk.CoreAssemblies and
install.data:image/s3,"s3://crabby-images/a9060/a9060adc5253f1e55e4dbde19eca36bf9cf189b7" alt=""
6. You want to install to both projects.data:image/s3,"s3://crabby-images/54c6f/54c6fb22f9052cad90938901c071c36d8e7b46a6" alt=""
7. Using NuGet greatly simplifies some tasks by
including required dependencies. In this case it is Microsoft.IdentifyModel.
Although technically not required to compile a plugin DLL if you were to use
the core assemblies in a client side app it is necessary and can be something
of pain when you do not realize you need it until runtime.data:image/s3,"s3://crabby-images/9e0e9/9e0e9531373c197f13ea2ebac4a84e1245c2cf0d" alt=""
8. You can tell that everything has succeeded by the
green check.data:image/s3,"s3://crabby-images/512dd/512dd44193d42acc76eb0cedb22b05698f7ef85b" alt=""
9. So now it is time to get ILMerge installed. Luckily
for the modern developer we have NuGet to make this super simple. We don’t want
just ILMerge, we want a tool that automates the ILMerge process into the build
process. Do a search for MSBuild.ILMerge.Task and click install.data:image/s3,"s3://crabby-images/7fe96/7fe968e0dedf960e50ce616bf33a96a25c212910" alt=""
10. We only want to install to the project that will
need to have DLLs merged, so uncheck the box next to ThinkCrmBlog.CrmHelper.
Now, if you had more projects creating plugin DLLs that referenced
ThinkCrmBlog.CrmHelper you would want to check the box for each of those
projects.data:image/s3,"s3://crabby-images/ffa0d/ffa0d3fa09f1f3dd707b1805c9ca2d6e378099e3" alt=""
11. The ILMerge application itself is available as a
NuGet package and is a dependency, so it will be installed.data:image/s3,"s3://crabby-images/a0ff5/a0ff503d6b6531f8e5d05f6b072e02803ba7efc9" alt=""
12. Again, the green check is your confirmation of
success.data:image/s3,"s3://crabby-images/2afd1/2afd161ebdb405784ee9644ea1ed96f91a4bbb58" alt=""
13. Now we need to add a reference
to ThinkCrmBlog.CrmHelper from ThinkCrmBlog.DemoPlugin.data:image/s3,"s3://crabby-images/a02ac/a02aced94ed9268dfe3ffd8fe720198133000540" alt=""
14. Just check the box.data:image/s3,"s3://crabby-images/cdf13/cdf131289deb3c189c8a3e46163fdc57298746e8" alt=""
15. Now we have to do a property update on the CRM SDK
DLL files so they don’t get merged into our final DLL. The ILMerge build task
we installed will merge all the DLLs in the \bin directory. To avoid merge
the CRM SDK DLLs and Microsoft.IdentityModel.DLL into our plugin DLL we need to
tell Visual Studio to not copy them on build. Select the three files and set
the property “Copy Local” to false.data:image/s3,"s3://crabby-images/c1421/c14213f193607e0e324dbb24d0fec618db87e356" alt=""
16. One last, but critical step, be sure to sign both
of the projects. Dynamics CRM requires that DLLs be signed and therefore any
DLLs we include must be signed.data:image/s3,"s3://crabby-images/377f6/377f6d17429dae4c1e0e5af83aa437caf8fc1d26" alt=""
17. The solution should look similar to this picture.data:image/s3,"s3://crabby-images/590a5/590a571aff89b6da4cc1ddcc054bc4d3d25f7de4" alt=""
18. For the source code, which I’ve placed online on
GitHub @ https://github.com/nicknow/ThinkCrmBlog.PluginDllMerge,
I created a very basic plugin will create an instance of a class in CrmHelper
and call a method before throwing an exception to allow the user to via the
trace log. This was done just for demonstration – it serves no real-world
purpose. Afte writing the code I did a Clean Solution and a Rebuild Solution.data:image/s3,"s3://crabby-images/5ef66/5ef66d38c8cdc9bafd0e31ae853dd1fa674935c2" alt=""
19. In the output directory you’ll see only the final
DLL – the merged DLL(s) have been deleted. If you open the DLL in a decompiler,
such as dotPeek, you’ll see that the DLL contains both the
ThinkCrmBlog.DemoPlugin and ThinkCrmBlog.CrmHelper namespaces.data:image/s3,"s3://crabby-images/84366/843664881fe11a1cef39f294c3bb90cb40511cab" alt=""
20. Install the DLL using the Plugin Registration Tool
(or any other preferred tool.) You’ll see that it is a Sandbox deployment – so
it will work with CRM Online.data:image/s3,"s3://crabby-images/bbc24/bbc24b37e25d90a685aa2678e9c690307bf516cd" alt=""
Remember, just because you can merge
DLLs doesn’t mean you don’t have any limitations. Code running in Sandbox must
still adhere to the rules – no SQL calls, disk access, reflection, etc. Also,
be careful that you aren’t merging so much that your plugin DLL becomes bloated
causing issues with deployment or (possibly) a negative performance hit.
No comments:
Post a Comment