ASP.NET 服务端接收Multipart/form-data文件
在网络编程过程中需要向服务器上传文件。
Multipart/form-data是上传文件的一种方式。
/// <summary>
/// 上传工程文件
/// </summary>
/// <returns></returns>
public async Task<HttpResponseMessage> UploadProjectFile()
{
ProjectFile postData = new ProjectFile();
IEnumerable<string> stringValues;
//获取请求头文件相关参数
Request.Headers.TryGetValues("token", out stringValues);
postData.token = stringValues.FirstOrDefault(); Request.Headers.TryGetValues("uid", out stringValues);
postData.uid = int.Parse(stringValues.FirstOrDefault()); Request.Headers.TryGetValues("project_id", out stringValues);
postData.project_id = int.Parse(stringValues.FirstOrDefault()); Request.Headers.TryGetValues("md5", out stringValues);
postData.md5 = stringValues.FirstOrDefault(); Request.Headers.TryGetValues("file_name", out stringValues);
postData.file_name = stringValues.FirstOrDefault();
//验证Token
var result = CheckToken(postData.token);
if (!result)
{
return CreateResponseError(, "请求拒绝");
}
//获取文件名,这里的文件名必须带扩展名
string fileName = postData.file_name; int projectId = postData.project_id;
//获取应用程序的当前工作目录
string rootPath = HostingEnvironment.MapPath("~/");
//通过Path类的Combine方法可以合并路径。获取当前工程文件夹地址
string directoryPath = Path.Combine(rootPath, "BgerProject", projectId.ToString());
//创建目录时如果目录已存在,则不会重新创建目录,且不会报错。创建目录时会自动创建路径中各级不存在的目录。
Directory.CreateDirectory(directoryPath);
//获取当前工程文件地址
string filePath = Path.Combine(directoryPath, fileName);
if (Request.Content.IsMimeMultipartContent())
{
Dictionary<string, string> dic = new Dictionary<string, string>();
var provider = new MultipartFormDataMemoryStreamProvider(); await Request.Content.ReadAsMultipartAsync(provider);
foreach (var item in provider.FileContents)
{
//Trace.WriteLine(item.Headers.ContentDisposition.FileName);//获取上传文件实际的文件名
//Trace.WriteLine("Server file path: " + item.LocalFileName);//获取上传文件在服务上默认的文件名
var stream = await item.ReadAsStreamAsync();
if (Path.GetExtension(item.Headers.ContentDisposition.FileName.Replace("\"", "")) == ".mp4")
{
//通过正则表达式判断是视频还是音频
Regex regex = new Regex("voice", RegexOptions.Compiled);
Match m = regex.Match(item.Headers.ContentDisposition.Name.ToLower());
if (m.Success)
{
//音频直接保存处理 //判断是否已存在该音频
if (!File.Exists(filePath))
{
using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
}
}
else
{
//视频进行压缩处理
//保存原片(如果路径不存在,创建路径) using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
//压缩并保存上传的视频
//压缩后的存放路径(如果路径不存在,创建路径)
////判断是否已存在该视频
//if (!File.Exists(filePath))
//{
// //压缩保存视频
// if (!FFMPEGHelper.CreateNewVideo(originalFilePath, filePath))
// {
// continue;
// };
//}
}
}
else
{
//保存原片(如果路径不存在,创建路径) using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
//压缩并保存上传的图片
//压缩后的存放路径(如果路径不存在,创建路径) ////判断是否已存在该图片
//if (!File.Exists(filePath))
//{
// var pxLimit = 481;
// //判断是否需要压缩
// System.Drawing.Image iSource = System.Drawing.Image.FromFile(originalFilePath);
// if (iSource.Width <= pxLimit || iSource.Height <= pxLimit)
// {
// //无需压缩
// System.IO.File.Copy(originalFilePath, filePath);
// }
// else
// {
// var min = Math.Min((int)iSource.Width, (int)iSource.Height);
// if (min > pxLimit)
// {
// //缩放倍率
// var rate = (double)pxLimit / min;
// var width = (int)Math.Ceiling(rate * (double)iSource.Width);
// var height = (int)Math.Ceiling(rate * (double)iSource.Height); // //压缩保存图片
// if (!ImageHelper.GetPicThumbnail(originalFilePath, filePath, height, width, 100))
// {
// continue;
// };
// }
// }
//}
}
}
return CreateResponse("请求成功"); }
return CreateResponseError(,"缺少MIME内容"); }
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.IO;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Threading;
using System.Threading.Tasks; namespace BgerServer.Models
{
/// <summary>
/// 与MultipartFormDataStreamProvider对应,但不将文件直接存入指定位置,而是需要自己指定数据流如何保存
/// </summary>
public class MultipartFormDataMemoryStreamProvider : MultipartStreamProvider
{
private NameValueCollection _formData = new NameValueCollection();
private Collection<bool> _isFormData = new Collection<bool>();
/// <summary>
/// 获取文件对应的HttpContent集合,文件如何读取由实际使用方确定,可以ReadAsByteArrayAsync,也可以ReadAsStreamAsync
/// </summary>
public Collection<HttpContent> FileContents
{
get
{
if (this._isFormData.Count != this.Contents.Count)//两者总数不一致,认为未执行过必须的Request.Content.ReadAsMultipartAsync(provider)方法
{
throw new InvalidOperationException("System.Net.Http.HttpContentMultipartExtensions.ReadAsMultipartAsync must be called first!");
}
return new Collection<HttpContent>(this.Contents.Where((ct, idx) => !this._isFormData[idx]).ToList());
}
}
/// <summary>Gets a <see cref="T:System.Collections.Specialized.NameValueCollection" /> of form data passed as part of the multipart form data.</summary>
/// <returns>The <see cref="T:System.Collections.Specialized.NameValueCollection" /> of form data.</returns>
public NameValueCollection FormData
{
get
{
return this._formData;
}
}
public override async Task ExecutePostProcessingAsync()
{
for (var i = ; i < this.Contents.Count; i++)
{
if (!this._isFormData[i])//非文件
{
continue;
}
var formContent = this.Contents[i];
ContentDispositionHeaderValue contentDisposition = formContent.Headers.ContentDisposition;
string formFieldName = UnquoteToken(contentDisposition.Name) ?? string.Empty;
string formFieldValue = await formContent.ReadAsStringAsync();
this.FormData.Add(formFieldName, formFieldValue);
}
}
public override Stream GetStream(HttpContent parent, HttpContentHeaders headers)
{
if (parent == null)
{
throw new ArgumentNullException("parent");
}
if (headers == null)
{
throw new ArgumentNullException("headers");
}
ContentDispositionHeaderValue contentDisposition = headers.ContentDisposition;
if (contentDisposition == null)
{
throw new InvalidOperationException("Content-Disposition is null");
}
this._isFormData.Add(string.IsNullOrEmpty(contentDisposition.FileName));
return new MemoryStream();
}
/// <summary>
/// 复制自 System.Net.Http.FormattingUtilities 下同名方法,因为该类为internal,不能在其它命名空间下被调用
/// </summary>
/// <param name="token"></param>
/// <returns></returns>
private static string UnquoteToken(string token)
{
if (string.IsNullOrWhiteSpace(token))
{
return token;
}
if (token.StartsWith("\"", StringComparison.Ordinal) && token.EndsWith("\"", StringComparison.Ordinal) && token.Length > )
{
return token.Substring(, token.Length - );
}
return token;
}
}
}
ASP.NET 服务端接收Multipart/form-data文件的更多相关文章
- Asp.Net_ 服务端向客户端写JavaScript脚本
在Asp.net 服务端处理脚本,一般都用 ClientScriptManager ,即web窗体服务端的this.ClientScript.该对象比较常用的方法: 1.RegisterArrayDe ...
- C# TCP socket发送大数据包时,接收端和发送端数据不一致 服务端接收Receive不完全
简单的c# TCP通讯(TcpListener) C# 的TCP Socket (同步方式) C# 的TCP Socket (异步方式) C# 的tcp Socket设置自定义超时时间 C# TCP ...
- 自己实现FormsAuthentication.SetAuthCookie方法,怎样在ASP.NET服务端代码中删除客户端Cookie
如何手动设置AuthCookie ASP.NET中实现可以自己实现FormsAuthentication.SetAuthCookie方法,控制更为灵活 /// <summary> /// ...
- android如何与asp.net服务端共享session
近期需要实现一个功能,就是需要通过发送短信进行注册,现在想把短信验证码放到服务器的session值中,当客户端收到短信并提交短信码时由asp.net服务端进行判断,那么如何共享这个session那么需 ...
- MVC文件上传04-使用客户端jQuery-File-Upload插件和服务端Backload组件实现多文件异步上传
本篇使用客户端jQuery-File-Upload插件和服务端Badkload组件实现多文件异步上传.MVC文件上传相关兄弟篇: MVC文件上传01-使用jquery异步上传并客户端验证类型和大小 ...
- 微信小程序开发之多图片上传+服务端接收
前言: 业务需求,这次需要做一个小程序同时选中三张图片一起上传到服务端,后端使用的.NET WEBAPI接收数据保存. 使用技术: 在这章中将会使用到微信小程序wx.uploadFile(Object ...
- java httpclient发送json 请求 ,go服务端接收
/***java客户端发送http请求*/package com.xx.httptest; /** * Created by yq on 16/6/27. */ import java.io.IOEx ...
- [转]微信小程序开发(二)图片上传+服务端接收
本文转自:http://blog.csdn.net/sk719887916/article/details/54312573 文/YXJ 地址:http://blog.csdn.net/sk71988 ...
- tcp syn-synack-ack 服务端接收ack
TCP 服务端 接收到ack tcp_v4_rcv() -> tcp_v4_do_rcv() -> tcp_v4_hnd_req() + tcp_child_process()tcp_v4 ...
随机推荐
- P2518 [HAOI2010]计数 类似数位dp
题意 你有一组非零数字(不一定唯一),你可以在其中插入任意个0,这样就可以产生无限个数.比如说给定{1,2},那么可以生成数字12,21,102,120,201,210,1002,1020,等等. 现 ...
- gym/101955/problem/E - The Kouga Ninja Scrolls 线段数 维护 切比雪夫距离 2018沈阳icpc
传送门 思路: 这道题要把给定的每个坐标利用切比雪夫坐标表示,这样两个点的距离就是max(dx,dy),而不是一开始的dx + dy,有利于线段树的维护,又由于询问的是区间的最大差值,限制是两个点是属 ...
- 2019 Multi-University Training Contest 1
2019 Multi-University Training Contest 1 A. Blank upsolved by F0_0H 题意 给序列染色,使得 \([l_i,r_i]\) 区间内恰出现 ...
- 区间dp专题
HDU4283You Are the One区间dp, 记忆话搜索运行时间: #include <iostream> #include <cstdio> #include ...
- CodeForces 785 D Anton and School - 2 范德蒙恒等式
Anton and School - 2 题解: 枚举每个左括号作为必选的. 那么方案数就应该是下面的 1 , 然后不断化简, 通过范德蒙恒等式 , 可以将其化为一个组合数. 代码: #include ...
- solr 的基本用法
上图为 solr 的搜索页面,常用字段的基本用法如下: 1. q: 查询字符串,过滤条件,不能为空,必须输入,如果查询全部就写 * : * name:“马” AND age:[0 TO 18] ...
- 【UEFI】---BIOS中UserPassword的重复校验总结
UEFI作为目前较为流行的一套X86架构初始化的标准框架,已受到业界内的广泛认可.而其中很多编程所采用的思想确实值得学习.今天总结下UEFI的框架下修改代码的一点小经验,仅供菜鸟参考. 先列干货,具体 ...
- day01小结
Java特点 1,面向对象的(write once,run anywhere) 2,跨平台的 ,,,,,, Java体系结构 JavaSE,JavaEE,JavaME,Java Card 对JDK,J ...
- FreeSql (三十)读写分离
FreeSql 支持数据库读写分离,本功能是客户端的读写分离行为,数据库服务器该怎么配置仍然那样配置,不受本功能影响,为了方便描术后面讲到的[读写分离]都是指客户端的功能支持. 各种数据库的读写方案不 ...
- Android四大组件初识之Activity
一.Activity的生命周期 Activity生命周期是一系列方法调用.熟悉各个方法调用时间,我们在创建activity就能根据具体实现选择合适的方法覆盖. 1. 覆盖Activity的生命周期方 ...