3

在 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我想获得尽可能多的详细信息以显示给用户。

4

2 回答 2

4

当前的 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,但从未实现。

于 2012-01-20T12:26:16.557 回答
3

RazorEngine 的 TemplateCompilationException 是一个包装 CompilerErrorCollection 的类,该 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()
于 2012-01-16T16:05:53.447 回答