HTTP multipart/form-data no sube archivo correctamente.

Iniciado por Kaxperday, 21 Marzo 2016, 17:10 PM

0 Miembros y 1 Visitante están viendo este tema.

Kaxperday

Bueno, pues estoy tratando de subir un archivo a una página web, que corre un script PHP..

El caso que he probado de todas las maneras posibles sin éxito alguno, he probado a hacer POST enviando en una variable el nombre del archivo y en otra el contenido del archivo con application/x-www-form-urlencoded y recibía el nombre correctamente pero el contenido mal (los bytes del jpg), me conseguía crear el archivo en el servidor pero estaba corrupto, en el PHP simplemente habría el archivo con ese nombre y escribía el contenido en binario, pero ya os digo el archivo ocupaba 80kb y escribió solo 1,9kbs si no mal recuerdo.

También probé a usar lo mismo pero con application/octet-stream, y bueno en este caso tenía que cambiar el script PHP, pero no me interesó realmente y no le dí muchas vueltas a este método, porque quiero subirlo o bien con POST con  application/x-www-form-urlencoded o con POST con multipart/form-data.

Visto que el primero escribía el 2% del archivo como que lo dejé y pase a multipart/form-data.
Copiando os datos de firefox, analizando paquetes de firefox con los de mi aplicación, como suben ambos el archivo, y SON JODIDAMENTE IGUALES EL BODY SOLO OJO, pero uno lo sube y otro no.

Os dejo el código de la función que sube un archivo y del PHP.

Código (php) [Seleccionar]

