ส่งไฟล์ผ่าน HTTP POST ด้วย C #


97

ฉันค้นหาและอ่านเรื่องนั้นมาตลอดและไม่สามารถปรับอะไรที่เป็นประโยชน์ได้เลย

ฉันกำลังเขียนแอป C # win ขนาดเล็กที่อนุญาตให้ผู้ใช้ส่งไฟล์ไปยังเว็บเซิร์ฟเวอร์ไม่ใช่โดย FTP แต่เป็น HTTP โดยใช้ POST คิดว่ามันเหมือนเว็บฟอร์ม แต่ทำงานบนแอพพลิเคชั่น windows

ฉันมีวัตถุ HttpWebRequest ของฉันที่สร้างขึ้นโดยใช้สิ่งนี้

HttpWebRequest req = WebRequest.Create(uri) as HttpWebRequest 

และยังตั้งMethod, ContentTypeและContentLengthคุณสมบัติ แต่นั่นคือสิ่งที่ฉันไปได้ไกล

นี่คือรหัสของฉัน:

HttpWebRequest req = WebRequest.Create(uri) as HttpWebRequest;
req.KeepAlive = false;
req.Method = "POST";
req.Credentials = new NetworkCredential(user.UserName, user.UserPassword);
req.PreAuthenticate = true;
req.ContentType = file.ContentType;
req.ContentLength = file.Length;
HttpWebResponse response = null;

try
{
    response = req.GetResponse() as HttpWebResponse;
}
catch (Exception e) 
{
}

ดังนั้นคำถามของฉันคือโดยพื้นฐานแล้วฉันจะส่ง fie (ไฟล์ข้อความรูปภาพเสียง ฯลฯ ) ด้วย C # ผ่าน HTTP POST ได้อย่างไร

ขอบคุณ!


1
โปรดตรวจสอบstackoverflow.com/questions/15738847/…
Sudha

คำตอบ:


113

การใช้. NET 4.5 (หรือ. NET 4.0 โดยการเพิ่มแพ็คเกจMicrosoft.Net.Httpจาก NuGet) มีวิธีที่ง่ายกว่าในการจำลองคำขอแบบฟอร์ม นี่คือตัวอย่าง:

private async Task<System.IO.Stream> Upload(string actionUrl, string paramString, Stream paramFileStream, byte [] paramFileBytes)
{
    HttpContent stringContent = new StringContent(paramString);
    HttpContent fileStreamContent = new StreamContent(paramFileStream);
    HttpContent bytesContent = new ByteArrayContent(paramFileBytes);
    using (var client = new HttpClient())
    using (var formData = new MultipartFormDataContent())
    {
        formData.Add(stringContent, "param1", "param1");
        formData.Add(fileStreamContent, "file1", "file1");
        formData.Add(bytesContent, "file2", "file2");
        var response = await client.PostAsync(actionUrl, formData);
        if (!response.IsSuccessStatusCode)
        {
            return null;
        }
        return await response.Content.ReadAsStreamAsync();
    }
}

8
ถ้าเป็นไปได้สามารถแสดงตัวอย่างง่ายๆของการเรียกวิธีนี้?
Jacobr365

10
พารามิเตอร์ paramString คืออะไร?
eran otzap

2
ขอบคุณตัวอย่างที่ครอบคลุมมาก! @eranotzap paramString คือค่าที่แท้จริงของพารามิเตอร์ที่จะส่ง อาร์กิวเมนต์ที่สามform.Addเป็นทางเลือกและมีประโยชน์สำหรับไฟล์เท่านั้น
StockBreak

1
@ เลียมยอมเต็มที่ รหัส async ถูกลบออกจากคำตอบปี 2013 ของฉันเพื่อให้ทุกอย่างง่ายขึ้น การเปลี่ยนกลับเป็นวิธี async อยู่ในรายการสิ่งที่ต้องทำของฉันเนื่องจากนักพัฒนา C # ส่วนใหญ่ควรพอใจกับมันในตอนนี้
Joshcodes

1
@Ammar ไม่ใช่ว่าฉันรู้ฉันคิดว่าคุณจะต้องอ่านไฟล์ในสตรีมหรือไบต์ [] และใช้ StreamContent หรือ ByteArrayContent ตามลำดับ
Joshcodes

52

ในการส่งไฟล์ดิบเท่านั้น :

using(WebClient client = new WebClient()) {
    client.UploadFile(address, filePath);
}

หากคุณต้องการเลียนแบบรูปแบบเบราว์เซอร์ด้วยแอ<input type="file"/>ปนั้นยากกว่า ดูคำตอบนี้สำหรับคำตอบแบบหลายส่วน / แบบฟอร์มข้อมูล


(แน่นอนคุณสามารถเพิ่มส่วนหัว / ข้อมูลรับรอง / ฯลฯ ได้ตามปกติ)
Marc Gravell

1
ขอบคุณฉันใช้มันกับสิ่งที่เรียบง่ายและไม่ได้ผล อย่างที่คุณพูดฉันจำเป็นต้องเลียนแบบไฟล์อินพุตของเบราว์เซอร์บางครั้งก็คล้ายกับ <intput type = "file" name "userFile" />
gabitoju

