55 Stimmen

Mehrteilige Formulare vom C#-Client

Ich versuche, ein Formular in einer PHP-Anwendung von einem C#-Client (Outlook Addin) zu füllen. Ich habe Fiddler verwendet, um die ursprüngliche Anfrage aus der PHP-Anwendung zu sehen und das Formular wird als multipart/form übertragen. Leider gibt es in .Net keine native Unterstützung für diese Art von Formularen (WebClient hat nur eine Methode zum Hochladen einer Datei). Kennt jemand eine Bibliothek oder hat einen Code, um dies zu erreichen? Ich möchte verschiedene Werte und zusätzlich (aber nur manchmal) eine Datei übertragen.

Vielen Dank für Ihre Hilfe, Sebastian

72voto

Brian Grinstead Punkte 3462

Vielen Dank für die Antworten, liebe Leute! Ich musste dies vor kurzem zum Laufen bringen und habe Ihre Vorschläge intensiv genutzt. Allerdings gab es ein paar knifflige Stellen, die nicht wie erwartet funktionierten, vor allem was das Einbinden der Datei betrifft (was ein wichtiger Teil der Frage war). Es gibt hier bereits viele Antworten, aber ich denke, dass dies für jemanden in der Zukunft nützlich sein könnte (ich konnte online nicht viele klare Beispiele dafür finden). I schrieb einen Blogbeitrag das erklärt es ein wenig mehr.

Ich habe zunächst versucht, die Dateidaten als UTF8-kodierte Zeichenfolge zu übergeben, aber ich hatte Probleme mit der Kodierung von Dateien (es funktionierte gut für eine einfache Textdatei, aber beim Hochladen eines Word-Dokuments, zum Beispiel, wenn ich versuchte, die Datei zu speichern, die mit Request.Files[0].SaveAs() an das Formular übergeben wurde, funktionierte das Öffnen der Datei in Word nicht richtig. Ich habe herausgefunden, dass es wie erwartet funktioniert, wenn man die Dateidaten direkt mit einem Stream (und nicht mit einem StringBuilder) schreibt. Außerdem habe ich ein paar Änderungen vorgenommen, die mir das Verständnis erleichtern.

Übrigens, die Mehrteilige Formulare Aufforderung zur Stellungnahme und die W3C-Empfehlung für mulitpart/form-data sind einige nützliche Quellen, falls jemand eine Referenz für die Spezifikation benötigt.

Ich habe die Klasse WebHelpers geändert, damit sie etwas kleiner ist und einfachere Schnittstellen hat. Sie heißt jetzt FormUpload . Wenn Sie eine FormUpload.FileParameter können Sie den Byte[]-Inhalt zusammen mit einem Dateinamen und einem Inhaltstyp übergeben, und wenn Sie eine Zeichenkette übergeben, wird diese als Standard-Namen/Wert-Kombination behandelt.

Hier ist die FormUpload-Klasse:

// Implements multipart/form-data POST in C# http://www.ietf.org/rfc/rfc2388.txt
// http://www.briangrinstead.com/blog/multipart-form-post-in-c
public static class FormUpload
{
    private static readonly Encoding encoding = Encoding.UTF8;
    public static HttpWebResponse MultipartFormDataPost(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);

        return PostForm(postUrl, userAgent, contentType, formData);
    }
    private static HttpWebResponse PostForm(string postUrl, string userAgent, string contentType, byte[] formData)
    {
        HttpWebRequest request = WebRequest.Create(postUrl) as HttpWebRequest;

        if (request == null)
        {
            throw new NullReferenceException("request is not a http request");
        }

        // Set up the request properties.
        request.Method = "POST";
        request.ContentType = contentType;
        request.UserAgent = userAgent;
        request.CookieContainer = new CookieContainer();
        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() as HttpWebResponse;
    }

    private static byte[] GetMultipartFormData(Dictionary<string, object> postParameters, string boundary)
    {
        Stream formDataStream = new System.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.Value is FileParameter)
            {
                FileParameter fileToUpload = (FileParameter)param.Value;

                // Add just the first part of this param, since we will write the file data directly to the Stream
                string 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 newline
        string 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 = new byte[formDataStream.Length];
        formDataStream.Read(formData, 0, formData.Length);
        formDataStream.Close();

        return formData;
    }

    public class FileParameter
    {
        public byte[] File { get; set; }
        public string FileName { get; set; }
        public string ContentType { get; set; }
        public FileParameter(byte[] file) : this(file, null) { }
        public FileParameter(byte[] file, string filename) : this(file, filename, null) { }
        public FileParameter(byte[] file, string filename, string contenttype)
        {
            File = file;
            FileName = filename;
            ContentType = contenttype;
        }
    }
}

Hier ist der aufrufende Code, der eine Datei und ein paar normale Postparameter hochlädt:

// Read file data
FileStream fs = new FileStream("c:\\people.doc", FileMode.Open, FileAccess.Read);
byte[] data = new byte[fs.Length];
fs.Read(data, 0, data.Length);
fs.Close();

// Generate post objects
Dictionary<string, object> postParameters = new Dictionary<string, object>();
postParameters.Add("filename", "People.doc");
postParameters.Add("fileformat", "doc");
postParameters.Add("file", new FormUpload.FileParameter(data, "People.doc", "application/msword"));

// Create request and receive response
string postURL = "http://localhost";
string userAgent = "Someone";
HttpWebResponse webResponse = FormUpload.MultipartFormDataPost(postURL, userAgent, postParameters);

