检查上传文件的类型

时间:2012-09-16 04:51:59

标签: c# asp.net file-upload

如何在ASP.NET C#网页中检查使用FileUploader控件上传的文件的文件类型?

  1. 我尝试检查文件扩展名,但是当JPEG图像(例如Leonardo.jpg)重命名为具有PDF扩展名(例如Leonardo.pdf)时,它显然会失败。

  2. 我试过

    FileUpload1.PostedFile.ContentType.ToLower().Equals("application/pdf")
    

    但是这失败了,因为上面的代码与第一个代码的行为方式相同。

  3. 有没有其他方法可以检查实际的文件类型,而不仅仅是扩展名?

    我看了ASP.NET how to check type of the file type irrespective of extension

    编辑:我尝试使用stackoverflow中的一个帖子下面的代码。但这不起作用。对此有任何想法。

    /// <summary>
    /// This class allows access to the internal MimeMapping-Class in System.Web
    /// </summary>
    class MimeMappingWrapper
    {
      static MethodInfo getMimeMappingMethod;
    
        static MimeMappingWrapper() {
        // dirty trick - Assembly.LoadWIthPartialName has been deprecated
        Assembly ass = Assembly.LoadWithPartialName("System.Web");
        Type t = ass.GetType("System.Web.MimeMapping");
    
        getMimeMappingMethod t.GetMethod("GetMimeMapping", BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public));
    }
    
    /// <summary>
    /// Returns a MIME type depending on the passed files extension
    /// </summary>
    /// <param name="fileName">File to get a MIME type for</param>
    /// <returns>MIME type according to the files extension</returns>
    public static string GetMimeMapping(string fileName) {
        return (string)getMimeMappingMethod.Invoke(null, new[] { fileName });
    }
    }
    

3 个答案:

答案 0 :(得分:5)

不要使用File Extensions来计算MIME类型,而是使用“Winista”进行二进制分析。

假设某人使用exe扩展名重命名jpg。您仍然可以确定真实的文件格式。它不会检测swf或flv,而是几乎所有其他众所周知的格式,你可以使用十六进制编辑器来添加它可以检测的更多文件。

下载Winista:here

如果Winista无法检测到真实的文件格式,我已经使用了URLMon方法:

public class urlmonMimeDetect
{
    [DllImport(@"urlmon.dll", CharSet = CharSet.Auto)]
    private extern static System.UInt32 FindMimeFromData(
        System.UInt32 pBC,
        [MarshalAs(UnmanagedType.LPStr)] System.String pwzUrl,
        [MarshalAs(UnmanagedType.LPArray)] byte[] pBuffer,
        System.UInt32 cbSize,
        [MarshalAs(UnmanagedType.LPStr)] System.String pwzMimeProposed,
        System.UInt32 dwMimeFlags,
        out System.UInt32 ppwzMimeOut,
        System.UInt32 dwReserverd
    );

public string GetMimeFromFile(string filename)
{
    if (!File.Exists(filename))
        throw new FileNotFoundException(filename + " not found");

    byte[] buffer = new byte[256];
    using (FileStream fs = new FileStream(filename, FileMode.Open, FileAccess.Read))
    {
        if (fs.Length >= 256)
            fs.Read(buffer, 0, 256);
        else
            fs.Read(buffer, 0, (int)fs.Length);
    }
    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);
        return mime;
    }
    catch (Exception e)
    {
        return "unknown/unknown";
    }
}
}

从Winista方法内部,我回到URLMon:

   public MimeType GetMimeTypeFromFile(string filePath)
    {
        sbyte[] fileData = null;
        using (FileStream srcFile = new FileStream(filePath, FileMode.Open, FileAccess.Read))
        {
            byte[] data = new byte[srcFile.Length];
            srcFile.Read(data, 0, (Int32)srcFile.Length);
            fileData = Winista.Mime.SupportUtil.ToSByteArray(data);
        }

        MimeType oMimeType = GetMimeType(fileData);
        if (oMimeType != null) return oMimeType;

        //We haven't found the file using Magic (eg a text/plain file)
        //so instead use URLMon to try and get the files format
        Winista.MimeDetect.URLMONMimeDetect.urlmonMimeDetect urlmonMimeDetect = new Winista.MimeDetect.URLMONMimeDetect.urlmonMimeDetect();
        string urlmonMimeType = urlmonMimeDetect.GetMimeFromFile(filePath);
        if (!string.IsNullOrEmpty(urlmonMimeType))
        {
            foreach (MimeType mimeType in types)
            {
                if (mimeType.Name == urlmonMimeType)
                {
                    return mimeType;
                }
            }
        }

        return oMimeType;
    }

<强>更新

使用魔法计算出更多文件是FILE SIGNATURES TABLE

答案 1 :(得分:0)

检查名称或扩展名绝不是一个可靠的想法。您可以确定的唯一方法是您实际读取文件的内容。

即。如果要检查文件中的图像,则应尝试从文件中加载图像,如果失败,则可以确定它不是图像文件。这可以使用GDI对象轻松完成。

PDF文件也是如此。

结论是,不要依赖用户提供的名称或扩展名。

答案 2 :(得分:0)

您可以通过

检查FileApload中的文件类型

<强> ValidationExpression =“^ +(([PP] [日] [FF])|([JJ] [PP] [gG的])|([PP] [NN] [gG的]))。 )$“

例如:您可以为Rar文件类型等添加([rR] [aA] [rR]) ...