Razor ViewEngine临时编译.cs文件

本文关键字:cs 文件 编译 ViewEngine Razor | 更新日期: 2023-09-27 18:20:17

在Razor ViewEngine中调用Parse方法时,编译错误会作为TemplateCompilationException抛出,其中包含错误列表。这些错误指的是临时文件名,但在您可以访问这些文件之前,这些文件已被删除。

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();
    }

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

Razor ViewEngine临时编译.cs文件

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

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

这将允许在引发异常时读取源代码。

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

v3(当前版本3.0.7beta)在Nuget(Install-Package RazorEngine)上可用。上周末我的目标是RTW,但一直没有抽出时间。

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