从PowerShell调用时无法在dll中转换透明代理,但在C#控制台应用程序中成功

时间:2011-09-24 02:34:01

标签: c# powershell appdomain marshalbyrefobject

我正在尝试创建一个open source library,它会生成一个新的AppDomain并在其中运行一个PowerShell脚本。我有一个静态方法,它取名为powershell文件的名称和AppDomain的名称。从C#控制台应用程序调用时,该方法成功执行,但不是PowerShell

我知道由于this entry in the fusionlog而在第二个应用域中加载了dll。

类声明和构造函数看起来像这样。

public class AppDomainPoshRunner : MarshalByRefObject{

    public AppDomainPoshRunner (){
        Console.WriteLine("Made it here.");
    }
}

当我调用CreateInstanceFromAndUnwrap时,无论是从C#控制台应用程序还是从PowerShell应用程序运行dll,构造函数中的消息都会输出。

当我在下面的静态方法中将CreateInstanceFromAndUnwrap返回的值转换为 AppDomainPoshRunner 时,会发生失败。

    public static string[] RunScriptInAppDomain(string fileName, string appDomainName = "Unamed")
    {
        var assembly = Assembly.GetExecutingAssembly();

        var setupInfo = new AppDomainSetup
                            {
                                ApplicationName = appDomainName,
                                // TODO: Perhaps we should setup an even handler to reload the AppDomain similar to ASP.NET in IIS.
                                ShadowCopyFiles = "true"
                            };
        var appDomain = AppDomain.CreateDomain(string.Format("AppDomainPoshRunner-{0}", appDomainName), null, setupInfo);
        try {
            var runner = appDomain.CreateInstanceFromAndUnwrap(assembly.Location, typeof(AppDomainPoshRunner).FullName);
            if (RemotingServices.IsTransparentProxy(runner))
                Console.WriteLine("The unwrapped object is a proxy.");
            else
                Console.WriteLine("The unwrapped object is not a proxy!");  
            Console.WriteLine("The unwrapped project is a {0}", runner.GetType().FullName);
            /* This is where the error happens */
            return ((AppDomainPoshRunner)runner).RunScript(fileName);
        }
        finally
        {
            AppDomain.Unload(appDomain);
        }
    }

在PowerShell中运行时,我会收到InvalidCastExcception消息无法将透明代理转换为JustAProgrammer.ADPR.AppDomainPoshRunner类型。

我做错了什么?

2 个答案:

答案 0 :(得分:2)

我遇到了同样的问题:我创建了沙箱,只有执行权限(最小的权限)可以在非常有限的环境中执行不受信任的代码。所有在C#应用程序中工作得很好,但是当起始点是创建.NET COM对象的vbs脚本时,它没有工作(相同的强制转换异常)。我认为PowerShell也使用COM。我找到了使用AppDomain.DoCallBack的解决方法,这避免了从appdomain获取代理。这是代码。 如果您找到更好的选择,请发布。在GAC注册对我来说不是一个好的解决方案......

    class Test
    {
        /*
         create appdomain as usually
        */

        public static object Execute(AppDomain appDomain, Type type, string method, params object[] parameters)
        {
            var call = new CallObject(type, method, parameters);
            appDomain.DoCallBack(call.Execute);
            return call.GetResult();
        }
    }
    [Serializable]
    public class CallObject
    {
        internal CallObject(Type type, string method, object[] parameters)
        {
            this.type = type;
            this.method = method;
            this.parameters = parameters;
        }

        [PermissionSet(SecurityAction.Assert, Unrestricted = true)]
        public void Execute()
        {
            object instance = Activator.CreateInstance(this.type);
            MethodInfo target = this.type.GetMethod(this.method);
            this.result.Data = target.Invoke(instance, this.parameters);
        }

        internal object GetResult()
        {
            return result.Data;
        }

        private readonly string method;
        private readonly object[] parameters;
        private readonly Type type;
        private readonly CallResult result = new CallResult();

        private class CallResult : MarshalByRefObject
        {
            internal object Data { get; set; }
        }
    }

答案 1 :(得分:0)

这听起来非常像加载上下文问题。类型标识不仅仅与物理组件文件有关;它也是关于它的装载方式和位置。这是来自Suzanne Cook的一篇旧博客文章,你可能需要阅读十五次才能开始理解你的问题。

选择绑定上下文

http://blogs.msdn.com/b/suzcook/archive/2003/05/29/57143.aspx

在您说“但它在控制台应用程序中运行”之前,请记住,当从powershell运行它时,您在调用appdomain的上下文,探测路径,身份等方面拥有完全不同的鱼。

祝你好运!