HttpSendRequest - 不支持Unicode的POST数据

时间:2016-04-21 15:44:58

标签: c++ unicode

我正在制作一个C ++代理,它将使用HttpSendRequest()将信息(例如系统主机名)发布回中央服务器。我希望它回发的其中一条信息是操作系统。我创建了以下函数来获取系统主机名。

wstring getOS()
{
     HKEY key;
     RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", 0, KEY_QUERY_VALUE, &key); // Obtains Registry handle

     DWORD type;
     wchar_t buffer[MAX_PATH]; // MAX_PATH = 260 - The system hostname should never exceed this value
     DWORD size = sizeof(buffer);
     RegQueryValueEx(key, L"ProductName", NULL, &type, (LPBYTE)&buffer, &size); // Queries Registry key - stores value in "buffer"
     wstring os(buffer); // Converts from C-style character array to wstring
     return os; // Returns wstring to caller
}

此函数将使用注册表获取操作系统并将其存储为wstring。然后我想传递返回的" os" wstring到下面的post()函数,但是我注意到你必须使用字符串而不是wstring来表示HTTP POST数据。下面是我的post()函数的代码:

void post()
{
     HINTERNET hInternetOpen = InternetOpen(userAgent.c_str(), INTERNET_OPEN_TYPE_PROXY, L"http://127.0.0.1:9999", NULL, 0);
     HINTERNET hInternetConnect = InternetConnect(hInternetOpen, host.c_str(), INTERNET_DEFAULT_HTTP_PORT, NULL, NULL, INTERNET_SERVICE_HTTP, 0, 0);
     HINTERNET hHttpOpenRequest = HttpOpenRequest(hInternetConnect, L"POST", file.c_str(), NULL, NULL, NULL, 0, 0);

     wstring headers = L"Content-Type: application/x-www-form-urlencoded"; // Content-Type is necessary to POST

     string postData = "os="; // Why does this have to be a string and not a wstring?

     HttpSendRequest(hHttpOpenRequest, headers.c_str(), headers.length(), (LPVOID)postData.c_str(), postData.size());

     InternetCloseHandle(hInternetOpen);
     InternetCloseHandle(hInternetConnect);
     InternetCloseHandle(hHttpOpenRequest);
}

如果我尝试制作" postData"一个wstring,我得到的东西看起来像下图:

HTTP POST data as wstring

是否有人能够轻松地将wstring作为POST数据包含在内?

1 个答案:

答案 0 :(得分:3)

Collection只知道原始字节,而不是字符串。您可以使用HttpSendRequest()发送UTF-16数据,但您必须通过std::wstring标题中的charset属性告诉服务器您正在发送UTF-16。

Content-Type

请注意上面使用的wstring headers = L"Content-Type: application/x-www-form-urlencoded; charset=utf-16"; // TODO: don't forget to URL-encode the value from getOS() to // escape reserved characters, including '=' and '&'... wstring postData = L"os=" + getOS(); HttpSendRequest(hHttpOpenRequest, headers.c_str(), headers.length(), postData.c_str(), postData.length() * sizeof(wchar_t)); 。在您的屏幕截图中,您的嗅探器显示原始数据,它显示的数据是UTF-16的样子,但您只看到sizeof(wchar_t)数据的一半,因为您正在设置wstring dwOptionalLength参数字符数(7)而不是字节数(14):

  

dwOptionalLength [in]
  可选数据的大小,以字节为单位。如果没有可选的数据要发送,则此参数可以为零。

当您使用HttpSendRequest()时,字符数字节数是相同的值。

你真正应该发送的是UTF-8而不是UTF-16,例如:

std::string

string Utf8Encode(const wstring &wstr)
{
    // NOTE: C++11 has built-in support for converting between
    // UTF-8 and UTF-16.  See the std::wstring_convert class...
    /*
    wstring_convert<codecvt_utf8_utf16<wchar_t>> conv;
    return conv.to_bytes(wstr);
    */

    string out;
    int len = WideCharToMultiByte(CP_UTF8, 0, wstr.c_str(), wstr.length(), NULL, 0, NULL, NULL);
    if (len > 0)
    {
        out.resize(len);
        WideCharToMultiByte(CP_UTF8, 0, wstr.c_str(), wstr.length(), &out[0], len, NULL, NULL);
    }
    return out;
}