c++ - 为什么在发送几 block 数据后文件上传停止(使用 multipart/form-data)?

标签 c++ windows multipartform-data libcurl

我正在使用 libcurl 将固件文件上传到硬件设备。我正在使用 multipart/form-data,看起来文件上传开始正常,但它没有加载整个文件。
我上传的文件是 144,855,725 字节,但似乎只发送了两个 64k block 。

在发布的代码中,我使用了一个读取回调函数。我也尝试过将文件名传递给 curl_mime_filedata 函数,结果是一样的。
一个有趣的注意事项是,当我运行该程序时,大约有一半的时间我会从 curl_easy_perform 获得成功响应。另一半时间我会收到错误 56“从对等方接收数据时失败”。 另一个有趣的注意事项是文件大小 (144,855,725) 与 curl 认为的下载大小 (144,856,042) 略有不同。我认为这是因为它考虑了正文中的所有字节(而不仅仅是文件)。对吗?

这是我运行程序时的一些(简化的)输出。
文件大小:144855725
总时间:0.000092
向上:第 0 个,共 0 个向下:第 0 个,共 0 个
向上:144856042 中的 0 向下:0 中的 0
ReadCallback: Size=1, Nmemb=65267
我们从文件中读取了 65267 个字节
向上:144856042 中的 65536 向下:0 中的 0
ReadCallback: Size=1, Nmemb=65536
我们从文件中读取了 65536 字节
向上:144856042 中的 131072 向下:0 中的 0
curl 结果 ERROR = <56: 从对等方接收数据时失败>
固件文件上传失败

size_t ReadCallback(char *BufferOut, size_t Size, size_t Nmemb, void *StreamIn)
{
   curl_off_t nread;
   size_t retcode = fread(BufferOut, Size, Nmemb, (FILE *)StreamIn);
   nread = (curl_off_t)retcode;
   cout << "ReadCallback: Size=" << Size << ", Nmemb=" << Nmemb << endl;
   cout << "We read " << nread << " bytes from the file" << endl;
   return retcode;
}
int main(void)
{
   CURL *pCurl;
   CURLcode res;
   std::stringstream ss;
   struct curl_slist *headerList = NULL;
   string accessToken;
   struct TransferProgress transProgress;
   string filePath;
   FILE * pFile;
   long lSize;

   curl_global_init(CURL_GLOBAL_ALL);
   pCurl = curl_easy_init();

   if (pCurl)
   {
      EC520UutComms comms;
      curl_mime *multipart;
      curl_mimepart *part;

      accessToken = comms.GetAccessToken(pCurl);
      SetOptionsToDefault(pCurl);

      // Specify the target URL
      std::string str(comms.BaseURL() + kAPI_Upgrade);
      cout << "URL <" + str + ">" << endl;
      curl_easy_setopt(pCurl, CURLOPT_URL, str.c_str());

      multipart = curl_mime_init(pCurl);
      // Add the Content-Disposition
      part = curl_mime_addpart(multipart);
      curl_mime_name(part, "Content-Disposition");
      curl_mime_data(part, "form-data; name=\"upgrade_file\"; filename=\"\"", CURL_ZERO_TERMINATED);
      // Add the file
      part = curl_mime_addpart(multipart);
      curl_mime_type(part, "application/octet-stream");

      filePath = "C:\\Temp\\TestFile.tst";
//       curl_mime_filedata(part, filePath.c_str());
       fopen_s(&pFile, filePath.c_str(), "rb");
      // obtain file size:
      fseek(pFile, 0, SEEK_END);
      lSize = ftell(pFile);
      rewind(pFile);
      cout << "File size: " << lSize << endl;

      curl_mime_data_cb(part, lSize, ReadCallback, NULL, NULL, pFile);

      curl_easy_setopt(pCurl, CURLOPT_MIMEPOST, multipart);
      // This is a long upload, disable the timeout
      curl_easy_setopt(pCurl, CURLOPT_TIMEOUT, 0L);

      // Create headers
      ss.str("");
      ss << "Authorization: Bearer " << accessToken;
      headerList = curl_slist_append(headerList, ss.str().c_str());
      // Accept
      headerList = curl_slist_append(headerList,
         "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8");

      curl_easy_setopt(pCurl, CURLOPT_HTTPHEADER, headerList);

      curl_easy_setopt(pCurl, CURLOPT_XFERINFOFUNCTION, TransferInfo);
      transProgress.curl = pCurl;
      curl_easy_setopt(pCurl, CURLOPT_XFERINFODATA, &transProgress);
      curl_easy_setopt(pCurl, CURLOPT_NOPROGRESS, 0L);

      // Now send the message
      res = curl_easy_perform(pCurl);

      curl_slist_free_all(headerList);
      curl_mime_free(multipart);

      if (res == CURLE_OK)
      {
         cout << "Firmware file successfully uploaded" << endl;
      }
      else
      {
         cout << "curl result ERROR = <" + to_string(res) + ": " + curl_easy_strerror(res) + ">" << endl;
         cout << "Failed to upload firmware file" << endl;
      }
   }
   curl_easy_cleanup(pCurl);
}

