【问题标题】:Razor ViewEngine Temporary compilation .cs filesRazor ViewEngine 临时编译.cs文件
【发布时间】:2012-01-13 14:22:31
【问题描述】:

在 Razor ViewEngine 中调用 Parse 方法时,编译错误将作为包含错误列表的 TemplateComplilationException 引发。这些错误指的是临时文件名,但这些文件在您访问它们之前已被删除。

static void Main(string[] args)  
{
    var service = TemplateServiceFactory.CreateTemplateService(Language.CSharp, true);
    string result = "";
    try
    {
       result = service.Parse("Hello @DateTime.NowXX ");
    }
    catch (TemplateCompilationException ex)
    {
      foreach (var error in ex.Errors)
         if (!string.IsNullOrEmpty(error.FileName))
             Console.WriteLine( File.ReadAllText( error.FileName ));
    }  //                                         ^^^^ File does not exist!

    Console.WriteLine( result );       
    Console.ReadKey();
    }

(一点背景) 我正在使用没有 MVC 的 Razor 引擎“独立”。当我拨打Parse 时,我希望获得尽可能多的详细信息以显示给用户。

【问题讨论】:

  • 您是否尝试过添加FileSystemWatcher 以将.cs 文件从执行/编译目录中复制出来?
  • 无法让 FileSystemWatcher 为我触发。

标签: c# .net razor razorengine


【解决方案1】:

当前的 v2.1 版本不提供吐出源代码的功能。新的 v3 代码库中有一个调试功能,允许推出源代码。默认情况下它不会这样做,因为我试图使代码尽可能高效(并且生成两次代码(一次作为 CodeDom,一次作为字符串)并不理想)。您需要在配置中启用 Debug 标志:

var config = new TemplateServiceConfiguration { Debug = true };
var service = new TemplateService(config);

这将允许在抛出异常时读取源代码。

兴趣点,通过使用 v3 代码库测试 Roslyn 编译器基础架构,它接受字符串源而不是 CodeDom,因此我可能会在未来进行更改以直接使用它而不是 CodeDom - 这反过来意味着我们有直接访问源代码,而不必担心启用任何可能会被弃用的Debug 标志。

v3(当前为 v3.0.7beta)在 Nuget (Install-Package RazorEngine) 上可用。上周末我的目标是 RTW,但一直没有实现。

【讨论】:

    【解决方案2】:

    RazorEngine 的 TemplateCompilationException 是一个包装 CompilerErrorCollection 的类,其中包含 CompilerError 对象,因此您可能从 TemplateCompilationException CompilerError 对象中获得的最详细信息是它们各自的属性,这似乎足以进行调试。考虑并尝试这个例子

    try
    {
        Razor.Parse("My erroneous @DateTime.Now.foo()");
    }
    catch(TemplateCompilationException ex)
    {
        foreach(var error in ex.Errors)
        {
            StringBuilder sb = new StringBuilder();
            sb.AppendLine("Compile Error Num: \t" + error.ErrorNumber);
            sb.AppendLine("Error Text:\n\t" + error.ErrorText);
            Console.WriteLine(sb.ToString());
        }
        Console.WriteLine("Erroneous Template:\n\t" + ex.Template);
    }
    

    当我运行我的示例时,这就是我得到的,它告诉您遇到的错误,您可以转储模板数据以供您的用户参考。

    Compile Error Num:   CS1061
    Error Text:
      'System.DateTime' does not contain a definition for 'foo' and no 
      extension method     'foo' accepting a first argument of type 
      'System.DateTime' could be found (are you missing a using directive 
      or an assembly reference?)
    
    Erroneous Template:
        My erroneous @DateTime.Now.foo()
    

    【讨论】:

    • 如果使用 OP 要求的这种方法出现解析异常,您是否能够获取临时 .cs 文件的内容?
    • 他们的剃须刀实施也遇到了问题,我的回答应该会有所帮助。由于我们正在讨论临时编译文件,您将如何在构建和/或部署后获取内容?我认为找到实际问题并解决它(这就是我回答这个问题的原因)比挖掘临时文件更重要。
    • 我认为 OP 的实现没有任何问题。如果 Razor 模板中没有错误,它会和你的一样正常工作。顺便说一句,在内部,您的 Razor.Parse 完全符合 OP 正在使用的功能。他正在尝试做的是获取编译期间使用的临时文件。
    • 谢谢,是的,我认为你是对的。 OP 正在寻找一种方法来尽可能多地获取有关编译错误的数据。我会更新我的答案以反映这一点。
    • @Darin 我最初的问题是“我如何获得 .CS”文件,但 jlafay 回答了我的问题(不是我的问题),即“到底出了什么问题!”我稍后会尝试这个例子,并在一两天内接受答案(只是为了给大家时间回复)
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-01-07
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-07-14
    • 1970-01-01
    相关资源
    最近更新 更多