近日在用WinHttpSendRequest做文件上传的时候。
使用ashx接收文件数据时可以正常接收到文件数据,但是在php页面empty($_FILES)显示为TRUE,这一现象让我开始了1天的追查。
最终请求代码如下
dwContentLength = strlen(szpt2); ZeroMemory(wszContentLength,1024); wsprintf(wszContentLength, _T("Content-Length: %d"), dwContentLength); WinHttpAddRequestHeaders(hRequest, wszContentLength, (ULONG) -1L, WINHTTP_ADDREQ_FLAG_ADD | WINHTTP_ADDREQ_FLAG_REPLACE);
1
bResults = WinHttpSendRequest( hRequest,
2
WINHTTP_NO_ADDITIONAL_HEADERS,
3
NULL,
4
(LPSTR)szpt2,
5
dwContentLength,
6
0,//dwTotalLength 参数必须填写,除非这个请求不包含数据WINHTTP_NO_REQUEST_DATA or 0
7 0);
关键在红色部分,如果为0,在ashx文件中是仍然可以收到文件的。。。但是在php显示无文件数据。抓包显示确实提交了数据。
几经寻找尝试后发现dwTotalLength填写dwContentLength值时,php接收也正常了。
bResults = WinHttpSendRequest( hRequest, WINHTTP_NO_ADDITIONAL_HEADERS, NULL, (LPSTR)szpt2, dwContentLength, dwContentLength, //dwTotalLength 参数必须填写,除非这个请求不包含数据WINHTTP_NO_REQUEST_DATA or 0 0);
MSDN上对这个参数的描述是
dwTotalLength [in]
一个无符号长整数值,其中包含发送总数据的长度(以字节为单位)。此参数指定请求的Content-Length头。如果该参数的值大于由指定的长度大于 dwOptionalLength,然后 WinHttpWriteData可以被用来发送额外的数据。
dwTotalLength不能通话之间切换,,以WinHttpSendRequest了同样的要求。如果需要更改dwTotalLength,则调用者应该创建一个新的请求。
这时候大致猜测到是平台实现的差异了,抓包显示dwTotalLength不管是设置0还是dwContentLength,http报文显示的都是dwContentLength的长度
不过我用其他抓包工具Wireshark时疑似看到了2个Content-Length数据,由于问题已经解决所以就没有在确认是否真的出现了2个Content-Length。
总结:PHP在文件接收方面可能存在数据验证bug