From 90c05874265e7047c0ca933a43c433eb0d7f04e4 Mon Sep 17 00:00:00 2001 From: spiralman Date: Sat, 15 Mar 2003 15:27:49 +0000 Subject: moved from using QSockets to normal unix sockets, images and chunked encoding **should** work --- (limited to 'noncore/net/ubrowser/httpfactory.cpp') diff --git a/noncore/net/ubrowser/httpfactory.cpp b/noncore/net/ubrowser/httpfactory.cpp index 50a3c9a..0586e96 100644 --- a/noncore/net/ubrowser/httpfactory.cpp +++ b/noncore/net/ubrowser/httpfactory.cpp @@ -18,10 +18,10 @@ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA HttpFactory::HttpFactory(QTextBrowser *newBrowser):QMimeSourceFactory() { - socket = new QSocket; +// socket = new QSocket; text = new QTextDrag; browser=newBrowser; - comm = new HttpComm(socket, browser); +// comm = new HttpComm(socket, browser); image = new QImageDrag; } @@ -31,9 +31,9 @@ const QMimeSource * HttpFactory::data(const QString &abs_name) const int port=80, addrEnd, portSep; QString host, file, portS, name, tempString; - bool done=false, isImage=false; + bool done=false, isText=true; - comm->setUp((QString *)&abs_name); +// comm->setUp((QString *)&abs_name); name = abs_name; // name = name.lower(); @@ -80,22 +80,86 @@ const QMimeSource * HttpFactory::data(const QString &abs_name) const portS="80"; } - if(file.find(".png", file.length()-4) != -1 || file.find(".gif", file.length()-4) != -1 || file.find(".jpg", file.length()-4) != -1) - { - isImage=true; - } +// if(file.find(".png", file.length()-4) != -1 || file.find(".gif", file.length()-4) != -1 || file.find(".jpg", file.length()-4) != -1) +// { +// isImage=true; +// } - comm->setStuff(host, portS, file, text, image, isImage); +// comm->setStuff(host, portS, file, text, image, isImage); - socket->connectToHost(host, port); +// socket->connectToHost(host, port); + + int con, bytesSent; + struct sockaddr_in serverAddr; + struct hostent * serverInfo = gethostbyname( host.latin1() ); + + if( serverInfo == NULL ) + { + QMessageBox *mb = new QMessageBox("Error!", + "couldnt find ip address", + QMessageBox::NoIcon, + QMessageBox::Ok, + QMessageBox::NoButton, + QMessageBox::NoButton); + mb->exec(); + perror("HttpFactory::data:"); + return 0; + } + + QByteArray data; + printf( "HttpFactory::data: %s\n", inet_ntoa(*((struct in_addr *)serverInfo->h_addr )) ); + + QString request("GET " + file + " HTTP/1.1\r\nHost: " + host + ':' + portS + "\r\nConnection: close\r\n\r\n"); + + con = socket( AF_INET, SOCK_STREAM, 0 ); + if( con == -1 ) + { + QMessageBox *mb = new QMessageBox("Error!", + "couldnt create socket", + QMessageBox::NoIcon, + QMessageBox::Ok, + QMessageBox::NoButton, + QMessageBox::NoButton); + mb->exec(); + perror("HttpFactory::data:"); + return 0; + } + + serverAddr.sin_family = AF_INET; + serverAddr.sin_port = htons( port ); + serverAddr.sin_addr.s_addr = inet_addr( inet_ntoa(*((struct in_addr *)serverInfo->h_addr )) ); + memset( &(serverAddr.sin_zero), '\0', 8 ); + + if(::connect( con, (struct sockaddr *)&serverAddr, sizeof(struct sockaddr)) == -1 ) + { + QMessageBox *mb = new QMessageBox("Error!", + "couldnt connect to socket", + QMessageBox::NoIcon, + QMessageBox::Ok, + QMessageBox::NoButton, + QMessageBox::NoButton); + mb->exec(); + perror("HttpFactory::data:"); + return 0; + } - if(!image) + + bytesSent = send( con, request.latin1(), request.length(), 0); + printf("HttpFactory::data: bytes written: %d out of: %d\n", bytesSent, request.length() ); + printf("HttpFactory::data: request sent:\n%s", request.latin1()); + + data = this->processResponse( con, isText ); + + ::close( con ); + + if(isText) { - text->setText(""); + text->setText( QString( data ) ); return text; } else { + image->setImage( QImage( data ) ); return image; } } @@ -115,3 +179,187 @@ const QMimeSource * HttpFactory::data(const QString &abs_or_rel_name, const QStr return 0; } + +const QByteArray HttpFactory::processResponse( int sockfd, bool &isText ) const +{ + QByteArray inputBin( 1 ); + QByteArray conClosedErr( 27 ); + char conClosedErrMsg[] = "Connection to server closed"; + QString currentLine; + bool done=false, chunked=false; + int dataLength = 0; + + for( int i = 0; i < 27; i++) + { + conClosedErr[i] = conClosedErrMsg[i]; + } + + while( !done ) + { + recv( sockfd, inputBin.data(), inputBin.size(), 0 ); + currentLine += *(inputBin.data()); + + if( *(inputBin.data()) == '\n' ) + { + if( currentLine.isEmpty() || currentLine.startsWith( "\r") ) + { + printf( "HttpFactory::processResponse: end of header\n" ); + if( chunked ) + { + return recieveChunked( sockfd ); + } else { + return recieveNormal( sockfd, dataLength ); + } + done = true; + } + + if( currentLine.contains( "Transfer-Encoding: chunked", false) >= 1 ) + { + chunked = true; + printf( "HttpFactory::processResponse: chunked encoding\n" ); + } + + if( currentLine.contains( "Content-Type: text", false ) >= 1 ) + { + isText = true; + printf( "HttpFactory::processResponse: content type text\n" ); + } + + if( currentLine.contains( "Content-Type: image", false ) >= 1 ) + { + isText = false; + printf( "HttpFactory::processResponse: content type image\n" ); + } + + if( currentLine.contains( "Content-Length", false ) >= 1 ) + { + currentLine.remove( 0, 16 ); + dataLength = currentLine.toInt(); + printf( "HttpFactory::processResponse: content length: %d\n", dataLength ); + } + + currentLine = ""; + printf("HttpFactory::processResponse: reseting currentLine: %s\n", currentLine.latin1() ); + } + } +} + +const QByteArray HttpFactory::recieveNormal( int sockfd, int dataLen ) const +{ + printf( "HttpFactory::recieveNormal: recieving w/out chunked encoding\n" ); + + QByteArray data( dataLen ); + QByteArray temp( dataLen ); + int recieved, i; + + recieved = recv( sockfd, temp.data(), temp.size(), 0 ); + printf( "HttpFactory::recieveNormal: found some data: %s\n", (char *)temp.data() ); + for( i = 0; i < recieved; i++ ) + { + data[i] = temp[i]; + } + dataLen -= recieved; + while( dataLen > 0 ) + { + recieved = recv( sockfd, temp.data(), temp.size(), 0 ); + dataLen -= recieved; +// printf( "HttpFactory::recieveNormal: found some more data: %s\n", (char *)temp.data() ); + for( int j = 0; j < recieved; j++ ) + { + data[i] = temp[j]; + i++; + } + temp.fill('\0'); + } + + printf( "HttpFactory::recieveNormal: end of data\n" ); + return data; +} + +const QByteArray HttpFactory::recieveChunked( int sockfd ) const +{ + printf( "HttpFactory::recieveChunked: recieving data with chunked encoding\n" ); + + QByteArray data; + QByteArray temp( 1 ); + int recieved, i = 0, cSize = 0; + QString cSizeS; + + printf( "HttpFactory::recieveChunked: temp.size(): %d\n", temp.size() ); + recv( sockfd, temp.data(), temp.size(), 0 ); + while( *(temp.data()) != '\n' && *(temp.data()) != ';' ) + { +// printf( "HttpFactory::recieveChunked: temp.size(): %d\n", temp.size() ); +// printf( "HttpFactory::recieveChunked: temp.data(): %c\n", temp[0] ); + cSizeS += temp[0]; + recv( sockfd, temp.data(), temp.size(), 0 ); + } + + printf( "HttpFactory::recieveChunked: cSizeS: %s\n", cSizeS.latin1() ); + cSize = cSizeS.toInt( 0, 16 ); + printf( "HttpFactory::recieveChunked: first chunk of size: %d\n", cSize ); + + if( *(temp.data()) == ';' ) + { + while( *(temp.data()) != '\n' ) + { + recv( sockfd, temp.data(), temp.size(), 0 ); + } + } + + temp.fill( '\0', cSize ); + data.fill( '\0', cSize ); + + while( cSize > 0 ) + { + while( cSize > 0 ) + { + recieved = recv( sockfd, temp.data(), temp.size(), 0 ); + cSize -= recieved; + for( int j = 0; j < recieved; j++ ) + { + data[i] = temp[j]; + i++; + } + temp.fill('\0', cSize); + } + + printf( "HttpFactory::recieveChunked: current data:\n%s", data.data() ); + + temp.fill('\0', 1); + cSizeS = ""; + cSize = 0; + + recv( sockfd, temp.data(), temp.size(), 0 ); + if( *(temp.data()) == '\r' ) + { + recv( sockfd, temp.data(), temp.size(), 0 ); + } + recv( sockfd, temp.data(), temp.size(), 0 ); + while( *(temp.data()) != '\n' && *(temp.data()) != ';' ) + { +// printf( "HttpFactory::recieveChunked: temp.size(): %d\n", temp.size() ); + printf( "HttpFactory::recieveChunked: temp.data(): %d\n", temp[0] ); + cSizeS += temp[0]; + recv( sockfd, temp.data(), temp.size(), 0 ); + } + + printf( "HttpFactory::recieveChunked: cSizeS: %s\n", cSizeS.latin1() ); + cSize = cSizeS.toInt( 0, 16 ); + printf( "HttpFactory::recieveChunked: next chunk of size: %d\n", cSize ); + + if( *(temp.data()) == ';' ) + { + while( *(temp.data()) != '\n' ) + { + recv( sockfd, temp.data(), temp.size(), 0 ); + } + } + + temp.fill( '\0', cSize ); + data.resize( data.size() + cSize ); + } + + printf( "HttpFactory::recieveChunked: end of data\n" ); + return data; +} -- cgit v0.9.0.2