0

因此,我正在使用 MSScriptControl 在我的应用程序中运行一些 javascript,并且我希望能够获取有关脚本可能导致的任何错误的一些信息。

MSScriptControl.ScriptControlClass script = new MSScriptControl.ScriptControlClass();
try
{
    script.Language = "JScript";
    script.Timeout = 15000;
    script.Eval(Code);
}
catch (Exception ex)
{
    MSScriptControl.Error err = script.Error;
    ret = new Exception("Error on line: " + err.Line + ", Description: " + err.Description);
}

该代码在我的开发机器上运行良好,一个 Windows 7 机器,并给了我一个错误的行号。所以我很高兴地将它发布并推送到生产机器上,它总是告诉我错误发生在第 0 行并且没有提供描述。

我尝试去http://www.microsoft.com/download/en/details.aspx?id=1949下载最新版本,但安装它没有效果。我还将属性 Embed Interop Types 设置为 false,并将我自己的 msscript.ocx 文件复制到 Windows 2008 服务器的 system32 目录中,但这些尝试都没有解决任何问题。

有人有什么建议吗?

4

2 回答 2

2

如果您想在没有任何第三方或“组件”外部依赖项的所有本机 c# 中执行此操作,请使用CodeDomProvider带有小型 JScript 引导程序的 a,如下所示:

private static readonly MethodInfo eval = CodeDomProvider
        .CreateProvider("JScript")
        .CompileAssemblyFromSource(new CompilerParameters(), "package e{class v{public static function e(e:String):Object{return eval(e);}}}")
        .CompiledAssembly
        .GetType("e.v")
        .GetMethod("e");

private static object JsEval(string jscript)
{
    try
    {
        return eval.Invoke(null, new[] { jscript });
    }
    catch (Exception ex)
    {
        return ex;
    }
}

它创建了一个JsEval(string)方法,您可以在代码中的任何位置使用该方法将字符串“评估”为 JavaScript(以及 JScript)......所以调用:

MessageBox.Show("" + JsEval("2 + 2")); // 4
MessageBox.Show("" + JsEval("(function(){ return 3+7; })();")); // 10
MessageBox.Show("" + JsEval("function yay(a) { return a + 1; } yay(2);")); // 3

根据您的用途,您可能不想静态实例化这些成员。如果你想操作复杂的对象,你需要创建一个包装器来反射性地提取数据(或者你可以转换为适当的 JScript 对应物,但我从未尝试过,因为你必须包含 JScript 程序集)。

这是一个包装器类的示例,它可以完成 JavaScript 允许您在本地执行的所有操作,添加更多高级功能可能会很麻烦,因此您最好将成员提取到字典/哈希表中,或者序列化和在另一端反序列化

private class JsObjectWrapper : IEnumerable
{
    public readonly object jsObject;
    private static PropertyInfo itemAccessor = null;
    private static MethodInfo getEnumerator = null;

    public JsObjectWrapper(object jsObject)
    {
        this.jsObject = jsObject;

        if (itemAccessor == null) 
        {
            itemAccessor = jsObject.GetType().GetProperty("Item", new Type[] { typeof(string) });
        }

        if (getEnumerator == null)
        {
            getEnumerator = jsObject.GetType().GetInterface("IEnumerable").GetMethod("GetEnumerator");
        }
    }

    public object this[string key]
    {
        get { return itemAccessor.GetValue(jsObject, new object[] { key }); }
        set { itemAccessor.SetValue(jsObject, value, new object[] { key }); } 
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        return (IEnumerator)getEnumerator.Invoke(jsObject, null);
    }
}

您可以通过执行以下操作看到这一点:

var jsObj = JsEval("var x = { a:7, b:9 };");
var csObj = new JsObjectWrapper(jsObj);

MessageBox.Show("a: " + csObj["a"]);  // a: 7
MessageBox.Show("b: " + csObj["b"]);  // b: 9

csObj["yay!"] = 69;

foreach (string key in csObj)
{
    MessageBox.Show("" + key + ": " + csObj[key]); // "key": "value"
}

我个人曾经使用过与此类似的代码,效果非常好,并且可以保证它在服务器环境中的可用性和可运行性。我希望这会有所帮助-ck

于 2012-03-20T15:35:49.957 回答
1

关于你面临的问题只是一些想法:

  • 根据您提供的链接,此控件既不支持 Windows 7 也不支持 Windows 2008
  • 这可能是关于 COM/UAC 等的安全问题。
  • 如果您为 AnyCPU 编译,可能会因为位数而出现问题,请尝试使用 x86

关于可能的替代方案:

  • 使用JScript,您可以很容易地构建一个评估器,它在 .NET 4 运行的任何地方(包括 Windows Server 2008)都受支持。
  • 使用JInt作为 JavaScript 解释器
于 2012-03-19T20:24:17.877 回答