#include "connection.h" #include <string.h> #include <stdio.h> #include <errno.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <netdb.h> #include <arpa/inet.h> Connection::Connection() { nSocket = -1; bActive = false; bDisconnectMe = false; pProtocol = NULL; } Connection::~Connection() { if( pProtocol != NULL ) delete pProtocol; } bool Connection::appendOutput( const char *lpOutput, int nSize ) { return xOutputBuf.appendData( lpOutput, nSize ); } bool Connection::appendOutput( const char lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const short lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const int lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const long lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const float lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const double lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const unsigned char lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const unsigned short lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const unsigned long lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendOutput( const unsigned int lOutput ) { return xOutputBuf.appendData( lOutput ); } bool Connection::appendInput( const char *lpInput, int nSize ) { return xInputBuf.appendData( lpInput, nSize ); } int Connection::scanInputFor( char cTarget ) { const char *lpTmp = xInputBuf.getData(); int jMax = xInputBuf.getLength(); for( int j = 0; j < jMax; j++ ) { if( lpTmp[j] == cTarget ) { return j; } } return -1; } const char *Connection::getOutput() { return xOutputBuf.getData(); } const char *Connection::getInput() { return xInputBuf.getData(); } void Connection::setSocket( int nNewSocket ) { nSocket = nNewSocket; } int Connection::getSocket() { return nSocket; } bool Connection::isActive() { return bActive; } void Connection::close() { if( bActive ) { fsync( nSocket ); ::close( nSocket ); } bActive = false; //nSocket = -1; xInputBuf.clearData(); xOutputBuf.clearData(); if( pProtocol != NULL ) { delete pProtocol; pProtocol = NULL; } } bool Connection::open( int nNewSocket ) { bActive = true; setSocket( nNewSocket ); bDisconnectMe = false; return true; } bool Connection::open( const char *sAddr, int nPort ) { struct sockaddr_in xServerName; bActive = false; /* Create the socket. */ nSocket = socket( PF_INET, SOCK_STREAM, 0 ); if( nSocket < 0 ) { bActive = false; return false; } /* Connect to the server. */ { struct hostent *hostinfo; xServerName.sin_family = AF_INET; xServerName.sin_port = htons( nPort ); hostinfo = gethostbyname( sAddr ); if (hostinfo == NULL) { return false; } xServerName.sin_addr = *(struct in_addr *) hostinfo->h_addr; } int ret = connect( nSocket, (struct sockaddr *)&xServerName, sizeof(xServerName) ); if( ret < 0 ) { return false; } bActive = true; bDisconnectMe = false; return true; } bool Connection::readInput() { char buffer[2048]; int nbytes; int nTotalRead=0; for(;;) { memset( buffer, 0, 2048 ); nbytes = read( nSocket, buffer, 2048 ); if (nbytes < 0) { /* Read error. */ //perror("readInput"); return false; } else if (nbytes == 0) { /* End-of-file. */ //perror("readInput"); return false; } else { nTotalRead += nbytes; appendInput( buffer, nbytes ); /* Data read. */ if( nbytes < 2047 ) { if( pProtocol != NULL && nTotalRead > 0 ) { pProtocol->onNewData(); } return true; } } } return true; } bool Connection::readInput( int nSec, int nUSec ) { fd_set rfds; struct timeval tv; int retval; /* Watch stdin (fd 0) to see when it has input. */ FD_ZERO(&rfds); FD_SET(nSocket, &rfds); /* Wait up to five seconds. */ tv.tv_sec = nSec; tv.tv_usec = nUSec; retval = select( nSocket+1, &rfds, NULL, NULL, &tv ); /* Don't rely on the value of tv now! */ if (retval == -1) { // Oh my god!!! some kind of horrible problem!!!! return false; } else if( retval ) { // None of them have data, but the connection is still active. return readInput(); } else { return true; } } bool Connection::clearOutput() { return xOutputBuf.clearData(); } bool Connection::clearInput() { return xInputBuf.clearData(); } #define min( a, b ) ((a<b)?(a):(b)) bool Connection::writeOutput() { int nBytes = TEMP_FAILURE_RETRY( write( nSocket, xOutputBuf.getData(), min( 2048, xOutputBuf.getLength() ) ) ); if( nBytes < 0 ) { perror("writeOutput"); return true; } /* if( nBytes < xOutputBuf.getLength() ) { printf("Havn't written all the data (%d/%d/%d%%)\n", nBytes, xOutputBuf.getLength(), nBytes/(xOutputBuf.getLength()*100) ); } else { printf("Wrote all pending data.\n"); } */ xOutputBuf.usedData( nBytes ); //clearOutput(); return true; } bool Connection::hasOutput() { if( xOutputBuf.getLength() == 0 ) { return false; } else { return true; } } bool Connection::hasInput() { if( xInputBuf.getLength() == 0 ) { return false; } else { return true; } } bool Connection::usedInput( int nAmount ) { return xInputBuf.usedData( nAmount ); } bool Connection::needDisconnect() { return bDisconnectMe; } void Connection::disconnect() { bDisconnectMe = true; } void Connection::setProtocol( class Protocol *pNewProtocol ) { pProtocol = pNewProtocol; pProtocol->setConnection( this ); } int Connection::getInputAmnt() { return xInputBuf.getLength(); } int Connection::getOutputAmnt() { return xOutputBuf.getLength(); } class Protocol *Connection::getProtocol() { return pProtocol; } void Connection::printInputDebug( const char *lpPrefix, FILE *fh, int nBytesMax ) { printDataDebug( (const unsigned char *)xInputBuf.getData(), xInputBuf.getLength(), "input", lpPrefix, fh, nBytesMax ); } void Connection::printOutputDebug( const char *lpPrefix, FILE *fh, int nBytesMax ) { printDataDebug( (const unsigned char *)xOutputBuf.getData(), xOutputBuf.getLength(), "output", lpPrefix, fh, nBytesMax ); } void Connection::printDataDebug( const unsigned char *pData, long nDataLen, const char *lpName, const char *lpPrefix, FILE *fh, int nBytesMax ) { if( nBytesMax > 0 ) { nDataLen = (nBytesMax<nDataLen)?(nBytesMax):(nDataLen); } fprintf( fh, "%sDisplaying %d bytes of %s.\n", lpPrefix, nDataLen, lpName ); int j = 0; fprintf( fh, lpPrefix ); for( int l = 0; l < 8*3+2*8+2; l++ ) fprintf( fh, (l!=8*3)?("-"):("+") ); fprintf( fh, "\n"); for(;;) { fprintf( fh, lpPrefix ); int kmax = 8; if( nDataLen-j < 8 ) kmax = nDataLen-j; for(int k = 0; k < 8; k++ ) { if( k < kmax ) { fprintf( fh, "%02X ", (int)((unsigned char)pData[j+k]) ); } else { fprintf( fh, "-- "); } } printf("| "); for(int k = 0; k < kmax; k++ ) { fprintf( fh, "%c ", (pData[j+k]>32 && pData[j+k]<=128)?(pData[j+k]):('.') ); } fprintf( fh, "\n"); j += kmax; if( j >= nDataLen ) break; } fprintf( fh, lpPrefix ); for( int l = 0; l < 8*3+2*8+2; l++ ) fprintf( fh, (l!=8*3)?("-"):("+") ); fprintf( fh, "\n"); }