ReadAsMultipartAsync和MultiPartFormDataStreamProvider出错

时间:2014-12-22 18:25:36

标签: c# .net asp.net-mvc async-await asp.net-web-api

我正在尝试将文件异步上传到我的Web服务器(ASP.NET MVC)。特别是我的web api端点。我遵循了几个教程,但我一直收到null参数的错误。

当我继承MultipartFormDataStreamProvider时会发生这种情况。方法GetLocalFileName首先使用正确的HttpContentHeaders正确调用,但随后会使用null标题第二次调用。

使用我设置的文件名成功保存文件,但是当我收到异常时,我无法知道生成的名称是什么。

但是,如果我只使用基础MultipartFormDataStreamProvider类,则文件保存完好,没有错误,但在这种情况下我无法控制文件名。

这是api方法:

[HttpPost]
    public async Task<FilesUploadResult> Save()
    {
        // Check if the request contains multipart/form-data.
        if (!Request.Content.IsMimeMultipartContent())
        {
            var ret = new FilesUploadResult(HttpStatusCode.UnsupportedMediaType, null);
            ret.Message = "Unsupported Media Type. Request must contain multipart/form-data form.";
            return ret;
        }

        var path = UploadsFolderPath;

        // to-do: this works but for some reason makes a call with empty headers.
        var provider = new SimpleMultipartFormDataStreamProvider(path);

        // this provider saves the file with an arbritrary name
        //var provider = new MultipartFormDataStreamProvider(path);

        try
        {
            // Read the form data.
            await Request.Content.ReadAsMultipartAsync(provider as MultipartFormDataStreamProvider);

            var files = from f in provider.FileData
                        let info = new FileInfo(f.LocalFileName)
                        select new FileUploadDescription(info.Name, path + "\\" + info.Name, info.Length, HttpStatusCode.Created);

            var ret = new FilesUploadResult(HttpStatusCode.OK, files);
            return ret;
        }
        catch (Exception ex)
        {
            var ret = new FilesUploadResult(HttpStatusCode.InternalServerError, null);
            ret.Message = ex.Message;
            return ret;
        }
    }

这是我SimpleMultipartFormDataStreamProvider的代码:

public class SimpleMultipartFormDataStreamProvider : MultipartFormDataStreamProvider
{
    public SimpleMultipartFormDataStreamProvider(string path) : base(path) { }

    public override string GetLocalFileName(HttpContentHeaders headers)
    {

        // make sure the headers are valid
        if (headers == null)
        {

            throw new ArgumentNullException("headers");
        }

        // filename
        var filename = CleanString(headers.ContentDisposition.FileName); ;

        // create the local file name
        var localFileName = string.Format("_{0}_{1}", GetRandomName(), filename);

        //this is here because Chrome submits files in quotation marks which get treated as part of the filename and get escaped
        return localFileName.Replace("\"", string.Empty);
    }

    private string GetRandomName()
    {
        return GuidHelper.ShortUniqueId(16);
    }

    private static string CleanString(string str)
    {
        if (string.IsNullOrWhiteSpace(str))
            return string.Empty;

        var sb = new StringBuilder();
        foreach (char c in str)
        {
            if ((c >= '0' && c <= '9') || (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z') || c == '.' || c == '_')
            {
                sb.Append(c);
            }
        }
        return sb.ToString();
    }
}

有人可以告诉我为什么我要两次调用GetLocalFileName,第二次调用空标题吗?

3 个答案:

答案 0 :(得分:2)

即使我在MVC4中遇到过这样的行为,它也会提交2个文件,其中一个是空的,所以最好检查文件是否有任何内容,然后继续保存文件。 可以有多种方法:  其中一个是从Request.Files获取所有文件的列表(这是一个字符串数组)

      foreach (string fileName in Request.Files)
            {
                HttpPostedFileBase file = Request.Files[fileName];
                //Save file content goes here
                if (file != null && file.ContentLength > 0)
                 {
                     //logic to save the content
                 }
             }

答案 1 :(得分:2)

对于多部分请求中的每个文件,都会调用

GetLocalFileName一次。如果您在同一请求中发送两个文件,则将调用GetLocalFileName两次。我认为您应该检查如何调用该服务并使用fiddler或类似的东西来查看您是否真的发送了两个文件。如果您在同一请求中发送多个部分,请尝试指定每个部分的ContentType。

答案 2 :(得分:2)

我自己想出来了。

问题是我实现的用于子类MultipartDataStreamProvider的类是在另一个项目中。

该项目引用了.Net .Http程序集,这些程序集与Web Api项目中的程序集不同。

我唯一做的就是将CustomMultipartDataStreamProvider类移动到同一个项目中,一切都运行良好。