Razor ViewEngine临时编译.cs文件

时间:2012-01-13 14:22:32

标签: c# .net razor razorengine

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

2 个答案:

答案 0 :(得分: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,但从未接触过它。

答案 1 :(得分:3)

RazorEngine的TemplateCompilationException是一个包装了包含CompilerError个对象的CompilerErrorCollection的类,因此您可能从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()