在MIME类型检查期间,应用程序池与URLMoniker urlmon.dll崩溃

时间:2017-05-13 11:08:25

标签: asp.net-mvc mime unmanaged application-pool urlmon

我的应用程序是部署在Windows Server 2012 R2上的ASP.NET MVC网站。我使用内置的Windows库URLMoniker - urlmon.dll来获取文件的MIME类型。我将其文件路径传递给GetMimeType方法。我面临的问题是,当我使用Visual Studio调试它时,它返回该文件的mime类型(在我的测试用例中,对于txt文件,它返回" application / octet-stream")。但是在生产服务器上部署之后,应用程序池意外崩溃,并且日志文件中没有日志,我花了3天时间才开始使用这个代码块(借助其他日志条目)。

    private string GetMimeType(string filePath)
    {
        log.Info("Getting mime type for " + filePath);

        byte[] buffer = new byte[256];
        using (FileStream fs = new FileStream(filePath, FileMode.Open))
        {
            if (fs.Length >= 256)
                fs.Read(buffer, 0, 256);
            else
                fs.Read(buffer, 0, (int)fs.Length);
        }
        log.Info("Done reading into byte array for " + filePath);
        try
        {
            System.UInt32 mimetype;
            FindMimeFromData(0, null, buffer, 256, null, 0, out mimetype, 0);
            System.IntPtr mimeTypePtr = new IntPtr(mimetype);
            string mime = Marshal.PtrToStringUni(mimeTypePtr);
            Marshal.FreeCoTaskMem(mimeTypePtr);
            log.Info("Got mime type for " + filePath + " " + mime);
            return mime;
        }
        catch (Exception e)
        {
            log.Error("Cannot get mime type for file " + filePath, e);
            return "unknown/unknown";
        }
    }

即使是事件查看器也没有显示应用程序池崩溃的任何原因,除此之外:为应用程序池服务的进程" SampleApp"与Windows Process Activation Service发生致命的通信错误。流程ID是' yyyy'。

在做了我能找到的详细研究之后,我有以下文章可能会提供一些解决方案,但我仍然无法找到问题的确切原因。

  1. https://www.experts-exchange.com/questions/24821266/Marshal-FreeCoTaskMem-crashing-application-in-x64-but-not-x86.html

  2. 这里也遇到了类似的问题:https://superuser.com/questions/568806/iis-worker-process-crashing-without-stack-trace-what-else-can-i-try

  3. 它说urlmon.dll可能没有在系统上注册,但我已在Windows注册表中检查并注册了它。事实上,该库是最新的,我需要在生产服务器上应用任何注册表更改之前找到根本原因。用新版本替换dll,修复注册表问题是我最后的选择。 https://answers.microsoft.com/en-us/ie/forum/ie8-windows_7/urlmondll-causing-many-programs-to-crash/cda9a6cb-cf51-499c-8855-45c97110eafe

  4. https://social.technet.microsoft.com/Forums/windows/en-US/c3f1517b-a8c5-422e-9317-2f539715badc/ie11-x64-on-win7-crash-ntdlldll-urlmondll?forum=w7itprohardware

1 个答案:

答案 0 :(得分:1)

你的问题是" FindMimeFromData"方法 ! 它在某些地方运行良好,它导致iis进程崩溃在其他地方。看看:https://stackoverflow.com/a/18554243/4257500 您需要更改" FindMimeFromData"的声明。 as:

[DllImport("urlmon.dll", CharSet = CharSet.Unicode, ExactSpelling = true, SetLastError = false)]static extern int FindMimeFromData(IntPtr pBC,
[MarshalAs(UnmanagedType.LPWStr)] string pwzUrl,
[MarshalAs(UnmanagedType.LPArray, ArraySubType=UnmanagedType.I1, SizeParamIndex=3)] 
byte[] pBuffer,
int cbSize,
[MarshalAs(UnmanagedType.LPWStr)] string pwzMimeProposed,
int dwMimeFlags,
out IntPtr ppwzMimeOut,
int dwReserved);

并且您还应该进行一些更改来调用此函数。例如:

 IntPtr mimeTypePtr;
 FindMimeFromData(IntPtr.Zero, null,file, 256, null, 0,out mimeTypePtr, 0);
 var mime = Marshal.PtrToStringUni(mimeTypePtr);