// Process response
StreamReader responseReader = new StreamReader(webResponse.GetResponseStream());
string fullResponse = responseReader.ReadToEnd();
webResponse.Close();
Response.Write(fullResponse);

35voto

dnolan Punkte 2259

Dies ist ein ausgeschnittener und eingefügter Beispielcode, den ich geschrieben habe und der hoffentlich die Grundlagen vermittelt. Es unterstützt im Moment nur Dateidaten und Formulardaten.

public class PostData
{

    private List<PostDataParam> m_Params;

    public List<PostDataParam> Params
    {
        get { return m_Params; }
        set { m_Params = value; }
    }

    public PostData()
    {
        m_Params = new List<PostDataParam>();

        // Add sample param
        m_Params.Add(new PostDataParam("email", "MyEmail", PostDataParamType.Field));
    }

    /// <summary>
    /// Returns the parameters array formatted for multi-part/form data
    /// </summary>
    /// <returns></returns>
    public string GetPostData()
    {
        // Get boundary, default is --AaB03x
        string boundary = ConfigurationManager.AppSettings["ContentBoundary"].ToString();

        StringBuilder sb = new StringBuilder();
        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();           
    }
}

public enum PostDataParamType
{
    Field,
    File
}

public class PostDataParam
{

    public PostDataParam(string name, string value, PostDataParamType type)
    {
        Name = name;
        Value = value;
        Type = type;
    }

    public string Name;
    public string FileName;
    public string Value;
    public PostDataParamType Type;
}

Um die Daten zu senden, müssen Sie Folgendes tun:

HttpWebRequest oRequest = null;
oRequest = (HttpWebRequest)HttpWebRequest.Create(oURL.URL);
oRequest.ContentType = "multipart/form-data";                       
oRequest.Method = "POST";
PostData pData = new PostData();

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();

Ich hoffe, das ist klar, ich habe aus einigen Quellen ausgeschnitten und eingefügt, um das ordentlicher zu machen.

35voto

codevision Punkte 4459

Mit .NET 4.5 können Sie derzeit System.Net.Http-Namespace verwenden. Nachfolgend das Beispiel für das Hochladen einer einzelnen Datei mit mehrteiligen Formulardaten.

using System;
using System.IO;
using System.Net.Http;

namespace HttpClientTest
{
    class Program
    {
        static void Main(string[] args)
        {
            var client = new HttpClient();
            var content = new MultipartFormDataContent();
            content.Add(new StreamContent(File.Open("../../Image1.png", FileMode.Open)), "Image", "Image.png");
            content.Add(new StringContent("Place string content here"), "Content-Id in the HTTP"); 
            var result = client.PostAsync("https://hostname/api/Account/UploadAvatar", content);
            Console.WriteLine(result.Result.ToString());
        }
    }
}

13voto

jumoel Punkte 1710

Aufbauend auf dnolans Beispiel ist dies die Version, die ich tatsächlich zum Laufen bringen konnte (es gab einige Fehler mit der Begrenzung, die Kodierung war nicht eingestellt) :-)

Zum Senden der Daten:

HttpWebRequest oRequest = null;
oRequest = (HttpWebRequest)HttpWebRequest.Create("http://you.url.here");
oRequest.ContentType = "multipart/form-data; boundary=" + PostData.boundary;
oRequest.Method = "POST";
PostData pData = new PostData();
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();

Die PostData-Klasse sollte wie folgt aussehen:

public class PostData
{
    // Change this if you need to, not necessary
    public static string boundary = "AaB03x";

    private List<PostDataParam> m_Params;

    public List<PostDataParam> Params
    {
        get { return m_Params; }
        set { m_Params = value; }
    }

    public PostData()
    {
        m_Params = new List<PostDataParam>();
    }

    /// <summary>
    /// Returns the parameters array formatted for multi-part/form data
    /// </summary>
    /// <returns></returns>
    public string GetPostData()
    {
        StringBuilder sb = new StringBuilder();
        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();
    }
}

public enum PostDataParamType
{
    Field,
    File
}

public class PostDataParam
{
    public PostDataParam(string name, string value, PostDataParamType type)
    {
        Name = name;
        Value = value;
        Type = type;
    }

    public PostDataParam(string name, string filename, string value, PostDataParamType type)
    {
        Name = name;
        Value = value;
        FileName = filename;
        Type = type;
    }

    public string Name;
    public string FileName;
    public string Value;
    public PostDataParamType Type;
}

8voto

eeeeaaii Punkte 3212

In der Version von .NET, die ich verwende, müssen Sie dies ebenfalls tun:

System.Net.ServicePointManager.Expect100Continue = false;

Wenn Sie das nicht tun, wird die HttpWebRequest Klasse wird automatisch die Expect:100-continue Anfragekopf, der alles durcheinanderbringt.

Außerdem habe ich auf die harte Tour gelernt, dass man auf die richtige Anzahl von Bindestrichen achten muss. was immer man sagt, ist die "Grenze" in der Content-Type Kopfzeile müssen zwei Bindestriche vorangestellt werden

--THEBOUNDARY

und am Ende

--THEBOUNDARY--

genau wie im Beispielcode. Wenn Ihre Begrenzung aus vielen Bindestrichen, gefolgt von einer Zahl, besteht, wird dieser Fehler bei der Betrachtung der http-Anfrage auf einem Proxy-Server nicht offensichtlich sein

CodeJaeger.com

CodeJaeger ist eine Gemeinschaft für Programmierer, die täglich Hilfe erhalten..
Wir haben viele Inhalte, und Sie können auch Ihre eigenen Fragen stellen oder die Fragen anderer Leute lösen.

Powered by:

X