c# - 如何将文件上传到 Web API 服务器并将参数发送到操作?

标签 c# jquery file-upload asp.net-web-api2 multipartform-data

美好的一天!

我正在从事 ASP.NET Web API 2 项目。在某个时候需要上传一些文件。这些文件需要链接到某个 FileModel(我们自己的类)。因此,客户端需要将 IEnumerable 作为参数发送,将文件作为内容发送。因为它是一个 RESTful API,所以两者都必须在同一个请求中发送。

我们能想到的最好的是跟随 Controller Action :

public async Task<HttpResponseMessage> Add([FromUri] IEnumerable<FileModel> fileModels)
{
   // REQUEST INTEGRITY TESTING

   var streamProvider = new CustomMultipartFormDataStreamProvider(fileSavePath, fileModels);
   // Read the MIME multipart content using the stream provider we just created.
   var work = await Request.Content.ReadAsMultipartAsync(streamProvider).ContinueWith(async t =>
        {
            // SOME WORK DONE AFTER SAVING THE FILES TO THE HARD DRIVE
        }

}

问题如下:上传的文件带有“multipart/form-data”Content-Type header 。 在服务器端操作文件之前,我们需要知道 FileModels 的内容。如果我们使用MultipartFormDataStreamProvider,我们只能在文件已经保存到硬盘后才能访问非文件参数。

我们能找到的唯一解决方法是在 URL 中发送 IEnumerable 参数。但鉴于 URL 的最大长度有限,这不是一个可靠的方法。

问题是:有没有办法同时提交 IEnumerable< FileModel > fileModels 参数和 请求正文 中的文件并获得对 fileModels 的访问权访问文件之前的参数?我们还希望能够使用 HttpContext.Current.Request.Files.Count;

我们当前用于文件上传的 jQuery 如下所示(出于早期测试目的,它仅支持一个文件上传):

$('#Upload').click(function(e) {
            e.preventDefault();

            var headers = new Array();
            headers["SessionId"] = sessionId;

            var files = $('#fileInput').get(0).files;
            var formData = new FormData();
            formData.append("files", files[0]);

            var fileModel = $('#fileSubmission').serialize();

            $.ajax({
                url: "api/Submissions/Add/?" + fileModel,
                headers: headers,
                type: 'POST',
                data: formData,
                cache: false,
                contentType: false,
                processData: false,
                dataType: 'json'
            });
        });

非常感谢!

最佳答案

很抱歉回答晚了,但我们解决了问题(我忘了我没有在这里上传答案)。基本上我们所做的是在临时位置调用 ReadAsMultiPartAsync 方法,然后从请求中提取其他参数。之后,我们验证了输入并将文件从临时位置移动到永久位置。

如果您想查看代码,这就是我们特定示例的工作方式,我相信它可以非常简单地适应任何工作案例场景:

在客户端,我们有以下表单(是的,这个实现是为了演示目的,只支持发送一个文件......而且,输入类型="file"字段确实在表单之外; fileId在我们的案例中,文本输入是手动完成的,仅用于测试目的)

<input type="file" name="data" id="fileInput" multiple="multiple" />

<form id="fileSubmission">            
    <input type="text" width="10" onchange="getFileDetails()" autocomplete="off" placeholder="FileId" name="files[0].Id" id="fileId" /> 
    <input type="hidden" name="files[0].FileName" id="FileName"/>
    <input type="hidden" name="files[0].Extension" id="Extension"/>
    <input type="hidden" name="files[0].EntityId" id="EntityId"/>
    <br /><br />
    <input type="submit" id="Upload" value="Upload" />
</form>

其中 getFileDetails() 填充其他输入字段。此外,使用以下 jQuery/Javascript 将表单发送到服务器:

$('#Upload').click(function(e) {
            e.preventDefault();

            var courseId = $('#courseId').val();
            var fileId = $('#fileId').val();
            if (!courseId || !fileId) {
                return;
            }

            var headers = new Array();
            headers["SessionId"] = sessionId;
            headers["contentType"] = "application/json; charset=UTF-8";

            var formData = new FormData();
            var opmlFile = $('#fileInput').get(0).files;

            // this is like the model we're expecting on the server
            var files = [];
            files.push({ 'Id': $('#fileId').val(), 'OriginalFileName': opmlFile[0].name, 'FileName': $('#FileName').val(), 'Extension': $('#Extension').val(), 'EntityId': $('#EntityId').val() });

            formData.append("fileModels", JSON.stringify(files));
            formData.append("File_0", opmlFile[0]);


            $.ajax({
                url: "api/Courses/" + courseId + "/Submissions/Add/",
                headers: headers,
                type: 'POST',
                data: formData,
                cache: false,
                contentType: false,
                processData: false,
                dataType: 'json'
            });
        });

在服务器端,我们有以下内容:

// POST: api/Courses/{courseId}/Submissions/Add
[HttpPost]
[ValidateModelState]
[ValidateMimeMultipartContent]
[PermissionsAuthorize(CoursePermissions.CanCreateSubmissions)]
public async Task<HttpResponseMessage> Add(int courseId)
    {
        // the same as in the jQuery part
        const string paramName = "fileModels";

        // Put the files in a temporary location
        // this way we call ReadAsMultiPartAsync and we get access to the other data submitted
        var tempPath = HttpContext.Current.Server.MapPath("~/App_Data/Temp/" + Guid.NewGuid());
        Directory.CreateDirectory(tempPath);

        var streamProvider = new MultipartFormDataStreamProvider(tempPath);
        var readResult = await Request.Content.ReadAsMultipartAsync(streamProvider);

        if (readResult.FormData[paramName] == null)
        {
            // We don't have the FileModels ... delete the TempFiles and return BadRequest
            Directory.Delete(tempPath, true);
            return Request.CreateResponse(HttpStatusCode.BadRequest);
        }

        // The files have been successfully saved in a TempLocation and the FileModels are not null
        // Validate that everything else is fine with this command
        var fileModels = JsonConvert.DeserializeObject<IEnumerable<FileModelExtension>>(readResult.FormData[paramName]).ToList();

        // AT THIS POINT, ON THE SERVER, WE HAVE ALL THE FILE MODELS 
        // AND ALL THE FILES ARE SAVED IN A TEMPORARY LOCATION

        // NEXT STEPS ARE VALIDATION OF THE INPUT AND THEN 
        // MOVING THE FILE FROM THE TEMP TO THE PERMANENT LOCATION

        // YOU CAN ACCESS THE INFO ABOUT THE FILES LIKE THIS:
        foreach (var tempFile in readResult.FileData)
            {
                var originalFileName = tempFile.Headers.ContentDisposition.FileName.Replace("\"", string.Empty);

                var localTempPath = tempFile.LocalFileName;
            }

    }

我希望这对尝试使用 Post 请求一次向服务器提交文件和其他参数的任何人有所帮助! :)

