I have powershell code in my C# application that works great. It executes a script using the code snippet below.
using (PowerShell powerShell = PowerShell.Create())
{
powerShell.AddScript(@"D:\C.ps1");
Collection<PSObject> output = powerShell.Invoke();
}
Like I said, it works. However, recently I wanted to add a Windows Form to the script. The very first thing I did was add the following line to the top of the script and nothing else. Without doing anything else, I just ran the C# app that executes the previously working script to make sure that loading of the assembly does what it's supposed too.
[void][System.Reflection.Assembly]::Load('System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a')
Immediately I get the strange following exception.
System.Exception {System.EntryPointNotFoundException}
Unable to find an entry point named 'AmsiNotifyOperation' in DLL 'amsi.dll'
I've search online for days and found nothing like this. I can't even get started adding my new script code because of this exception for the assembly load line.
Oh, very important, When I run that script with that new line from the Powershell ISE app, the command line, or several other powershell script tools I have. THE PROBLEM DOESN'T HAPPEN.
Anyone got any ideas?
Here is the nuget packages that the code is using, and the stack trace.
Nuget packages:
System.Management.Automation
Microsoft.Powershell.SDK
Stack: Source "System.Management.Automation" at System.Management.Automation.AmsiUtils.AmsiNativeMethods.AmsiNotifyOperation(IntPtr amsiContext, IntPtr buffer, UInt32 length, String contentName, AMSI_RESULT& result) at System.Management.Automation.AmsiUtils.WinReportContent(String name, String content)
TargetSite {Int32 AmsiNotifyOperation(IntPtr, IntPtr, UInt32, System.String, AMSI_RESULT ByRef)}
I'm not sure if the following will solve all your problems - notably with respect to the AMSI (Antimalware Scan Interface) errors - but the following steps worked for me:
The PowerShell SDK itself does not include the WinForms assemblies as part of its
Microsoft.PowerShell.SDK
NuGet package (which is the NuGet package for PowerShell (Core) 7+).Note that this is the only package needed to host PowerShell (Core) in a .NET application - do not also include the
System.Management.Automation
package, whose direct use is recommended against.See also: Choosing the right PowerShell NuGet package for your .NET project
Therefore, you need to make sure that your .NET application comes with the WinForms assemblies:
The simplest approach is to initialize your .NET SDK project as follows (which targets .NET (Core) rather than .NET Framework, just like your own attempt):
Form1.cs
andForm1.cs.designer
files (which you would only need if you want to interact with WinForms from C#), and replace the startup code inProgram.cs
Then, in your PowerShell script file (
.ps1
), load the WinForms assemblies as follows, which should use the ones that come bundled with your .NET application:Optionally place statements such as
using namespace System.Windows.Forms
at the start of your script, to allow you to refer to theForm
class as just[Form]
rather than as[System.Windows.Forms.Form]
, for instance.Add-Type
call withusing assembly System.Windows.Forms
is called for too, but, regrettably, up to at least PowerShell (Core) 7.3.6 (unlike in Windows PowerShell), attempting to do so is broken - see GitHub issue #11856.A simple test to see if loading succeeded is to execute the following PowerShell statement in your script, after the
Add-Type
call:As an aside:
Unless you need to execute a piece of PowerShell code, it is better to use the
.AddCommand()
rather than the.AddScript()
SDK method:See this answer for an explanation.