Datei downloaden probleme mit dem HTTP Header



  • Hallo
    ich hab ein Problem.. ich will mit winsock eine Datei von einem Server downloaden.
    würde ja auch gehen. doch ein problem gibt es leider. Wen man an einen Server ja eine Anfrage schickt, bekommt man noch einen HTTP Header nun ist mein Problem, wie ich den wegschneiden soll. nun mit einem recv() das vor dem eigentlichen Austausch aufgerufen wird geht das ja, doch leider ist der Header nicht immer gleich gross. Also, weiss jemand wie ich den Header wegschneiden könnte?

    burnner



  • Der Header ist doch vom den Seitendaten mit \r\n\r\n getrennt.



  • und wie kann ich das jetzt einbauen?
    ich empfange die Daten in einem char buffer.
    nur weiss ich jetzt nicht wie ich da den header genau auschneiden soll.



  • so wäre ne möglichkeit

    so in Http_All kommt der string der alle daten enthält
    in Http_Header hast du anschließend den header
    in Http_Data hast anschließend deine nutzdaten

    std::string Http_Data;
      std::string::size_type qm = Http_All.find("\r\n\r\n");
      if (qm != std::string::npos) 
      {
        std::string Http_Header = Http_All.substr(0, qm);
    
        Http_Data = Http_All.substr(qm+1);
      }
    


  • ah ich vergass zu erwähnen.. es sind binäre dateien, also Programme und Bilder, darum hab ich da skrubel std::string zu verwenden



  • oder gibt es eine andere Möglichkeit Dateien herunterzuladen?



  • ich würd nicht erst die ganze antwort lesen und dann rausschneiden sondern immer die datenblöcke sofort parsen.



  • ja aber ist die grösse des headers nicht variabel? und wie soll ich rausschneiden? soll ja binär bleiben damit man danach die exe datei wieder ausführen kann



  • ja ist variabel, aber du kennst ja den trenner.
    du kannst doch eine funktion schreiben die einen datenblock übergeben bekommt und dann guckt wieviel bytes davon zum header gehören und sich diese dann z.B. in einen std::string kopiert oder ignoriert. wenn der header dann irgendwann vollständig da ist, kannst du in einer statusvariablen speichern das du jetzt die eigentlichen daten empfangen willst.



  • ja aber naja am meisten probleme hab ich damit. Wen ich nun eine Char Variabel habe und weiss wo sich der header befindet, wie kann ich den nun rausschneiden?
    zum speichern nutze ich die fstream klasse. nur da kann man ja nur angeben wie viel gespeichert wird und nich von wo bis wo.

    mfg



  • outfile.write(VonWo, Wieviel);



  • ? write braucht doch ein (char*) da kann ich doch nicht einfach 2 zahlen angeben



  • du suchst im buffer nach dem trenner, dann weist du position
    dann könntest du es in einen neuen buffer kopieren (strncpy).
    wo happert es den genau. zeig doch mal deinen ansatz.



  • So hab ich mir den Code fürs Parsen vorgestellt. Er ist höchstwahrscheinlich falsch und eventuell kann man es einfacher machen.

    const unsigned int STATE_0 = 0;
    const unsigned int STATE_R = 1;
    const unsigned int STATE_RN = 2;
    const unsigned int STATE_RNR = 3;
    const unsigned int STATE_RNRN = 4;
    
    unsigned int state = STATE_0;
    
    void parse(const char* pBuffer, size_t bufferSize)
    {
    	for(size_t i = 0; i < bufferSize; ++i)
    	{
    		if(pBuffer[i] == '\r')
    		{
    			if(state == STATE_0)
    			{
    				state = STATE_R;
    			}
    			else if(state == STATE_RN)
    			{
    				state = STATE_RNR;
    			}
    			else
    			{
    				state = STATE_0;
    			}
    		}
    		else if(pBuffer[i] == '\n')
    		{
    			if(state == STATE_R)
    			{
    				state = STATE_RN;
    			}
    			else if(state == STATE_RNR)
    			{
    				state = STATE_RNRN;
    			}
    			else
    			{
    				state = STATE_0;
    			}
    		}
    		else
    		{
    			state = STATE_0;
    		}
    
    		if(state == STATE_RNRN)
    		{
    			std::cout << "Header complete" << std::endl;
    			break;
    		}
    	}
    }
    

    Ich wüsste nicht wie man es sonst machen kann.



  • Hi all,

    wozu den ganze Stress???

    Wenn du von einer URL eine Datei downloaden willst, benutze einfach die Funktion
    "URLDownloadToFileA()" aus "urlmon.dll"

    geht viel einfacher 🙄



  • char buffer[1000];
    char buffer2[1000];
    
    rc=recv(s,buffer,1000,0);
    temp = buffer;
    pos = temp.find("\r\n\r");
    
    if(pos != std::string::npos)
    {
    	temp.erase(0, pos + 4);
    	strcpy(buffer2, (char*)temp.c_str());
    	out.write((char*)buffer2, temp.length());
    }
    

    das ist mein ansatzt, funktioniert eben wie erwartet nicht





  • burnner es ist nicht garantiert das \r\n\r\n in genau einem Aufruf von recv bekommst. Es könnte im "schlimmsten" Fall sein das du mit jedem Aufruf von recv nur ein Zeichen bekommst.

    Aber ich glaub du benutzt besser die Funktionen aus der WinInet Library. InternetXXX.



  • Keine Ahnung scheint Ahnung zu haben wie man es sich einfach macht. 😉



  • so:

    char url[] = "http:/bla.com/source.exe";
    char file[] = "C:\\destin.exe";

    URLDownloadToFile(NULL, url, file, NULL, NULL);

    irgenwie so muss es in C++ aussehen 😃

    sorry aber ich komme aus win32-Assemblerprogrammierer-Welt 😉


Anmelden zum Antworten