<?php 
if($_FILES["archivo"]["name"]){     
      if (
$_FILES["archivo"]["error"] > 0) { 
        echo 
$_FILES["archivo"]["error"] . "<br/>"
      } else { 
          if (
file_exists("" $_FILES["archivo"]["name"])) {
            echo 
$_FILES["archivo"]["name"] . " ya existe. "
          } else { 
            
move_uploaded_file($_FILES["archivo"]["tmp_name"], 
            
"" $_FILES["archivo"]["name"]);
            echo 
"Archivo Subido <br />"
          } 
      } 

?>



Código (cpp) [Seleccionar]

bool tracker::SendFile(std::string _path)
{
HINTERNET hInternet;

if ((hInternet = InternetOpenA(NULL, INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0)) == NO_ERROR){
return false;
}

if ((hInternet = InternetConnectA(hInternet, domain.c_str(), INTERNET_DEFAULT_HTTP_PORT,
NULL, NULL, INTERNET_SERVICE_HTTP, 0, 0)) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

DWORD requestFlags = INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTP |
INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTPS |
INTERNET_FLAG_KEEP_CONNECTION |
INTERNET_FLAG_NO_AUTO_REDIRECT |
INTERNET_FLAG_NO_COOKIES |
INTERNET_FLAG_NO_CACHE_WRITE |
INTERNET_FLAG_NO_UI |
INTERNET_FLAG_RELOAD;

if ((hInternet = HttpOpenRequestA(hInternet,
"POST",
"/catcher.php",
"HTTP/1.1",
NULL, NULL,
requestFlags, 0)) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

std::ifstream reader("C:\\Users\\User\\Desktop\\ciclista.jpg", std::ifstream::ate | std::ifstream::binary);
int size = reader.tellg();
char* buffer = new char[size];
reader.seekg(0, std::ios::beg);
reader.read(buffer, size);
reader.close();

std::string content;
content = std::string(buffer, size);
delete[] buffer;

std::string body;
std::string boundary = "-----------------------------268991947030948";

/*Body*/
body += boundary + "\r\n";
body += "Content-Disposition: form-data; name=\"archivo\"; filename=\"ciclista.jpg\"\r\n";
body += "Content-Type: image/jpeg\r\n\r\n";
body += content + "\r\n";
body += boundary + "\r\n";
body += "Content-Disposition: form-data; name=\"boton\"\r\n\r\n";
body += "Enviar archivo\r\n";
body += boundary + "--\r\n";

/*Header*/
std::string headers = "Content-Type: multipart/form-data; boundary=" + boundary + "\r\n";
HttpAddRequestHeadersA(hInternet, headers.c_str(), headers.length(), HTTP_ADDREQ_FLAG_ADD);

/*Body*/
std::cout << body.substr(0, 600) << body.substr(body.length() - 200, 200);
std::cout << body.length() << std::endl;

if (HttpSendRequestA(hInternet, NULL, 0, (LPVOID)body.c_str(), body.length()) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

std::string response;
DWORD dwBytes;
char ch;

while (InternetReadFile(hInternet, &ch, 1, &dwBytes))
{
if (dwBytes != 1) break;
response += ch;
}

InternetCloseHandle(hInternet);
std::cout << response;
}


Bueno me he vuelto loco comparando el contenido de los paquetes, y solo puede fallar la cabecera (luego a saber que será), pero el Content-Length es igual en firefox que en mi función, luego si cambia algo sería la cabecera que tiene HOST, CONTENT-LENGTH, CONTENT-TYPE, CONNECTION KEEP ALIVE, CACHE-CONTROL NO CACHE. Al acabar el header \r\n\r\n como siempre y mando el body, ahí está el código, cuando recibo la respuesta del server recibo solo el HTML sin el "se ha subido correctamente" o "ya existe". Pero no hay errores en el PHP.

¿Qué pasa?.

Saludos :X
Cuando el poder económico parasita al político ningún partido ni dictador podrá liberarnos de él. Se reserva el 99% ese poder.

Kaxperday

#1
En serio, volviendo al tema que me tiene loco:

Esta vez he intentado subir un simple archivo de texto txt llamado logger.txt con este contenido:

Citarhola que buenos esto es una jodida pruebaaa

FF lo manda perfecto como sieeempre, y yo mando prácticamente lo mismo pero no me funciona, el body (los datos que mando) son IGUALES. El content-length es el mismo Y HE ABIERTO 2 WIRESHARK PARA DIVIDIR LA PANTALLA Y COMPROBAR BYTE A BYTE EL BODY, Y SI SON ** IGUALES.

Tambien el header HTTP es igual que el de FF, solo que yo incluyo un header mas el de Cache-Control= no-cache, que supongo es algo totalmente irrelevante.

Peeero si que capta una diferencia el wireshark, y es que si nos vamos al campo frame del paquete en el de firefox pone:

Citareth:ethertype:ip:tcp:http:mime_multipart:data-text-lines:data

Y en el mío pone:

Citareth:ethertype:ip:tcp:http:mime_multipart:data

Es como que se come el data-text que es un protocolo dentro de http que esta dentro de mime_multipart, ¿pero si lo que mando es igual, excepto una cabecera más indiferente, y si todo igual, porqué eso es distinto porqueeee? :@

¿Será la winapi? ¿será la magia? ¿qué sera esta vez?.

Saludos y PD siento el doble post, tomadlo como una respuesta a mi mismo XD.

Y para decorar el pastel os paso el código actualizado:

Código (cpp) [Seleccionar]

bool Tracker::SendFile(std::wstring _path)
{
HINTERNET hInternet;

if ((hInternet = InternetOpenA(NULL, INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0)) == NO_ERROR){
return false;
}

if ((hInternet = InternetConnectA(hInternet, domain.c_str(), INTERNET_DEFAULT_HTTP_PORT,
NULL, NULL, INTERNET_SERVICE_HTTP, 0, 0)) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

if ((hInternet = HttpOpenRequestA(hInternet, "POST", "/catcher.php", "HTTP/1.1",
NULL, NULL, NULL, 0)) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

HANDLE handle;
DWORD size;

if ((handle = CreateFile(_path.c_str(), GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_HIDDEN, 0)) == INVALID_HANDLE_VALUE){
InternetCloseHandle(hInternet);
return false;
}

if ((size = GetFileSize(handle, NULL)) == INVALID_FILE_SIZE){
CloseHandle(handle);
InternetCloseHandle(hInternet);
return false;
}

char *buffer = new char[size]();

if (ReadFile(handle, buffer, size, NULL, NULL) == 0){
CloseHandle(handle);
InternetCloseHandle(hInternet);
return false;
}

CloseHandle(handle);
std::string data = std::string(buffer, size);
delete[] buffer;

std::string boundary = "-----------------------------68991947030948";
std::string headers;

headers += "Host: web.com\r\n";
headers += "User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64; rv:42.0) Gecko/20100101 Firefox/42.0\r\n";
headers += "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\r\n";
headers += "Accept-Language: es-ES,es;q=0.8,en-US;q=0.5,en;q=0.3\r\n";
headers += "Accept-Encoding: gzip, deflate\r\n";
headers += "DNT: 1\r\n";
headers += "Referer: http://web.com/catcher.php\r\n";
headers += "Connection: keep-alive\r\n";
headers += "Content-Type: multipart/form-data; boundary=" + boundary + "\r\n";
HttpAddRequestHeadersA(hInternet, headers.c_str(), headers.length(), HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE);

std::string body;
body += boundary + "\r\n";
body += "Content-Disposition: form-data; name=\"archivo\"; filename=\"logger.txt\"\r\n";
body += "Content-Type: text/plain\r\n\r\n";
body += data;
body += "\r\n" + boundary + "\r\n";
body += "Content-Disposition: form-data; name=\"boton\"\r\n\r\n";
body += "Enviar archivo";
body += "\r\n" + boundary + "--\r\n";

std::cout << body;
//std::cout << body.substr(0, 800) << body.substr(body.length() - 600, 600);
//std::cout << body.length() << std::endl;

if (HttpSendRequestA(hInternet, NULL, 0, (LPVOID)body.c_str(), body.length()) == NO_ERROR){
InternetCloseHandle(hInternet);
return false;
}

std::string response;
DWORD dwBytes;
char ch;

while (InternetReadFile(hInternet, &ch, 1, &dwBytes))
{
if (dwBytes != 1) break;
response += ch;
}

InternetCloseHandle(hInternet);
std::cout << response;
}


El PHP sigue igual y me carga el html de la página pero no me sale el mensajito de "se ha subido el archivo" ni "ya  hay otro con ese nombre", algo falla.

Y es que en wireshark la propia descripción del paquete en el de FF pone:

CitarPOST /catcher.php HTTP/1.1  (text/plain)

Y en el mío:

CitarPOST /catcher.php HTTP/1.1

EDITO: Hola lo he solucionado, adios. ¬¬.

Cuando el poder económico parasita al político ningún partido ni dictador podrá liberarnos de él. Se reserva el 99% ese poder.