注意:服务器上使用的一些属性是自定义的。 PermissionAuthorize、ValidateModelState 和 ValidateMimeMultiPartContent 是我们使用的自定义过滤器。后两者的实现受到 http://benfoster.io/blog/automatic-modelstate-validation-in-aspnet-mvc 的启发。

multipartcontent 属性只是检查 actionContext.Request.Content.IsMimeMultipartContent(),如下所示:

public class ValidateMimeMultipartContent : ActionFilterAttribute
{
    public override void OnActionExecuting(HttpActionContext actionContext)
    {
        if (!actionContext.Request.Content.IsMimeMultipartContent())
        {
            actionContext.Response = actionContext.Request.CreateErrorResponse(HttpStatusCode.UnsupportedMediaType, Messages.UnsupportedMediaType);
        }
    }
}

关于c# - 如何将文件上传到 Web API 服务器并将参数发送到操作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27885052/

相关文章:

c# - 有没有办法直接从 C# 中的 Amazon S3 制表符分隔文件批量插入 Amazon Aurora RDS?

c# - 接口(interface)不接受继承的成员(OO问题)

javascript - 如何在没有更改功能的情况下在标签中显示 <select> 的当前值

javascript - css - 显示元素然后对其进行动画处理(由 jquery addClass 触发)

asp.net - 在(asp.net + wcf)Web应用程序中存储文件的最佳方式是什么

c# - 避免循环中的对象实例化 c# - 如何避免这种情况?

c# - 在 C# 中解析 XML 文件

php - 如何确保上传的文件是MS Word文档?

javascript - jQuery 与 Magento 中的原型(prototype)冲突

将文件共享到任何可能的应用程序的 iOS native 方式