Multipart forms from C# client
publicclassPostData{privateList<PostDataParam> m_Params;publicList<PostDataParam>Params{
get {return m_Params;}set{ m_Params = value;}}publicPostData(){
m_Params =newList<PostDataParam>();// Add sample param
m_Params.Add(newPostDataParam("email","MyEmail",PostDataParamType.Field));}/// <summary>/// Returns the parameters array formatted for multi-part/form data/// </summary>/// <returns></returns>publicstringGetPostData(){// Get boundary, default is --AaB03xstring boundary =ConfigurationManager.AppSettings["ContentBoundary"].ToString();StringBuilder sb =newStringBuilder();foreach(PostDataParam p in m_Params){
sb.AppendLine(boundary);if(p.Type==PostDataParamType.File){
sb.AppendLine(string.Format("Content-Disposition: file; name=\"{0}\"; filename=\"{1}\"", p.Name, p.FileName));
sb.AppendLine("Content-Type: text/plain");
sb.AppendLine();
sb.AppendLine(p.Value);}else{
sb.AppendLine(string.Format("Content-Disposition: form-data; name=\"{0}\"", p.Name));
sb.AppendLine();
sb.AppendLine(p.Value);}}
sb.AppendLine(boundary);return sb.ToString();}}publicenumPostDataParamType{Field,File}publicclassPostDataParam{publicPostDataParam(string name,string value,PostDataParamType type){Name= name;Value= value;Type= type;}publicstringName;publicstringFileName;publicstringValue;publicPostDataParamTypeType;}
To send the data you then need to:
HttpWebRequest oRequest =null;
oRequest =(HttpWebRequest)HttpWebRequest.Create(oURL.URL);
oRequest.ContentType="multipart/form-data";
oRequest.Method="POST";PostData pData =newPostData();byte[] buffer = encoding.GetBytes(pData.GetPostData());// Set content length of our data
oRequest.ContentLength= buffer.Length;// Dump our buffered postdata to the stream, booyah
oStream = oRequest.GetRequestStream();
oStream.Write(buffer,0, buffer.Length);
oStream.Close();// get the response
oResponse =(HttpWebResponse)oRequest.GetResponse();
Here is the FormUpload class:
// Implements multipart/form-data POST in C# http://www.ietf.org/rfc/rfc2388.txt// http://www.briangrinstead.com/blog/multipart-form-post-in-cpublicstaticclassFormUpload{privatestaticreadonlyEncoding encoding =Encoding.UTF8;publicstaticHttpWebResponseMultipartFormDataPost(string postUrl,string userAgent,Dictionary<string,object> postParameters){string formDataBoundary =String.Format("----------{0:N}",Guid.NewGuid());string contentType ="multipart/form-data; boundary="+ formDataBoundary;byte[] formData =GetMultipartFormData(postParameters, formDataBoundary);returnPostForm(postUrl, userAgent, contentType, formData);}privatestaticHttpWebResponsePostForm(string postUrl,string userAgent,string contentType,byte[] formData){HttpWebRequest request =WebRequest.Create(postUrl)asHttpWebRequest;if(request ==null){thrownewNullReferenceException("request is not a http request");}// Set up the request properties.
request.Method="POST";
request.ContentType= contentType;
request.UserAgent= userAgent;
request.CookieContainer=newCookieContainer();
request.ContentLength= formData.Length;// You could add authentication here as well if needed:// request.PreAuthenticate = true;// request.AuthenticationLevel = System.Net.Security.AuthenticationLevel.MutualAuthRequested;// request.Headers.Add("Authorization", "Basic " + Convert.ToBase64String(System.Text.Encoding.Default.GetBytes("username" + ":" + "password")));// Send the form data to the request.
using (Stream requestStream = request.GetRequestStream()){
requestStream.Write(formData,0, formData.Length);
requestStream.Close();}return request.GetResponse()asHttpWebResponse;}privatestaticbyte[]GetMultipartFormData(Dictionary<string,object> postParameters,string boundary){Stream formDataStream =newSystem.IO.MemoryStream();
bool needsCLRF =false;foreach(var param in postParameters){// Thanks to feedback from commenters, add a CRLF to allow multiple parameters to be added.// Skip it on the first parameter, add it to subsequent parameters.if(needsCLRF)
formDataStream.Write(encoding.GetBytes("\r\n"),0, encoding.GetByteCount("\r\n"));
needsCLRF =true;if(param.ValueisFileParameter){FileParameter fileToUpload =(FileParameter)param.Value;// Add just the first part of this param, since we will write the file data directly to the Streamstring header =string.Format("--{0}\r\nContent-Disposition: form-data; name=\"{1}\"; filename=\"{2}\";\r\nContent-Type: {3}\r\n\r\n",
boundary,
param.Key,
fileToUpload.FileName?? param.Key,
fileToUpload.ContentType??"application/octet-stream");
formDataStream.Write(encoding.GetBytes(header),0, encoding.GetByteCount(header));// Write the file data directly to the Stream, rather than serializing it to a string.
formDataStream.Write(fileToUpload.File,0, fileToUpload.File.Length);}else{string postData =string.Format("--{0}\r\nContent-Disposition: form-data; name=\"{1}\"\r\n\r\n{2}",
boundary,
param.Key,
param.Value);
formDataStream.Write(encoding.GetBytes(postData),0, encoding.GetByteCount(postData));}}// Add the end of the request. Start with a newlinestring footer ="\r\n--"+ boundary +"--\r\n";
formDataStream.Write(encoding.GetBytes(footer),0, encoding.GetByteCount(footer));// Dump the Stream into a byte[]
formDataStream.Position=0;byte[] formData =newbyte[formDataStream.Length];
formDataStream.Read(formData,0, formData.Length);
formDataStream.Close();return formData;}publicclassFileParameter{publicbyte[]File{ get;set;}publicstringFileName{ get;set;}publicstringContentType{ get;set;}publicFileParameter(byte[] file):this(file,null){}publicFileParameter(byte[] file,string filename):this(file, filename,null){}publicFileParameter(byte[] file,string filename,string contenttype){File= file;FileName= filename;ContentType= contenttype;}}}
Here is the calling code, which uploads a file and a few normal post parameters:
// Read file dataFileStream fs =newFileStream("c:\\people.doc",FileMode.Open,FileAccess.Read);byte[] data =newbyte[fs.Length];
fs.Read(data,0, data.Length);
fs.Close();// Generate post objectsDictionary<string,object> postParameters =newDictionary<string,object>();
postParameters.Add("filename","People.doc");
postParameters.Add("fileformat","doc");
postParameters.Add("file",newFormUpload.FileParameter(data,"People.doc","application/msword"));// Create request and receive responsestring postURL ="http://localhost";string userAgent ="Someone";HttpWebResponse webResponse =FormUpload.MultipartFormDataPost(postURL, userAgent, postParameters);// Process responseStreamReader responseReader =newStreamReader(webResponse.GetResponseStream());string fullResponse = responseReader.ReadToEnd();
webResponse.Close();Response.Write(fullResponse);
Building on dnolans example, this is the version I could actually get to work (there were some errors with the boundary, encoding wasn't set) :-)
To send the data:
HttpWebRequest oRequest =null;
oRequest =(HttpWebRequest)HttpWebRequest.Create("http://you.url.here");
oRequest.ContentType="multipart/form-data; boundary="+PostData.boundary;
oRequest.Method="POST";PostData pData =newPostData();Encoding encoding =Encoding.UTF8;Stream oStream =null;/* ... set the parameters, read files, etc. IE:
pData.Params.Add(new PostDataParam("email", "example@example.com", PostDataParamType.Field));
pData.Params.Add(new PostDataParam("fileupload", "filename.txt", "filecontents" PostDataParamType.File));
*/byte[] buffer = encoding.GetBytes(pData.GetPostData());
oRequest.ContentLength= buffer.Length;
oStream = oRequest.GetRequestStream();
oStream.Write(buffer,0, buffer.Length);
oStream.Close();HttpWebResponse oResponse =(HttpWebResponse)oRequest.GetResponse();
The PostData class should look like:
publicclassPostData{// Change this if you need to, not necessarypublicstaticstring boundary ="AaB03x";privateList<PostDataParam> m_Params;publicList<PostDataParam>Params{
get {return m_Params;}set{ m_Params = value;}}publicPostData(){
m_Params =newList<PostDataParam>();}/// <summary>/// Returns the parameters array formatted for multi-part/form data/// </summary>/// <returns></returns>publicstringGetPostData(){StringBuilder sb =newStringBuilder();foreach(PostDataParam p in m_Params){
sb.AppendLine("--"+ boundary);if(p.Type==PostDataParamType.File){
sb.AppendLine(string.Format("Content-Disposition: file; name=\"{0}\"; filename=\"{1}\"", p.Name, p.FileName));
sb.AppendLine("Content-Type: application/octet-stream");
sb.AppendLine();
sb.AppendLine(p.Value);}else{
sb.AppendLine(string.Format("Content-Disposition: form-data; name=\"{0}\"", p.Name));
sb.AppendLine();
sb.AppendLine(p.Value);}}
sb.AppendLine("--"+ boundary +"--");return sb.ToString();}}publicenumPostDataParamType{Field,File}publicclassPostDataParam{publicPostDataParam(string name,string value,PostDataParamType type){Name= name;Value= value;Type= type;}publicPostDataParam(string name,string filename,string value,PostDataParamType type){Name= name;Value= value;FileName= filename;Type= type;}publicstringName;publicstringFileName;publicstringValue;publicPostDataParamTypeType;}
http://stackoverflow.com/questions/219827/multipart-forms-from-c-sharp-client
namespace WindowsFormsApplication1{publicstaticclassFormUpload{privatestaticstringNewDataBoundary(){Random rnd =newRandom();string formDataBoundary ="";while(formDataBoundary.Length<15){
formDataBoundary = formDataBoundary + rnd.Next();}
formDataBoundary = formDataBoundary.Substring(0,15);
formDataBoundary ="-----------------------------"+ formDataBoundary;return formDataBoundary;}publicstaticHttpWebResponseMultipartFormDataPost(string postUrl,IEnumerable<Cookie> cookies,Dictionary<string,string> postParameters){string boundary =NewDataBoundary();HttpWebRequest request =(HttpWebRequest)WebRequest.Create(postUrl);// Set up the request properties
request.Method="POST";
request.ContentType="multipart/form-data; boundary="+ boundary;
request.UserAgent="PhasDocAgent 1.0";
request.CookieContainer=newCookieContainer();foreach(var cookie in cookies){
request.CookieContainer.Add(cookie);}#region WRITING STREAM
using (Stream formDataStream = request.GetRequestStream()){foreach(var param in postParameters){if(param.Value.StartsWith("file://")){string filepath = param.Value.Substring(7);// Add just the first part of this param, since we will write the file data directly to the Streamstring header =string.Format("--{0}\r\nContent-Disposition: form-data; name=\"{1}\"; filename=\"{2}\";\r\nContent-Type: {3}\r\n\r\n",
boundary,
param.Key,Path.GetFileName(filepath)?? param.Key,MimeTypes.GetMime(filepath));
formDataStream.Write(Encoding.UTF8.GetBytes(header),0, header.Length);// Write the file data directly to the Stream, rather than serializing it to a string.byte[] buffer =newbyte[2048];FileStream fs =newFileStream(filepath,FileMode.Open);for(int i =0; i < fs.Length;){int k = fs.Read(buffer,0, buffer.Length);if(k >0){
formDataStream.Write(buffer,0, k);}
i = i + k;}
fs.Close();}else{string postData =string.Format("--{0}\r\nContent-Disposition: form-data; name=\"{1}\"\r\n\r\n{2}\r\n",
boundary,
param.Key,
param.Value);
formDataStream.Write(Encoding.UTF8.GetBytes(postData),0, postData.Length);}}// Add the end of the requestbyte[] footer =Encoding.UTF8.GetBytes("\r\n--"+ boundary +"--\r\n");
formDataStream.Write(footer,0, footer.Length);
request.ContentLength= formDataStream.Length;
formDataStream.Close();}#endregionreturn request.GetResponse()asHttpWebResponse;}}}
参考
C# Socket 实现的淘宝秒杀器(抢拍器)
http://www.cnblogs.com/chengulv/archive/2013/01/25/2877383.html