我希望上传整个文件,而不是只上传几个文件 block 。

最佳答案

您没有正确填充 curl_mime 结构。您显示的代码与您在 previous question 中描述的 MIME 格式不匹配:

--1a2fc07a-d882-4470-a1da-79716d34cd9b
Content-Disposition: form-data; name="upgrade_file"; filename=""
Content-Type: application/octet-stream

// File data goes here //
--1a2fc07a-d882-4470-a1da-79716d34cd9b
Content-Disposition: form-data; name="submit"

Install OS
--1a2fc07a-d882-4470-a1da-79716d34cd9b--

Try this instead:

size_t ReadCallback(char *buffer, size_t size, size_t nitems, void *arg)
{
    cout << "ReadCallback: size=" << size << ", nitems=" << nitems << endl;
    size_t retcode = fread(buffer, size, nitems, (FILE *)arg);
    cout << "We read " << retcode << " bytes from the file" << endl;
    return retcode;
}

int SeekCallback(void *arg, curl_off_t offset, int origin)
{
    if (fseek((FILE *)arg, offset, origin) == 0)
        return CURL_SEEKFUNC_OK;
    else
        return CURL_SEEKFUNC_FAIL;
}

int main()
{
    CURL *pCurl;
    CURLcode res;
    struct curl_slist *headerList = NULL;
    string accessToken;
    struct TransferProgress transProgress;
    string filePath;
    FILE * pFile;
    long lSize;

    curl_global_init(CURL_GLOBAL_ALL);
    pCurl = curl_easy_init();
    if (pCurl)
    {
        EC520UutComms comms;
        curl_mime *multipart;
        curl_mimepart *part;

        accessToken = comms.GetAccessToken(pCurl);
        SetOptionsToDefault(pCurl);

        // Specify the target URL
        std::string str(comms.BaseURL() + kAPI_Upgrade);
        cout << "URL <" + str + ">" << endl;
        curl_easy_setopt(pCurl, CURLOPT_URL, str.c_str());

        multipart = curl_mime_init(pCurl);

        part = curl_mime_addpart(multipart);
        curl_mime_name(part, "upgrade_file");
        curl_mime_filename(part, "");
        curl_mime_type(part, "application/octet-stream");
        filePath = "C:\\Temp\\TestFile.tst";
        // curl_mime_filedata(part, filePath.c_str());
        fopen_s(&pFile, filePath.c_str(), "rb");
        // obtain file size:
        fseek(pFile, 0, SEEK_END);
        lSize = ftell(pFile);
        rewind(pFile);
        cout << "File size: " << lSize << endl;
        curl_mime_data_cb(part, lSize, ReadCallback, SeekCallback, NULL, pFile);

        part = curl_mime_addpart(multipart);
        curl_mime_name(part, "submit");
        curl_mime_data(part, "Install OS", CURL_ZERO_TERMINATED);

        curl_easy_setopt(pCurl, CURLOPT_MIMEPOST, multipart);

        // This is a long upload, disable the timeout
        curl_easy_setopt(pCurl, CURLOPT_TIMEOUT, 0L);

        // Create headers
        headerList = curl_slist_append(headerList, ("Authorization: Bearer " + accessToken).c_str());
        headerList = curl_slist_append(headerList, "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8");

        curl_easy_setopt(pCurl, CURLOPT_HTTPHEADER, headerList);

        curl_easy_setopt(pCurl, CURLOPT_XFERINFOFUNCTION, TransferInfo);

        transProgress.curl = pCurl;
        curl_easy_setopt(pCurl, CURLOPT_XFERINFODATA, &transProgress);

        curl_easy_setopt(pCurl, CURLOPT_NOPROGRESS, 0L);

        // Now send the message
        res = curl_easy_perform(pCurl);

        fclose(pFile);
        curl_slist_free_all(headerList);
        curl_mime_free(multipart);

        if (res == CURLE_OK)
        {
            cout << "Firmware file successfully uploaded" << endl;
        }
        else
        {
            cout << "curl result ERROR = <" + to_string(res) + ": " + curl_easy_strerror(res) + ">" << endl;
            cout << "Failed to upload firmware file" << endl;
        }

        curl_easy_cleanup(pCurl);
    }

    return 0;
}

关于c++ - 为什么在发送几 block 数据后文件上传停止(使用 multipart/form-data)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55409192/

相关文章:

jquery - 如何使用 jquery 提交多部分表单数据

java - org.apache.tomcat.util.http.fileupload.FileUploadBase$FileSizeLimitExceededException

C++ 在循环内/提示时生成新随机数的问题

c++ - 如何在 Linux 中启用 C++11

c++,usleep() 已过时,Windows/MingW 的解决方法?

javascript - Microsoft Edge JS userAgent 检测不完整

c# - 通过表单发布上传图片

c++ - 如果 std::addressof 是 & 的可读版本。 *& 的可读版本是什么?

c++ - 将 "new"与initializer_list一起使用不会分配字符串值,而 int 值可以工作

windows - Windows x64 是否有任何 POSIX 兼容层?