// Copyright (c) Microsoft Corporation. All rights reserved. See License.txt in the project root for license information.
using System.Collections.Generic;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.IO;
using System.Threading;
using System.Web.Razor.Generator;
using System.Web.Razor.Parser;
using System.Web.Razor.Text;
namespace System.Web.Razor
{
///
/// Entry-point to the Razor Template Engine
///
public class RazorTemplateEngine
{
public static readonly string DefaultClassName = "Template";
public static readonly string DefaultNamespace = String.Empty;
///
/// Constructs a new RazorTemplateEngine with the specified host
///
/// The host which defines the environment in which the generated template code will live
public RazorTemplateEngine(RazorEngineHost host)
{
if (host == null)
{
throw new ArgumentNullException("host");
}
Host = host;
}
///
/// The RazorEngineHost which defines the environment in which the generated template code will live
///
public RazorEngineHost Host { get; private set; }
public ParserResults ParseTemplate(ITextBuffer input)
{
return ParseTemplate(input, null);
}
///
/// Parses the template specified by the TextBuffer and returns it's result
///
///
/// IMPORTANT: This does NOT need to be called before GeneratedCode! GenerateCode will automatically
/// parse the document first.
///
/// The cancel token provided can be used to cancel the parse. However, please note
/// that the parse occurs _synchronously_, on the callers thread. This parameter is
/// provided so that if the caller is in a background thread with a CancellationToken,
/// it can pass it along to the parser.
///
/// The input text to parse
/// A token used to cancel the parser
/// The resulting parse tree
[SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope", Justification = "Input object would be disposed if we dispose the wrapper. We don't own the input so we don't want to dispose it")]
public ParserResults ParseTemplate(ITextBuffer input, CancellationToken? cancelToken)
{
return ParseTemplateCore(input.ToDocument(), cancelToken);
}
// See ParseTemplate(ITextBuffer, CancellationToken?),
// this overload simply wraps a TextReader in a TextBuffer (see ITextBuffer and BufferingTextReader)
public ParserResults ParseTemplate(TextReader input)
{
return ParseTemplate(input, null);
}
[SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope", Justification = "Input object would be disposed if we dispose the wrapper. We don't own the input so we don't want to dispose it")]
public ParserResults ParseTemplate(TextReader input, CancellationToken? cancelToken)
{
return ParseTemplateCore(new SeekableTextReader(input), cancelToken);
}
protected internal virtual ParserResults ParseTemplateCore(ITextDocument input, CancellationToken? cancelToken)
{
// Construct the parser
RazorParser parser = CreateParser();
Debug.Assert(parser != null);
return parser.Parse(input);
}
public GeneratorResults GenerateCode(ITextBuffer input)
{
return GenerateCode(input, null, null, null, null);
}
public GeneratorResults GenerateCode(ITextBuffer input, CancellationToken? cancelToken)
{
return GenerateCode(input, null, null, null, cancelToken);
}
public GeneratorResults GenerateCode(ITextBuffer input, string className, string rootNamespace, string sourceFileName)
{
return GenerateCode(input, className, rootNamespace, sourceFileName, null);
}
///
/// Parses the template specified by the TextBuffer, generates code for it, and returns the constructed CodeDOM tree
///
///
/// The cancel token provided can be used to cancel the parse. However, please note
/// that the parse occurs _synchronously_, on the callers thread. This parameter is
/// provided so that if the caller is in a background thread with a CancellationToken,
/// it can pass it along to the parser.
///
/// The className, rootNamespace and sourceFileName parameters are optional and override the default
/// specified by the Host. For example, the WebPageRazorHost in System.Web.WebPages.Razor configures the
/// Class Name, Root Namespace and Source File Name based on the virtual path of the page being compiled.
/// However, the built-in RazorEngineHost class uses constant defaults, so the caller will likely want to
/// change them using these parameters
///
/// The input text to parse
/// A token used to cancel the parser
/// The name of the generated class, overriding whatever is specified in the Host. The default value (defined in the Host) can be used by providing null for this argument
/// The namespace in which the generated class will reside, overriding whatever is specified in the Host. The default value (defined in the Host) can be used by providing null for this argument
/// The file name to use in line pragmas, usually the original Razor file, overriding whatever is specified in the Host. The default value (defined in the Host) can be used by providing null for this argument
/// The resulting parse tree AND generated Code DOM tree
[SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope", Justification = "Input object would be disposed if we dispose the wrapper. We don't own the input so we don't want to dispose it")]
public GeneratorResults GenerateCode(ITextBuffer input, string className, string rootNamespace, string sourceFileName, CancellationToken? cancelToken)
{
return GenerateCodeCore(input.ToDocument(), className, rootNamespace, sourceFileName, cancelToken);
}
// See GenerateCode override which takes ITextBuffer, and BufferingTextReader for details.
public GeneratorResults GenerateCode(TextReader input)
{
return GenerateCode(input, null, null, null, null);
}
public GeneratorResults GenerateCode(TextReader input, CancellationToken? cancelToken)
{
return GenerateCode(input, null, null, null, cancelToken);
}
public GeneratorResults GenerateCode(TextReader input, string className, string rootNamespace, string sourceFileName)
{
return GenerateCode(input, className, rootNamespace, sourceFileName, null);
}
[SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope", Justification = "Input object would be disposed if we dispose the wrapper. We don't own the input so we don't want to dispose it")]
public GeneratorResults GenerateCode(TextReader input, string className, string rootNamespace, string sourceFileName, CancellationToken? cancelToken)
{
return GenerateCodeCore(new SeekableTextReader(input), className, rootNamespace, sourceFileName, cancelToken);
}
protected internal virtual GeneratorResults GenerateCodeCore(ITextDocument input, string className, string rootNamespace, string sourceFileName, CancellationToken? cancelToken)
{
className = (className ?? Host.DefaultClassName) ?? DefaultClassName;
rootNamespace = (rootNamespace ?? Host.DefaultNamespace) ?? DefaultNamespace;
// Run the parser
RazorParser parser = CreateParser();
Debug.Assert(parser != null);
ParserResults results = parser.Parse(input);
// Generate code
RazorCodeGenerator generator = CreateCodeGenerator(className, rootNamespace, sourceFileName);
generator.DesignTimeMode = Host.DesignTimeMode;
generator.Visit(results);
// Post process code
Host.PostProcessGeneratedCode(generator.Context);
// Extract design-time mappings
IDictionary designTimeLineMappings = null;
if (Host.DesignTimeMode)
{
designTimeLineMappings = generator.Context.CodeMappings;
}
// Collect results and return
return new GeneratorResults(results, generator.Context.CompileUnit, designTimeLineMappings);
}
protected internal virtual RazorCodeGenerator CreateCodeGenerator(string className, string rootNamespace, string sourceFileName)
{
return Host.DecorateCodeGenerator(
Host.CodeLanguage.CreateCodeGenerator(className, rootNamespace, sourceFileName, Host));
}
protected internal virtual RazorParser CreateParser()
{
ParserBase codeParser = Host.CodeLanguage.CreateCodeParser();
ParserBase markupParser = Host.CreateMarkupParser();
return new RazorParser(Host.DecorateCodeParser(codeParser),
Host.DecorateMarkupParser(markupParser))
{
DesignTimeMode = Host.DesignTimeMode
};
}
}
}