ASP.NET在Bundling中使用嵌入式资源

Pas*_*ger 15 c# asp.net-mvc virtualpathprovider embedded-resource bundling-and-minification

我正在尝试实现一种通用方法,为我的Web解决方案中的不同程序集提供使用嵌入式资源中嵌入的JavaScript和CSS文件的可能性.此博客文章显示了使用VirtualPathProvider的技术.这很好,但VirtualPathProvider需要包含在包含嵌入资源的每个程序集中.

我试图从博客文章中增强VirtualPathProvider,以便可以将程序集传递给它并从程序集中加载资源:

public EmbeddedVirtualPathProvider(VirtualPathProvider previous, Assembly assembly)
{
    this.previous = previous;
    this.assembly = assembly;
}
Run Code Online (Sandbox Code Playgroud)

初始化时,它从传递的程序集中读取所有嵌入的资源:

protected override void Initialize()
{
    base.Initialize();

    this.assemblyResourceNames = this.assembly.GetManifestResourceNames();
    this.assemblyName = this.assembly.GetName().Name;
}
Run Code Online (Sandbox Code Playgroud)

GetFile从传递的程序集中读取内容:

public override VirtualFile GetFile(string virtualPath)
{
    if (IsEmbeddedPath(virtualPath))
    {
        if (virtualPath.StartsWith("~", System.StringComparison.OrdinalIgnoreCase))
        {
            virtualPath = virtualPath.Substring(1);
        }

        if (!virtualPath.StartsWith("/", System.StringComparison.OrdinalIgnoreCase))
        {
            virtualPath = string.Concat("/", virtualPath);
        }

        var resourceName = string.Concat(this.assembly.GetName().Name, virtualPath.Replace("/", "."));
        var stream = this.assembly.GetManifestResourceStream(resourceName);

        if (stream != null)
        {
            return new EmbeddedVirtualFile(virtualPath, stream);
        }
        else
        {
            return _previous.GetFile(virtualPath);
        }
    }
    else
        return _previous.GetFile(virtualPath);
}
Run Code Online (Sandbox Code Playgroud)

检查资源是否是此程序集的嵌入资源是通过检查Initialize方法中读取的资源名称:

private bool IsEmbeddedPath(string path)
{
    var resourceName = string.Concat(this.assemblyName, path.TrimStart('~').Replace("/", "."));
    return this.assemblyResourceNames.Contains(resourceName, StringComparer.OrdinalIgnoreCase);
}
Run Code Online (Sandbox Code Playgroud)

我将EmbeddedVirtualPathProvider类移动到主Web项目(ProjectA),因此它不需要包含在包含嵌入资源的每个程序集中,并使用以下代码注册它Global.asax:

HostingEnvironment.RegisterVirtualPathProvider(
    new EmbeddedVirtualPathProvider(
        HostingEnvironment.VirtualPathProvider,
        typeof(ProjectB.SomeType).Assembly));
Run Code Online (Sandbox Code Playgroud)

在包含嵌入资源(ProjectB)的项目中,我仍然在以下位置创建以下包PostApplicationStartMethod:

 BundleTable.Bundles.Add(new ScriptBundle("~/Embedded/Js")
     .Include("~/Scripts/SomeFolder/MyScript.js")
 );
Run Code Online (Sandbox Code Playgroud)

Scripts/MyScript.js 是ProjectB中的嵌入式资源.

有了这个,我收到以下异常:

目录'C:\ webs\ProjectA\Scripts\SomeFolder \'不存在.无法开始监视文件更改.

更新此Gist中 可用的完整堆栈跟踪.

更新 此外,VirtualPathProvider本身似乎工作正常.如果我直接加载文件而不是通过bundle web.config加载并在其中设置以下条目,则从ProjectB加载嵌入的javascript:

<system.webServer>
  <handlers>
    <add name="MyStaticFileHandler" path="*.js" verb="GET,HEAD" type="System.Web.StaticFileHandler"/>
  </handlers>
</system.webServer>
Run Code Online (Sandbox Code Playgroud)

Cyr*_*and 1

当 ASP.net 优化创建包时,它会调用GetCacheDependency脚本的虚拟目录。您的实现仅检查虚拟文件,对于虚拟目录,它依赖于检查目录是否存在和失败的GetCacheDependency基础。VirtualPathProvider

要解决此问题,您必须检查路径是否是脚本之一的目录,并为GetCacheDependency.

要安全地确定是否virtualPath是捆绑包目录,您可以使用 BundleTable.Bundles集合或使用约定(即:每个捆绑包都应以 ~/Embedded 开头)。

public override CacheDependency GetCacheDependency(
    string virtualPath, 
    IEnumerable virtualPathDependencies, 
    DateTime utcStart)
{
    // if(virtualPath.StartsWith("~/Embedded"))
    if(BundleTables.Bundles.Any(b => b.Path == virtualPath))
    {
        return null; 
    }
    if (this.IsEmbeddedPath(virtualPath))
    {
        return null;
    }
    else
    {
        return this._previous
                   .GetCacheDependency(virtualPath, virtualPathDependencies, utcStart);
    }
}
Run Code Online (Sandbox Code Playgroud)