1
ฉันใช้รหัสข้างต้นแล้วและได้รับข้อผิดพลาดเช่นรหัสผู้ใช้ไม่สามารถจัดการข้อยกเว้นของอาร์กิวเมนต์ได้: {"ไม่รองรับรูปแบบ URI"} ฉันจะทำเช่นนี้ได้อย่างไร? โมฆะที่มีการป้องกัน Page_Load (ผู้ส่งวัตถุ, EventArgs e) {string address = "http: www.testproject.com/SavedFiles"; สตริง filepath = @ "D: \ test \ FileOperations \ testfile.txt"; โดยใช้ (ไคลเอนต์ WebClient = WebClient ใหม่ ()) {client.UploadFile (ที่อยู่, filepath); }}
Sudha

1
@ คุณสุธาลองใช้ที่อยู่เว็บจริงหรือยัง? http://www.testproject.com/SavedFiles- สังเกต//
Marc Gravell

8

สำหรับฉันclient.UploadFileยังคงรวมเนื้อหาไว้ในคำขอหลายส่วนดังนั้นฉันจึงต้องทำเช่นนี้:

using (WebClient client = new WebClient())
{
    client.Headers.Add("Content-Type", "application/octet-stream");
    using (Stream fileStream = File.OpenRead(filePath))
    using (Stream requestStream = client.OpenWrite(new Uri(fileUploadUrl), "POST"))
    {
        fileStream.CopyTo(requestStream);
    }
}

5

ฉันมีปัญหาเดียวกันและรหัสต่อไปนี้ตอบโจทย์ปัญหานี้ได้อย่างสมบูรณ์:

//Identificate separator
string boundary = "---------------------------" + DateTime.Now.Ticks.ToString("x");
//Encoding
byte[] boundarybytes = System.Text.Encoding.ASCII.GetBytes("\r\n--" + boundary + "\r\n");

//Creation and specification of the request
HttpWebRequest wr = (HttpWebRequest)WebRequest.Create(url); //sVal is id for the webService
wr.ContentType = "multipart/form-data; boundary=" + boundary;
wr.Method = "POST";
wr.KeepAlive = true;
wr.Credentials = System.Net.CredentialCache.DefaultCredentials;

string sAuthorization = "login:password";//AUTHENTIFICATION BEGIN
byte[] toEncodeAsBytes = System.Text.ASCIIEncoding.ASCII.GetBytes(sAuthorization);
string returnValue = System.Convert.ToBase64String(toEncodeAsBytes);
wr.Headers.Add("Authorization: Basic " + returnValue); //AUTHENTIFICATION END
Stream rs = wr.GetRequestStream();


string formdataTemplate = "Content-Disposition: form-data; name=\"{0}\"\r\n\r\n{1}"; //For the POST's format

//Writting of the file
rs.Write(boundarybytes, 0, boundarybytes.Length);
byte[] formitembytes = System.Text.Encoding.UTF8.GetBytes(Server.MapPath("questions.pdf"));
rs.Write(formitembytes, 0, formitembytes.Length);

rs.Write(boundarybytes, 0, boundarybytes.Length);

string headerTemplate = "Content-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"\r\nContent-Type: {2}\r\n\r\n";
string header = string.Format(headerTemplate, "file", "questions.pdf", contentType);
byte[] headerbytes = System.Text.Encoding.UTF8.GetBytes(header);
rs.Write(headerbytes, 0, headerbytes.Length);

FileStream fileStream = new FileStream(Server.MapPath("questions.pdf"), FileMode.Open, FileAccess.Read);
byte[] buffer = new byte[4096];
int bytesRead = 0;
while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0)
{
    rs.Write(buffer, 0, bytesRead);
}
fileStream.Close();

byte[] trailer = System.Text.Encoding.ASCII.GetBytes("\r\n--" + boundary + "--\r\n");
rs.Write(trailer, 0, trailer.Length);
rs.Close();
rs = null;

WebResponse wresp = null;
try
{
    //Get the response
    wresp = wr.GetResponse();
    Stream stream2 = wresp.GetResponseStream();
    StreamReader reader2 = new StreamReader(stream2);
    string responseData = reader2.ReadToEnd();
}
catch (Exception ex)
{
    string s = ex.Message;
}
finally
{
    if (wresp != null)
    {
        wresp.Close();
        wresp = null;
    }
    wr = null;
}

คุณจะรับข้อมูลและบันทึกไฟล์ลงในดิสก์ได้อย่างไร?
KumarHarsh


1

ในการโพสต์ไฟล์จากอาร์เรย์ไบต์:

private static string UploadFilesToRemoteUrl(string url, IList<byte[]> files, NameValueCollection nvc) {

        string boundary = "----------------------------" + DateTime.Now.Ticks.ToString("x");

        var request = (HttpWebRequest) WebRequest.Create(url);
        request.ContentType = "multipart/form-data; boundary=" + boundary;
        request.Method = "POST";
        request.KeepAlive = true;
        var postQueue = new ByteArrayCustomQueue();

        var formdataTemplate = "\r\n--" + boundary + "\r\nContent-Disposition: form-data; name=\"{0}\";\r\n\r\n{1}";

        foreach (string key in nvc.Keys) {
            var formitem = string.Format(formdataTemplate, key, nvc[key]);
            var formitembytes = Encoding.UTF8.GetBytes(formitem);
            postQueue.Write(formitembytes);
        }

        var headerTemplate = "\r\n--" + boundary + "\r\n" +
            "Content-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"\r\n" + 
            "Content-Type: application/zip\r\n\r\n";

        var i = 0;
        foreach (var file in files) {
            var header = string.Format(headerTemplate, "file" + i, "file" + i + ".zip");
            var headerbytes = Encoding.UTF8.GetBytes(header);
            postQueue.Write(headerbytes);
            postQueue.Write(file);
            i++;
        }

        postQueue.Write(Encoding.UTF8.GetBytes("\r\n--" + boundary + "--"));

        request.ContentLength = postQueue.Length;

        using (var requestStream = request.GetRequestStream()) {
            postQueue.CopyToStream(requestStream);
            requestStream.Close();
        }

        var webResponse2 = request.GetResponse();

        using (var stream2 = webResponse2.GetResponseStream())
        using (var reader2 = new StreamReader(stream2)) {

            var res =  reader2.ReadToEnd();
            webResponse2.Close();
            return res;
        }
    }

public class ByteArrayCustomQueue {

    private LinkedList<byte[]> arrays = new LinkedList<byte[]>();

    /// <summary>
    /// Writes the specified data.
    /// </summary>
    /// <param name="data">The data.</param>
    public void Write(byte[] data) {
        arrays.AddLast(data);
    }

    /// <summary>
    /// Gets the length.
    /// </summary>
    /// <value>
    /// The length.
    /// </value>
    public int Length { get { return arrays.Sum(x => x.Length); } }

    /// <summary>
    /// Copies to stream.
    /// </summary>
    /// <param name="requestStream">The request stream.</param>
    /// <exception cref="System.NotImplementedException"></exception>
    public void CopyToStream(Stream requestStream) {
        foreach (var array in arrays) {
            requestStream.Write(array, 0, array.Length);
        }
    }
}

1
     public string SendFile(string filePath)
            {
                WebResponse response = null;
                try
                {
                    string sWebAddress = "Https://www.address.com";

                    string boundary = "---------------------------" + DateTime.Now.Ticks.ToString("x");
                    byte[] boundarybytes = System.Text.Encoding.ASCII.GetBytes("\r\n--" + boundary + "\r\n");
                    HttpWebRequest wr = (HttpWebRequest)WebRequest.Create(sWebAddress);
                    wr.ContentType = "multipart/form-data; boundary=" + boundary;
                    wr.Method = "POST";
                    wr.KeepAlive = true;
                    wr.Credentials = System.Net.CredentialCache.DefaultCredentials;
                    Stream stream = wr.GetRequestStream();
                    string formdataTemplate = "Content-Disposition: form-data; name=\"{0}\"\r\n\r\n{1}";

                    stream.Write(boundarybytes, 0, boundarybytes.Length);
                    byte[] formitembytes = System.Text.Encoding.UTF8.GetBytes(filePath);
                    stream.Write(formitembytes, 0, formitembytes.Length);
                    stream.Write(boundarybytes, 0, boundarybytes.Length);
                    string headerTemplate = "Content-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"\r\nContent-Type: {2}\r\n\r\n";
                    string header = string.Format(headerTemplate, "file", Path.GetFileName(filePath), Path.GetExtension(filePath));
                    byte[] headerbytes = System.Text.Encoding.UTF8.GetBytes(header);
                    stream.Write(headerbytes, 0, headerbytes.Length);

                    FileStream fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read);
                    byte[] buffer = new byte[4096];
                    int bytesRead = 0;
                    while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0)
                        stream.Write(buffer, 0, bytesRead);
                    fileStream.Close();

                    byte[] trailer = System.Text.Encoding.ASCII.GetBytes("\r\n--" + boundary + "--\r\n");
                    stream.Write(trailer, 0, trailer.Length);
                    stream.Close();

                    response = wr.GetResponse();
                    Stream responseStream = response.GetResponseStream();
                    StreamReader streamReader = new StreamReader(responseStream);
                    string responseData = streamReader.ReadToEnd();
                    return responseData;
                }
                catch (Exception ex)
                {
                    return ex.Message;
                }
                finally
                {
                    if (response != null)
                        response.Close();
                }
            }

0

การใช้. NET 4.5 พยายามอัปโหลดไฟล์แบบ POST พยายามใช้วิธีการส่วนใหญ่ข้างต้น แต่ก็ไม่เกิดประโยชน์ พบวิธีแก้ปัญหาที่นี่ https://www.c-sharpcorner.com/article/upload-any-file-using-http-post-multipart-form-data

แต่ฉันไม่กระตือรือร้นเพราะฉันไม่เข้าใจว่าทำไมเรายังต้องจัดการกับการเขียนโปรแกรมระดับต่ำเช่นนี้ในการใช้งานทั่วไปเหล่านี้ (ควรจัดการอย่างดีตามกรอบ)

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.