diff options
Diffstat (limited to 'tqtinterface/qt4/src/network')
18 files changed, 0 insertions, 15979 deletions
diff --git a/tqtinterface/qt4/src/network/qt_network.pri b/tqtinterface/qt4/src/network/qt_network.pri deleted file mode 100644 index 6b8986a..0000000 --- a/tqtinterface/qt4/src/network/qt_network.pri +++ /dev/null @@ -1,23 +0,0 @@ -# Qt network module - -network { - HEADERS += $$NETWORK_H/tqdns.h \ - $$NETWORK_H/tqftp.h \ - $$NETWORK_H/tqhttp.h \ - $$NETWORK_H/tqhostaddress.h \ - $$NETWORK_H/tqnetwork.h \ - $$NETWORK_H/tqserversocket.h \ - $$NETWORK_H/tqsocket.h \ - $$NETWORK_H/tqsocketdevice.h - NETWORK_SOURCES = $$NETWORK_CPP/tqdns.cpp \ - $$NETWORK_CPP/tqftp.cpp \ - $$NETWORK_CPP/tqhttp.cpp \ - $$NETWORK_CPP/tqhostaddress.cpp \ - $$NETWORK_CPP/tqnetwork.cpp \ - $$NETWORK_CPP/tqserversocket.cpp \ - $$NETWORK_CPP/tqsocket.cpp \ - $$NETWORK_CPP/tqsocketdevice.cpp - unix:NETWORK_SOURCES += $$NETWORK_CPP/tqsocketdevice_unix.cpp - win32:NETWORK_SOURCES += $$NETWORK_CPP/tqsocketdevice_win.cpp - SOURCES += $$NETWORK_SOURCES -} diff --git a/tqtinterface/qt4/src/network/tqdns.cpp b/tqtinterface/qt4/src/network/tqdns.cpp deleted file mode 100644 index 4ba3a85..0000000 --- a/tqtinterface/qt4/src/network/tqdns.cpp +++ /dev/null @@ -1,5294 +0,0 @@ -#include "tqtglobaldefines.h" - -#ifdef USE_QT4 - -/**************************************************************************** -** -** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). -** All rights reserved. -** Contact: Nokia Corporation ([email protected]) -** -** This file is part of the Qt3Support module of the Qt Toolkit. -** -** $QT_BEGIN_LICENSE:LGPL$ -** Commercial Usage -** Licensees holding valid Qt Commercial licenses may use this file in -** accordance with the Qt Commercial License Agreement provided with the -** Software or, alternatively, in accordance with the terms contained in -** a written agreement between you and Nokia. -** -** GNU Lesser General Public License Usage -** Alternatively, this file may be used under the terms of the GNU Lesser -** General Public License version 2.1 as published by the Free Software -** Foundation and appearing in the file LICENSE.LGPL included in the -** packaging of this file. Please review the following information to -** ensure the GNU Lesser General Public License version 2.1 requirements -** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. -** -** In addition, as a special exception, Nokia gives you certain additional -** rights. These rights are described in the Nokia Qt LGPL Exception -** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. -** -** GNU General Public License Usage -** Alternatively, this file may be used under the terms of the GNU -** General Public License version 3.0 as published by the Free Software -** Foundation and appearing in the file LICENSE.GPL included in the -** packaging of this file. Please review the following information to -** ensure the GNU General Public License version 3.0 requirements will be -** met: http://www.gnu.org/copyleft/gpl.html. -** -** If you have questions regarding the use of this file, please contact -** Nokia at [email protected]. -** $QT_END_LICENSE$ -** -****************************************************************************/ - -// #include "Qt/qplatformdefs.h" - -#include "Qt/qbytearray.h" -#if defined(Q_OS_WIN32) || defined(Q_OS_WINCE) || defined(Q_OS_CYGWIN) -# include "qt_windows.h" -#else -# include <sys/types.h> -# include <netinet/in.h> -# include <arpa/nameser.h> -# include <resolv.h> -extern "C" int res_init(); -#endif - -// POSIX Large File Support redefines open -> open64 -#if defined(open) -# undef open -#endif - -// POSIX Large File Support redefines truncate -> truncate64 -#if defined(truncate) -# undef truncate -#endif - -// Solaris redefines connect -> __xnet_connect with _XOPEN_SOURCE_EXTENDED. -#if defined(connect) -# undef connect -#endif - -// UnixWare 7 redefines socket -> _socket -#if defined(socket) -# undef socket -#endif - -#include "tqdns.h" - -#ifndef QT_NO_DNS - -#include "Qt/qdatetime.h" -#include "tqdict.h" -#include "tqptrlist.h" -#include "Qt/qstring.h" -#include "Qt/qtimer.h" -#include "Qt/qapplication.h" -#include "tqptrvector.h" -#include "tqstrlist.h" -#include "tqptrdict.h" -#include "Qt/qfile.h" -#include "Qt/qtextstream.h" -#include "tqsocketdevice.h" -#include "tqcleanuphandler.h" -#include <limits.h> - -// #include "Qt/qhostaddress.h" -// #include "Qt/qsocketnotifier.h" - -QT_BEGIN_NAMESPACE - -//#define TQDNS_DEBUG - -static Q_UINT16 theId; // ### seeded started by now() - - -static QDateTime * originOfTime = 0; - -static TQCleanupHandler<QDateTime> q3dns_cleanup_time; - -static Q_UINT32 now() -{ - if ( originOfTime ) - return originOfTime->secsTo( QDateTime::currentDateTime() ); - - originOfTime = new QDateTime( QDateTime::currentDateTime() ); - theId = originOfTime->time().msec() * 60 + originOfTime->time().second(); - q3dns_cleanup_time.add( &originOfTime ); - return 0; -} - - -static TQPtrList<TQHostAddress> * theNs = 0; -static TQStrList * theDomains = 0; -static bool ipv6support = false; - -class TQDnsPrivate { -public: - TQDnsPrivate() : queryTimer( 0 ), noNames(false) - { -#if defined(Q_DNS_SYNCHRONOUS) -#if defined(Q_OS_UNIX) - noEventLoop = qApp==0 || qApp->loopLevel()==0; -#else - noEventLoop = false; -#endif -#endif - } - ~TQDnsPrivate() - { - delete queryTimer; - } -private: - TQTimer * queryTimer; - bool noNames; -#if defined(Q_DNS_SYNCHRONOUS) - bool noEventLoop; -#endif - - friend class TQDns; - friend class TQDnsAnswer; -}; - - -class TQDnsRR; -class TQDnsDomain; - - - -// TQDnsRR is the class used to store a single RR. TQDnsRR can store -// all of the supported RR types. a TQDnsRR is always cached. - -// TQDnsRR is mostly constructed from the outside. a but hacky, but -// permissible since the entire class is internal. - -class TQDnsRR { -public: - TQDnsRR( const TQString & label ); - ~TQDnsRR(); - -public: - TQDnsDomain * domain; - TQDns::RecordType t; - bool nxdomain; - bool current; - Q_UINT32 expireTime; - Q_UINT32 deleteTime; - // somewhat space-wasting per-type data - // a / aaaa - TQHostAddress address; - // cname / mx / srv / ptr - TQString target; - // mx / srv - Q_UINT16 priority; - // srv - Q_UINT16 weight; - Q_UINT16 port; - // txt - TQString text; // could be overloaded into target... -private: - -}; - - -class TQDnsDomain { -public: - TQDnsDomain( const TQString & label ); - ~TQDnsDomain(); - - static void add( const TQString & label, TQDnsRR * ); - static TQPtrList<TQDnsRR> * cached( const TQDns * ); - - void take( TQDnsRR * ); - - void sweep( Q_UINT32 thisSweep ); - - bool isEmpty() const { return rrs == 0 || rrs->isEmpty(); } - - TQString name() const { return l; } - -public: - TQString l; - TQPtrList<TQDnsRR> * rrs; -}; - - -class TQDnsQuery: public TQTimer { // this inheritance is a very evil hack -public: - TQDnsQuery(): - id( 0 ), t( TQDns::None ), step(0), started(0), - dns( new TQPtrDict<void>(17) ) {} - ~TQDnsQuery() { delete dns; } - Q_UINT16 id; - TQDns::RecordType t; - TQString l; - - uint step; - Q_UINT32 started; - - TQPtrDict<void> * dns; -}; - - - -class TQDnsAnswer { -public: - TQDnsAnswer( TQDnsQuery * ); - TQDnsAnswer( const TQByteArray &, TQDnsQuery * ); - ~TQDnsAnswer(); - - void parse(); - void notify(); - - bool ok; - -private: - TQDnsQuery * query; - - Q_UINT8 * answer; - int size; - int pp; - - TQPtrList<TQDnsRR> * rrs; - - // convenience - int next; - int ttl; - TQString label; - TQDnsRR * rr; - - TQString readString(bool multipleLabels = true); - void parseA(); - void parseAaaa(); - void parseMx(); - void parseSrv(); - void parseCname(); - void parsePtr(); - void parseTxt(); - void parseNs(); -}; - - -TQDnsRR::TQDnsRR( const TQString & label ) - : domain( 0 ), t( TQDns::None ), - nxdomain( false ), current( false ), - expireTime( 0 ), deleteTime( 0 ), - priority( 0 ), weight( 0 ), port( 0 ) -{ - TQDnsDomain::add( label, this ); -} - - -// not supposed to be deleted except by TQDnsDomain -TQDnsRR::~TQDnsRR() -{ - // nothing is necessary -} - - -// this one just sticks in a NXDomain -TQDnsAnswer::TQDnsAnswer( TQDnsQuery * query_ ) -{ - ok = true; - - answer = 0; - size = 0; - query = query_; - pp = 0; - rrs = new TQPtrList<TQDnsRR>; - rrs->setAutoDelete( false ); - next = size; - ttl = 0; - label.clear(); - rr = 0; - - TQDnsRR * newrr = new TQDnsRR( query->l ); - newrr->t = query->t; - newrr->deleteTime = query->started + 10; - newrr->expireTime = query->started + 10; - newrr->nxdomain = true; - newrr->current = true; - rrs->append( newrr ); -} - - -TQDnsAnswer::TQDnsAnswer( const TQByteArray& answer_, - TQDnsQuery * query_ ) -{ - ok = true; - - answer = (Q_UINT8 *)(answer_.data()); - size = (int)answer_.size(); - query = query_; - pp = 0; - rrs = new TQPtrList<TQDnsRR>; - rrs->setAutoDelete( false ); - next = size; - ttl = 0; - label.clear(); - rr = 0; -} - - -TQDnsAnswer::~TQDnsAnswer() -{ - if ( !ok && rrs ) { - TQPtrListIterator<TQDnsRR> it( *rrs ); - TQDnsRR * tmprr; - while( (tmprr=it.current()) != 0 ) { - ++it; - tmprr->t = TQDns::None; // will be deleted soonish - } - } - delete rrs; -} - - -TQString TQDnsAnswer::readString(bool multipleLabels) -{ - int p = pp; - TQString r; - Q_UINT8 b; - for( ;; ) { - b = 128; - // Read one character - if ( p >= 0 && p < size ) - b = answer[p]; - - switch( b >> 6 ) { - case 0: - // b is less than 64 - p++; - - // Detect end of data - if ( b == 0 ) { - if ( p > pp ) - pp = p; - return r.isNull() ? TQT_TQSTRING(TQLatin1String( "." )) : r; - } - - // Read a label of size 'b' characters - if ( !r.isNull() ) - r += QLatin1Char('.'); - while( b-- > 0 ) - r += QLatin1Char( answer[p++] ); - - // Return immediately if we were only supposed to read one - // label. - if (!multipleLabels) - return r; - - break; - default: - // Ignore unrecognized control character, or p was out of - // range. - goto not_ok; - case 3: - // Use the next character to determine the relative offset - // to jump to before continuing the packet parsing. - int q = ( (answer[p] & 0x3f) << 8 ) + answer[p+1]; - - if ( q >= pp || q >= p ) - goto not_ok; - if ( p >= pp ) - pp = p + 2; - p = q; - } - } -not_ok: - ok = false; - return TQString(); -} - - - -void TQDnsAnswer::parseA() -{ - if ( next != pp + 4 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN A for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::A; - rr->address = TQHostAddress( ( answer[pp+0] << 24 ) + - ( answer[pp+1] << 16 ) + - ( answer[pp+2] << 8 ) + - ( answer[pp+3] ) ); -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN A %s (ttl %d)", label.ascii(), - rr->address.toString().ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseAaaa() -{ - if ( next != pp + 16 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN Aaaa for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Aaaa; - rr->address = TQHostAddress( answer+pp ); -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN Aaaa %s (ttl %d)", label.ascii(), - rr->address.toString().ascii(), ttl ); -#endif -} - - - -void TQDnsAnswer::parseMx() -{ - if ( next < pp + 2 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN MX for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->priority = (answer[pp] << 8) + answer[pp+1]; - pp += 2; - rr->target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad string in MX for %s", label.ascii() ); -#endif - return; - } - rr->t = TQDns::Mx; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN MX %d %s (ttl %d)", label.ascii(), - rr->priority, rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseSrv() -{ - if ( next < pp + 6 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN SRV for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->priority = (answer[pp] << 8) + answer[pp+1]; - rr->weight = (answer[pp+2] << 8) + answer[pp+3]; - rr->port = (answer[pp+4] << 8) + answer[pp+5]; - pp += 6; - rr->target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad string in SRV for %s", label.ascii() ); -#endif - return; - } - rr->t = TQDns::Srv; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN SRV %d %d %d %s (ttl %d)", label.ascii(), - rr->priority, rr->weight, rr->port, rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseCname() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad cname for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Cname; - rr->target = target; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN CNAME %s (ttl %d)", label.ascii(), - rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseNs() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad cname for for %s", label.ascii() ); -#endif - return; - } - - // parse, but ignore - -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN NS %s (ttl %d)", label.ascii(), - target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parsePtr() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad PTR for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Ptr; - rr->target = target; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN PTR %s (ttl %d)", label.ascii(), - rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseTxt() -{ - TQString text = readString(false); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad TXT for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Txt; - rr->text = text; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN TXT \"%s\" (ttl %d)", label.ascii(), - rr->text.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parse() -{ - // okay, do the work... - if ( (answer[2] & 0x78) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: answer to wrong query type (%d)", answer[1] ); -#endif - ok = false; - return; - } - - // AA - bool aa = (answer[2] & 4) != 0; - - // TC - if ( (answer[2] & 2) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: truncated answer; pressing on" ); -#endif - } - - // RD - bool rd = (answer[2] & 1) != 0; - - // we don't test RA - // we don't test the MBZ fields - - if ( (answer[3] & 0x0f) == 3 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: saw NXDomain for %s", query->l.ascii() ); -#endif - // NXDomain. cache that for one minute. - rr = new TQDnsRR( query->l ); - rr->t = query->t; - rr->deleteTime = query->started + 60; - rr->expireTime = query->started + 60; - rr->nxdomain = true; - rr->current = true; - rrs->append( rr ); - return; - } - - if ( (answer[3] & 0x0f) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: error code %d", answer[3] & 0x0f ); -#endif - ok = false; - return; - } - - int qdcount = ( answer[4] << 8 ) + answer[5]; - int ancount = ( answer[6] << 8 ) + answer[7]; - int nscount = ( answer[8] << 8 ) + answer[9]; - int adcount = (answer[10] << 8 ) +answer[11]; - - pp = 12; - - // read query - while( qdcount > 0 && pp < size ) { - // should I compare the string against query->l? - (void)readString(); - if ( !ok ) - return; - pp += 4; - qdcount--; - } - - // answers and stuff - int rrno = 0; - // if we parse the answer completely, but there are no answers, - // ignore the entire thing. - int answers = 0; - while( ( rrno < ancount || - ( ok && answers >0 && rrno < ancount + nscount + adcount ) ) && - pp < size ) { - label = readString().lower(); - if ( !ok ) - return; - int rdlength = 0; - if ( pp + 10 <= size ) - rdlength = ( answer[pp+8] << 8 ) + answer[pp+9]; - if ( pp + 10 + rdlength > size ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: ran out of stuff to parse (%d+%d>%d (%d)", - pp, rdlength, size, rrno < ancount ); -#endif - // if we're still in the AN section, we should go back and - // at least down the TTLs. probably best to invalidate - // the results. - // the rrs list is good for this - ok = ( rrno < ancount ); - return; - } - uint type, clas; - type = ( answer[pp+0] << 8 ) + answer[pp+1]; - clas = ( answer[pp+2] << 8 ) + answer[pp+3]; - ttl = ( answer[pp+4] << 24 ) + ( answer[pp+5] << 16 ) + - ( answer[pp+6] << 8 ) + answer[pp+7]; - pp = pp + 10; - if ( clas != 1 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: class %d (not internet) for %s", - clas, label.isNull() ? "." : label.ascii() ); -#endif - } else { - next = pp + rdlength; - rr = 0; - switch( type ) { - case 1: - parseA(); - break; - case 28: - parseAaaa(); - break; - case 15: - parseMx(); - break; - case 33: - parseSrv(); - break; - case 5: - parseCname(); - break; - case 12: - parsePtr(); - break; - case 16: - parseTxt(); - break; - case 2: - parseNs(); - break; - default: - // something we don't know -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: type %d for %s", type, - label.isNull() ? "." : label.ascii() ); -#endif - break; - } - if ( rr ) { - rr->deleteTime = 0; - if ( ttl > 0 ) - rr->expireTime = query->started + ttl; - else - rr->expireTime = query->started + 20; - if ( rrno < ancount ) { - answers++; - rr->deleteTime = rr->expireTime; - } - rr->current = true; - rrs->append( rr ); - } - } - if ( !ok ) - return; - pp = next; - next = size; - rrno++; - } - if ( answers == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: answer contained no answers" ); -#endif - ok = ( aa && rd ); - } - - // now go through the list and mark all the As that are referenced - // by something we care about. we want to cache such As. - rrs->first(); - TQDict<void> used( 17 ); - used.setAutoDelete( false ); - while( (rr=rrs->current()) != 0 ) { - rrs->next(); - if ( rr->target.length() && rr->deleteTime > 0 && rr->current ) - used.insert( rr->target, (void*)42 ); - if ( ( rr->t == TQDns::A || rr->t == TQDns::Aaaa ) && - used.find( rr->domain->name() ) != 0 ) - rr->deleteTime = rr->expireTime; - } - - // next, for each RR, delete any older RRs that are equal to it - rrs->first(); - while( (rr=rrs->current()) != 0 ) { - rrs->next(); - if ( rr && rr->domain && rr->domain->rrs ) { - TQPtrList<TQDnsRR> * drrs = rr->domain->rrs; - drrs->first(); - TQDnsRR * older; - while( (older=drrs->current()) != 0 ) { - if ( older != rr && - older->t == rr->t && - older->nxdomain == rr->nxdomain && - older->address == rr->address && - older->target == rr->target && - older->priority == rr->priority && - older->weight == rr->weight && - older->port == rr->port && - older->text == rr->text ) { - // well, it's equal, but it's not the same. so we kill it, - // but use its expiry time. -#if defined(TQDNS_DEBUG) - qDebug( "killing off old %d for %s, expire was %d", - older->t, older->domain->name().latin1(), - rr->expireTime ); -#endif - older->t = TQDns::None; - rr->expireTime = QMAX( older->expireTime, rr->expireTime ); - rr->deleteTime = QMAX( older->deleteTime, rr->deleteTime ); - older->deleteTime = 0; -#if defined(TQDNS_DEBUG) - qDebug( " adjusted expire is %d", rr->expireTime ); -#endif - } - drrs->next(); - } - } - } - -#if defined(TQDNS_DEBUG) - //qDebug( "DNS Manager: ()" ); -#endif -} - - -class TQDnsUgleHack: public TQDns { -public: - void ugle( bool emitAnyway=false ); -}; - - -void TQDnsAnswer::notify() -{ - if ( !rrs || !ok || !query || !query->dns ) - return; - - TQPtrDict<void> notified; - notified.setAutoDelete( false ); - - TQPtrDictIterator<void> it( *query->dns ); - TQDns * dns; - it.toFirst(); - while( (dns=(TQDns*)(it.current())) != 0 ) { - ++it; - if ( notified.find( (void*)dns ) == 0 ) { - notified.insert( (void*)dns, (void*)42 ); - if ( rrs->count() == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: found no answers!" ); -#endif - dns->d->noNames = true; - ((TQDnsUgleHack*)dns)->ugle( true ); - } else { - TQStringList n = dns->qualifiedNames(); - if ( query && n.contains(query->l) ) - ((TQDnsUgleHack*)dns)->ugle(); -#if defined(TQDNS_DEBUG) - else - qDebug( "DNS Manager: DNS thing %s not notified for %s", - dns->label().ascii(), query->l.ascii() ); -#endif - } - } - } -} - - -// -// -// TQDnsManager -// -// - - -class TQDnsManager: public TQDnsSocket { -private: -public: // just to silence the moronic g++. - TQDnsManager(); - ~TQDnsManager(); -public: - static TQDnsManager * manager(); - - TQDnsDomain * domain( const TQString & ); - - void transmitQuery( TQDnsQuery * ); - void transmitQuery( int ); - - // reimplementation of the slots - void cleanCache(); - void retransmit(); - void answer(); - -public: - TQPtrVector<TQDnsQuery> queries; - TQDict<TQDnsDomain> cache; - TQSocketDevice * ipv4Socket; -#if !defined (QT_NO_IPV6) - TQSocketDevice * ipv6Socket; -#endif -}; - - - -static TQDnsManager * globalManager = 0; - -static void cleanupDns() -{ - delete globalManager; - globalManager = 0; -} - -TQDnsManager * TQDnsManager::manager() -{ - if ( !globalManager ) { - qAddPostRoutine(cleanupDns); - new TQDnsManager(); - } - return globalManager; -} - - -void TQDnsUgleHack::ugle( bool emitAnyway) -{ - if ( emitAnyway || !isWorking() ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: status change for %s (type %d)", - label().ascii(), recordType() ); -#endif - emit resultsReady(); - } -} - - -TQDnsManager::TQDnsManager() - : TQDnsSocket( qApp, "Internal DNS manager" ), - queries( TQPtrVector<TQDnsQuery>( 0 ) ), - cache( TQDict<TQDnsDomain>( 83, false ) ), - ipv4Socket( new TQSocketDevice( TQSocketDevice::Datagram, TQSocketDevice::IPv4, 0 ) ) -#if !defined (QT_NO_IPV6) - , ipv6Socket( new TQSocketDevice( TQSocketDevice::Datagram, TQSocketDevice::IPv6, 0 ) ) -#endif -{ - cache.setAutoDelete( true ); - globalManager = this; - - TQTimer * sweepTimer = new TQTimer( this ); - sweepTimer->start( 1000 * 60 * 3 ); - connect( sweepTimer, TQT_SIGNAL(timeout()), - this, TQT_SLOT(cleanCache()) ); - - TQSocketNotifier * rn4 = new TQSocketNotifier( ipv4Socket->socket(), - TQSocketNotifier::Read, - TQT_TQOBJECT(this), "dns IPv4 socket watcher" ); - ipv4Socket->setAddressReusable( false ); - ipv4Socket->setBlocking( false ); - connect( rn4, TQT_SIGNAL(activated(int)), TQT_SLOT(answer()) ); - -#if !defined (QT_NO_IPV6) - // Don't connect the IPv6 socket notifier if the host does not - // support IPv6. - if ( ipv6Socket->socket() != -1 ) { - TQSocketNotifier * rn6 = new TQSocketNotifier( ipv6Socket->socket(), - TQSocketNotifier::Read, - TQT_TQOBJECT(this), "dns IPv6 socket watcher" ); - - ipv6support = true; - ipv6Socket->setAddressReusable( false ); - ipv6Socket->setBlocking( false ); - connect( rn6, TQT_SIGNAL(activated(int)), TQT_SLOT(answer()) ); - } -#endif - - if ( !theNs ) - TQDns::doResInit(); - - // O(n*n) stuff here. but for 3 and 6, O(n*n) with a low k should - // be perfect. the point is to eliminate any duplicates that - // might be hidden in the lists. - TQPtrList<TQHostAddress> * ns = new TQPtrList<TQHostAddress>; - - theNs->first(); - TQHostAddress * h; - while( (h=theNs->current()) != 0 ) { - ns->first(); - while( ns->current() != 0 && !(*ns->current() == *h) ) - ns->next(); - if ( !ns->current() ) { - ns->append( new TQHostAddress(*h) ); -#if defined(TQDNS_DEBUG) - qDebug( "using name server %s", h->toString().latin1() ); - } else { - qDebug( "skipping address %s", h->toString().latin1() ); -#endif - } - theNs->next(); - } - - delete theNs; - theNs = ns; - theNs->setAutoDelete( true ); - - TQStrList * domains = new TQStrList( true ); - - theDomains->first(); - const char * s; - while( (s=theDomains->current()) != 0 ) { - domains->first(); - while( domains->current() != 0 && qstrcmp( domains->current(), s ) ) - domains->next(); - if ( !domains->current() ) { - domains->append( s ); -#if defined(TQDNS_DEBUG) - qDebug( "searching domain %s", s ); - } else { - qDebug( "skipping domain %s", s ); -#endif - } - theDomains->next(); - } - - delete theDomains; - theDomains = domains; - theDomains->setAutoDelete( true ); -} - - -TQDnsManager::~TQDnsManager() -{ - if ( globalManager ) - globalManager = 0; - queries.setAutoDelete( true ); - cache.setAutoDelete( true ); - delete ipv4Socket; -#if !defined (QT_NO_IPV6) - delete ipv6Socket; -#endif -} - -static Q_UINT32 lastSweep = 0; - -void TQDnsManager::cleanCache() -{ - bool again = false; - TQDictIterator<TQDnsDomain> it( cache ); - TQDnsDomain * d; - Q_UINT32 thisSweep = now(); -#if defined(TQDNS_DEBUG) - qDebug( "TQDnsManager::cleanCache(: Called, time is %u, last was %u", - thisSweep, lastSweep ); -#endif - - while( (d=it.current()) != 0 ) { - ++it; - d->sweep( thisSweep ); // after this, d may be empty - if ( !again ) - again = !d->isEmpty(); - } - if ( !again ) - delete this; - lastSweep = thisSweep; -} - - -void TQDnsManager::retransmit() -{ - const TQT_BASE_OBJECT_NAME * o = sender(); - if ( o == 0 || globalManager == 0 || this != globalManager ) - return; - uint q = 0; - while( q < queries.size() && queries[q] != o ) - q++; - if ( q < queries.size() ) - transmitQuery( q ); -} - - -void TQDnsManager::answer() -{ - TQByteArray a( 16383 ); // large enough for anything, one suspects - - int r; -#if defined (QT_NO_IPV6) - r = ipv4Socket->readBlock(a.data(), a.size()); -#else - if (((TQSocketNotifier *)sender())->socket() == ipv4Socket->socket()) - r = ipv4Socket->readBlock(a.data(), a.size()); - else - r = ipv6Socket->readBlock(a.data(), a.size()); -#endif -#if defined(TQDNS_DEBUG) -#if !defined (QT_NO_IPV6) - qDebug("DNS Manager: answer arrived: %d bytes from %s:%d", r, - useIpv4Socket ? ipv4Socket->peerAddress().toString().ascii() - : ipv6Socket->peerAddress().toString().ascii(), - useIpv4Socket ? ipv4Socket->peerPort() : ipv6Socket->peerPort() ); -#else - qDebug("DNS Manager: answer arrived: %d bytes from %s:%d", r, - ipv4Socket->peerAddress().toString().ascii(), ipv4Socket->peerPort());; -#endif -#endif - if ( r < 12 ) - return; - // maybe we should check that the answer comes from port 53 on one - // of our name servers... - a.resize( r ); - - Q_UINT16 aid = (((Q_UINT8)a[0]) << 8) + ((Q_UINT8)a[1]); - uint i = 0; - while( i < queries.size() && - !( queries[i] && queries[i]->id == aid ) ) - i++; - if ( i == queries.size() ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: bad id (0x%04x) %d", aid, i ); -#endif - return; - } - - // at this point queries[i] is whatever we asked for. - - if ( ( (Q_UINT8)(a[2]) & 0x80 ) == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: received a query" ); -#endif - return; - } - - TQDnsQuery * q = queries[i]; - TQDnsAnswer answer( a, q ); - answer.parse(); - if ( answer.ok ) { - queries.take( i ); - answer.notify(); - delete q; - } -} - - -void TQDnsManager::transmitQuery( TQDnsQuery * query_ ) -{ - if ( !query_ ) - return; - - uint i = 0; - while( i < queries.size() && queries[i] != 0 ) - i++; - if ( i == queries.size() ) - queries.resize( i+1 ); - queries.insert( i, query_ ); - transmitQuery( i ); -} - - -void TQDnsManager::transmitQuery( int i ) -{ - if ( i < 0 || i >= (int)queries.size() ) - return; - TQDnsQuery * q = queries[i]; - - if ( q && q->step > 8 ) { - // okay, we've run out of retransmissions. we fake an NXDomain - // with a very short life time... - TQDnsAnswer answer( q ); - answer.notify(); - // and then get rid of the query - queries.take( i ); -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: giving up on query 0x%04x", q->id ); -#endif - delete q; - TQTimer::singleShot( 0, TQDnsManager::manager(), TQT_SLOT(cleanCache()) ); - // and don't process anything more - return; - } - - if ((q && !q->dns) || q->dns->isEmpty()) - // no one currently wants the answer, so there's no point in - // retransmitting the query. we keep it, though. an answer may - // arrive for an earlier query transmission, and if it does we - // may benefit from caching the result. - return; - - TQByteArray p( 12 + q->l.length() + 2 + 4 ); - if ( p.size() > 500 ) - return; // way over the limit, so don't even try - - // header - // id - p[0] = (q->id & 0xff00) >> 8; - p[1] = q->id & 0x00ff; - p[2] = 1; // recursion desired, rest is 0 - p[3] = 0; // all is 0 - // one query - p[4] = 0; - p[5] = 1; - // no answers, name servers or additional data - p[6] = p[7] = p[8] = p[9] = p[10] = p[11] = 0; - - // the name is composed of several components. each needs to be - // written by itself... so we write... - // oh, and we assume that there's no funky characters in there. - int pp = 12; - uint lp = 0; - while( lp < (uint) q->l.length() ) { - int le = q->l.find( QLatin1Char('.'), lp ); - if ( le < 0 ) - le = q->l.length(); - TQString component = q->l.mid( lp, le-lp ); - p[pp++] = component.length(); - int cp; - for( cp=0; cp < (int)component.length(); cp++ ) - p[pp++] = component[cp].latin1(); - lp = le + 1; - } - // final null - p[pp++] = 0; - // query type - p[pp++] = 0; - switch( q->t ) { - case TQDns::A: - p[pp++] = 1; - break; - case TQDns::Aaaa: - p[pp++] = 28; - break; - case TQDns::Mx: - p[pp++] = 15; - break; - case TQDns::Srv: - p[pp++] = 33; - break; - case TQDns::Cname: - p[pp++] = 5; - break; - case TQDns::Ptr: - p[pp++] = 12; - break; - case TQDns::Txt: - p[pp++] = 16; - break; - default: - p[pp++] = (char)255; // any - break; - } - // query class (always internet) - p[pp++] = 0; - p[pp++] = 1; - - // if we have no name servers, we should regenerate ns in case - // name servers have recently been defined (like on windows, - // plugging/unplugging the network cable will change the name - // server entries) - if ( !theNs || theNs->isEmpty() ) - TQDns::doResInit(); - - if ( !theNs || theNs->isEmpty() ) { - // we don't find any name servers. We fake an NXDomain - // with a very short life time... - TQDnsAnswer answer( q ); - answer.notify(); - // and then get rid of the query - queries.take( i ); -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: no DNS server found on query 0x%04x", q->id ); -#endif - delete q; - TQTimer::singleShot( 1000*10, TQDnsManager::manager(), TQT_SLOT(cleanCache()) ); - // and don't process anything more - return; - } - - TQHostAddress receiver = *theNs->at( q->step % theNs->count() ); - if (receiver.isIPv4Address()) - ipv4Socket->writeBlock( p.data(), pp, receiver, 53 ); -#if !defined (QT_NO_IPV6) - else - ipv6Socket->writeBlock( p.data(), pp, receiver, 53 ); -#endif -#if defined(TQDNS_DEBUG) - qDebug( "issuing query 0x%04x (%d) about %s type %d to %s", - q->id, q->step, q->l.ascii(), q->t, - ns->at( q->step % ns->count() )->toString().ascii() ); -#endif - if ( theNs->count() > 1 && q->step == 0 && queries.count() == 1 ) { - // if it's the first time, and we don't have any other - // outstanding queries, send nonrecursive queries to the other - // name servers too. - p[2] = 0; - TQHostAddress * server; - while( (server=theNs->next()) != 0 ) { - if (server->isIPv4Address()) - ipv4Socket->writeBlock( p.data(), pp, *server, 53 ); -#if !defined (QT_NO_IPV6) - else - ipv6Socket->writeBlock( p.data(), pp, *server, 53 ); -#endif -#if defined(TQDNS_DEBUG) - qDebug( "copying query to %s", server->toString().ascii() ); -#endif - } - } - q->step++; - // some testing indicates that normal dns queries take up to 0.6 - // seconds. the graph becomes steep around that point, and the - // number of errors rises... so it seems good to retry at that - // point. - q->start( q->step < theNs->count() ? 800 : 1500, true ); -} - - -TQDnsDomain * TQDnsManager::domain( const TQString & label ) -{ - TQDnsDomain * d = cache.find( label ); - if ( !d ) { - d = new TQDnsDomain( label ); - cache.insert( label, d ); - } - return d; -} - - -// -// -// the TQDnsDomain class looks after and coordinates queries for TQDnsRRs for -// each domain, and the cached TQDnsRRs. (A domain, in DNS terminology, is -// a node in the DNS. "no", "trolltech.com" and "lupinella.troll.no" are -// all domains.) -// -// - - -TQDnsDomain::TQDnsDomain( const TQString & label ) -{ - l = label; - rrs = 0; -} - - -TQDnsDomain::~TQDnsDomain() -{ - delete rrs; - rrs = 0; -} - - -void TQDnsDomain::add( const TQString & label, TQDnsRR * rr ) -{ - TQDnsDomain * d = TQDnsManager::manager()->domain( label ); - if ( !d->rrs ) { - d->rrs = new TQPtrList<TQDnsRR>; - d->rrs->setAutoDelete( true ); - } - d->rrs->append( rr ); - rr->domain = d; -} - - -TQPtrList<TQDnsRR> * TQDnsDomain::cached( const TQDns * r ) -{ - TQPtrList<TQDnsRR> * l = new TQPtrList<TQDnsRR>; - - // test at first if you have to start a query at all - if ( r->recordType() == TQDns::A ) { - if ( r->label().lower() == TQLatin1String("localhost") ) { - // undocumented hack. ipv4-specific. also, may be a memory - // leak? not sure. would be better to do this in doResInit(), - // anyway. - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - rrTmp->t = TQDns::A; - rrTmp->address = TQHostAddress( 0x7f000001 ); - rrTmp->current = true; - l->append( rrTmp ); - return l; - } - TQHostAddress tmp; - if ( tmp.setAddress( r->label() ) ) { - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - if ( tmp.isIPv4Address() ) { - rrTmp->t = TQDns::A; - rrTmp->address = tmp; - rrTmp->current = true; - l->append( rrTmp ); - } else { - rrTmp->nxdomain = true; - } - return l; - } - } - if ( r->recordType() == TQDns::Aaaa ) { - TQHostAddress tmp; - if ( tmp.setAddress(r->label()) ) { - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - if ( tmp.isIPv6Address() ) { - rrTmp->t = TQDns::Aaaa; - rrTmp->address = tmp; - rrTmp->current = true; - l->append( rrTmp ); - } else { - rrTmp->nxdomain = true; - } - return l; - } - } - - // if you reach this point, you have to do the query - TQDnsManager * m = TQDnsManager::manager(); - TQStringList n = r->qualifiedNames(); - bool nxdomain; - int cnamecount = 0; - int it = 0; - while( it < n.count() ) { - TQString s = *(n.at(it++)); - nxdomain = false; -#if defined(TQDNS_DEBUG) - qDebug( "looking at cache for %s (%s %d)", - s.ascii(), r->label().ascii(), r->recordType() ); -#endif - TQDnsDomain * d = m->domain( s ); -#if defined(TQDNS_DEBUG) - qDebug( " - found %d RRs", d && d->rrs ? d->rrs->count() : 0 ); -#endif - if ( d->rrs ) - d->rrs->first(); - TQDnsRR * rr; - bool answer = false; - while( d->rrs && (rr=d->rrs->current()) != 0 ) { - if ( rr->t == TQDns::Cname && r->recordType() != TQDns::Cname && - !rr->nxdomain && cnamecount < 16 ) { - // cname. if the code is ugly, that may just - // possibly be because the concept is. -#if defined(TQDNS_DEBUG) - qDebug( "found cname from %s to %s", - r->label().ascii(), rr->target.ascii() ); -#endif - s = rr->target; - d = m->domain( s ); - if ( d->rrs ) - d->rrs->first(); - it = n.count(); - // we've elegantly moved over to whatever the cname - // pointed to. well, not elegantly. let's remember - // that we've done something, anyway, so we can't be - // fooled into an infinte loop as well. - cnamecount++; - } else { - if ( rr->t == r->recordType() ) { - if ( rr->nxdomain ) - nxdomain = true; - else - answer = true; - l->append( rr ); - if ( rr->deleteTime <= lastSweep ) { - // we're returning something that'll be - // deleted soon. we assume that if the client - // wanted it twice, it'll want it again, so we - // ask the name server again right now. - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++theId; - query->t = rr->t; - query->l = rr->domain->name(); - // note that here, we don't bother about - // notification. but we do bother about - // timeouts: we make sure to use high timeouts - // and few tramsissions. - query->step = theNs->count(); - TQT_BASE_OBJECT_NAME::connect( query, TQT_SIGNAL(timeout()), - TQDnsManager::manager(), - TQT_SLOT(retransmit()) ); - TQDnsManager::manager()->transmitQuery( query ); - } - } - d->rrs->next(); - } - } - // if we found a positive result, return quickly - if ( answer && l->count() ) { -#if defined(TQDNS_DEBUG) - qDebug( "found %d records for %s", - l->count(), r->label().ascii() ); - l->first(); - while( l->current() ) { - qDebug( " type %d target %s address %s", - l->current()->t, - l->current()->target.latin1(), - l->current()->address.toString().latin1() ); - l->next(); - } -#endif - l->first(); - return l; - } - -#if defined(TQDNS_DEBUG) - if ( nxdomain ) - qDebug( "found NXDomain %s", s.ascii() ); -#endif - - if ( !nxdomain ) { - // if we didn't, and not a negative result either, perhaps - // we need to transmit a query. - uint q = 0; - while ( q < m->queries.size() && - ( m->queries[q] == 0 || - m->queries[q]->t != r->recordType() || - m->queries[q]->l != s ) ) - q++; - // we haven't done it before, so maybe we should. but - // wait - if it's an unqualified name, only ask when all - // the other alternatives are exhausted. - if ( q == m->queries.size() && ( s.find( QLatin1Char('.') ) >= 0 || - int(l->count()) >= n.count()-1 ) ) { - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++theId; - query->t = r->recordType(); - query->l = s; - query->dns->replace( (void*)r, (void*)r ); - TQT_BASE_OBJECT_NAME::connect( query, TQT_SIGNAL(timeout()), - TQDnsManager::manager(), TQT_SLOT(retransmit()) ); - TQDnsManager::manager()->transmitQuery( query ); - } else if ( q < m->queries.size() ) { - // if we've found an earlier query for the same - // domain/type, subscribe to its answer - m->queries[q]->dns->replace( (void*)r, (void*)r ); - } - } - } - l->first(); - return l; -} - - -void TQDnsDomain::sweep( Q_UINT32 thisSweep ) -{ - if ( !rrs ) - return; - - TQDnsRR * rr; - rrs->first(); - while( (rr=rrs->current()) != 0 ) { - if ( !rr->deleteTime ) - rr->deleteTime = thisSweep; // will hit next time around - -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::sweep: %s type %d expires %u %u - %s / %s", - rr->domain->name().latin1(), rr->t, - rr->expireTime, rr->deleteTime, - rr->target.latin1(), rr->address.toString().latin1()); -#endif - if ( rr->current == false || - rr->t == TQDns::None || - rr->deleteTime <= thisSweep || - rr->expireTime <= thisSweep ) - rrs->remove(); - else - rrs->next(); - } - - if ( rrs->isEmpty() ) { - delete rrs; - rrs = 0; - } -} - - - - -// the itsy-bitsy little socket class I don't really need except for -// so I can subclass and reimplement the slots. - - -TQDnsSocket::TQDnsSocket( TQT_BASE_OBJECT_NAME * parent, const char * name ) - : TQObject( parent, name ) -{ - // nothing -} - - -TQDnsSocket::~TQDnsSocket() -{ - // nothing -} - - -void TQDnsSocket::cleanCache() -{ - // nothing -} - - -void TQDnsSocket::retransmit() -{ - // nothing -} - - -void TQDnsSocket::answer() -{ - // nothing -} - - -/*! - \class TQDns - \brief The TQDns class provides asynchronous DNS lookups. - - \compat - - Both Windows and Unix provide synchronous DNS lookups; Windows - provides some asynchronous support too. At the time of writing - neither operating system provides asynchronous support for - anything other than hostname-to-address mapping. - - TQDns rectifies this shortcoming, by providing asynchronous caching - lookups for the record types that we expect modern GUI - applications to need in the near future. - - The class is \e not straightforward to use (although it is much - simpler than the native APIs); TQSocket provides much easier to use - TCP connection facilities. The aim of TQDns is to provide a correct - and small API to the DNS and nothing more. (We use "correctness" - to mean that the DNS information is correctly cached, and - correctly timed out.) - - The API comprises a constructor, functions to set the DNS node - (the domain in DNS terminology) and record type (setLabel() and - setRecordType()), the corresponding get functions, an isWorking() - function to determine whether TQDns is working or reading, a - resultsReady() signal and query functions for the result. - - There is one query function for each RecordType, namely - addresses(), mailServers(), servers(), hostNames() and texts(). - There are also two generic query functions: canonicalName() - returns the name you'll presumably end up using (the exact meaning - of this depends on the record type) and qualifiedNames() returns a - list of the fully qualified names label() maps to. - - \sa TQSocket -*/ - -/*! - Constructs a DNS query object with invalid settings for both the - label and the search type. -*/ - -TQDns::TQDns() -{ - d = new TQDnsPrivate; - t = None; -} - - - - -/*! - Constructs a DNS query object that will return record type \a rr - information about \a label. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \a rr defaults to \c A, IPv4 addresses. -*/ - -TQDns::TQDns( const TQString & label, RecordType rr ) -{ - d = new TQDnsPrivate; - t = rr; - setLabel( label ); - setStartQueryTimer(); // start query the next time we enter event loop -} - - - -/*! - Constructs a DNS query object that will return record type \a rr - information about host address \a address. The label is set to the - IN-ADDR.ARPA domain name. This is useful in combination with the - \c Ptr record type (e.g. if you want to look up a hostname for a - given address). - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \a rr defaults to \c Ptr, that maps addresses to hostnames. -*/ - -TQDns::TQDns( const TQHostAddress & address, RecordType rr ) -{ - d = new TQDnsPrivate; - t = rr; - setLabel( address ); - setStartQueryTimer(); // start query the next time we enter event loop -} - - - - -/*! - Destroys the DNS query object and frees its allocated resources. -*/ - -TQDns::~TQDns() -{ - if ( globalManager ) { - uint q = 0; - TQDnsManager * m = globalManager; - while( q < m->queries.size() ) { - TQDnsQuery * query=m->queries[q]; - if ( query && query->dns ) - (void)query->dns->take( (void*) this ); - q++; - } - - } - - delete d; - d = 0; -} - - - - -/*! - Sets this DNS query object to query for information about \a - label. - - This does not change the recordType(), but its isWorking() status - will probably change as a result. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. -*/ - -void TQDns::setLabel( const TQString & label ) -{ - l = label; - d->noNames = false; - - // construct a list of qualified names - n.clear(); - if ( l.length() > 1 && l[(int)l.length()-1] == QLatin1Char('.') ) { - n.append( TQT_TQSTRING(l.left( l.length()-1 )).lower() ); - } else { - int i = l.length(); - int dots = 0; - const int maxDots = 2; - while( i && dots < maxDots ) { - if ( l[--i] == QLatin1Char('.') ) - dots++; - } - if ( dots < maxDots ) { - (void)TQDnsManager::manager(); // create a TQDnsManager, if it is not already there - TQStrListIterator it( *theDomains ); - const char * dom; - while( (dom=it.current()) != 0 ) { - ++it; - n.append( l.lower() + QLatin1Char('.') + TQLatin1String(dom) ); - } - } - n.append( l.lower() ); - } - -#if defined(Q_DNS_SYNCHRONOUS) - if ( d->noEventLoop ) { - doSynchronousLookup(); - } else { - setStartQueryTimer(); // start query the next time we enter event loop - } -#else - setStartQueryTimer(); // start query the next time we enter event loop -#endif -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::setLabel: %d address(es) for %s", n.count(), l.ascii() ); - int i = 0; - for( i = 0; i < (int)n.count(); i++ ) - qDebug( "TQDns::setLabel: %d: %s", i, n[i].ascii() ); -#endif -} - - -/*! - \overload - - Sets this DNS query object to query for information about the host - address \a address. The label is set to the IN-ADDR.ARPA domain - name. This is useful in combination with the \c Ptr record type - (e.g. if you want to look up a hostname for a given address). -*/ - -void TQDns::setLabel( const TQHostAddress & address ) -{ - setLabel( toInAddrArpaDomain( address ) ); -} - - -/*! - \fn TQStringList TQDns::qualifiedNames() const - - Returns a list of the fully qualified names label() maps to. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 0 - -*/ - - -/*! - \fn TQString TQDns::label() const - - Returns the domain name for which this object returns information. - - \sa setLabel() -*/ - -/*! - \enum TQDns::RecordType - - This enum type defines the record types TQDns can handle. The DNS - provides many more; these are the ones we've judged to be in - current use, useful for GUI programs and important enough to - support right away: - - \value None No information. This exists only so that TQDns can - have a default. - - \value A IPv4 addresses. By far the most common type. - - \value Aaaa IPv6 addresses. So far mostly unused. - - \value Mx Mail eXchanger names. Used for mail delivery. - - \value Srv SeRVer names. Generic record type for finding - servers. So far mostly unused. - - \value Cname Canonical names. Maps from nicknames to the true - name (the canonical name) for a host. - - \value Ptr name PoinTeRs. Maps from IPv4 or IPv6 addresses to hostnames. - - \value Txt arbitrary TeXT for domains. - - We expect that some support for the - \l{http://www.rfc-editor.org/rfc/rfc2535.txt}{RFC 2535} - extensions will be added in future versions. -*/ - -/*! - Sets this object to query for record type \a rr records. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \sa RecordType -*/ - -void TQDns::setRecordType( RecordType rr ) -{ - t = rr; - d->noNames = false; - setStartQueryTimer(); // start query the next time we enter event loop -} - -/*! - \internal - - Private slot for starting the query. -*/ -void TQDns::startQuery() -{ - // isWorking() starts the query (if necessary) - if ( !isWorking() ) - emit resultsReady(); -} - -/*! - The three functions TQDns::TQDns(TQString, RecordType), - TQDns::setLabel() and TQDns::setRecordType() may start a DNS lookup. - This function handles setting up the single shot timer. -*/ -void TQDns::setStartQueryTimer() -{ -#if defined(Q_DNS_SYNCHRONOUS) - if ( !d->queryTimer && !d->noEventLoop ) -#else - if ( !d->queryTimer ) -#endif - { - // start the query the next time we enter event loop - d->queryTimer = new TQTimer( this ); - connect( d->queryTimer, TQT_SIGNAL(timeout()), - this, TQT_SLOT(startQuery()) ); - d->queryTimer->start( 0, true ); - } -} - -/* - Transforms the host address \a address to the IN-ADDR.ARPA domain - name. Returns something indeterminate if you're sloppy or - naughty. This function has an IPv4-specific name, but works for - IPv6 too. -*/ -TQString TQDns::toInAddrArpaDomain( const TQHostAddress &address ) -{ - TQString s; - if ( address.isNull() ) { - // if the address isn't valid, neither of the other two make - // cases make sense. better to just return. - } else if ( address.isIp4Addr() ) { - Q_UINT32 i = address.ip4Addr(); - s.sprintf( "%d.%d.%d.%d.IN-ADDR.ARPA", - i & 0xff, (i >> 8) & 0xff, (i>>16) & 0xff, (i>>24) & 0xff ); - } else { - // RFC 3152. (1886 is deprecated, and clients no longer need to - // support it, in practice). - Q_IPV6ADDR i = address.toIPv6Address(); - s = TQLatin1String("ip6.arpa"); - uint b = 0; - while( b < 16 ) { - s = TQString::number( i.c[b]%16, 16 ) + QLatin1Char('.') + - TQString::number( i.c[b]/16, 16 ) + QLatin1Char('.') + s; - b++; - } - } - return s; -} - - -/*! - \fn TQDns::RecordType TQDns::recordType() const - - Returns the record type of this DNS query object. - - \sa setRecordType() RecordType -*/ - -/*! - \fn void TQDns::resultsReady() - - This signal is emitted when results are available for one of the - qualifiedNames(). -*/ - -/*! - Returns true if TQDns is doing a lookup for this object (i.e. if it - does not already have the necessary information); otherwise - returns false. - - TQDns emits the resultsReady() signal when the status changes to false. -*/ - -bool TQDns::isWorking() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::isWorking (%s, %d)", l.ascii(), t ); -#endif - if ( t == None ) - return false; - -#if defined(Q_DNS_SYNCHRONOUS) - if ( d->noEventLoop ) - return true; -#endif - - TQPtrList<TQDnsRR> * ll = TQDnsDomain::cached( this ); - Q_LONG queries = n.count(); - while( ll->current() != 0 ) { - if ( ll->current()->nxdomain ) { - queries--; - } else { - delete ll; - return false; - } - ll->next(); - } - delete ll; - - if ( queries <= 0 ) - return false; - if ( d->noNames ) - return false; - return true; -} - - -/*! - Returns a list of the addresses for this name if this TQDns object - has a recordType() of TQDns::A or TQDns::Aaaa and the answer - is available; otherwise returns an empty list. - - As a special case, if label() is a valid numeric IP address, this - function returns that address. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 1 - -*/ - -TQValueList<TQHostAddress> TQDns::addresses() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::addresses (%s)", l.ascii() ); -#endif - TQValueList<TQHostAddress> result; - if ( t != A && t != Aaaa ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) - result.append( rr->address ); - cached->next(); - } - delete cached; - return result; -} - - -/*! - \class TQDns::MailServer - \brief The TQDns::MailServer class is described in TQDns::mailServers(). - - \internal -*/ - -/*! - Returns a list of mail servers if the record type is \c Mx. The - class TQDns::MailServer contains the following public variables: - \list - \i TQString TQDns::MailServer::name - \i Q_UINT16 TQDns::MailServer::priority - \endlist - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 2 - -*/ -TQValueList<TQDns::MailServer> TQDns::mailServers() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::mailServers (%s)", l.ascii() ); -#endif - TQValueList<TQDns::MailServer> result; - if ( t != Mx ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - MailServer ms( rr->target, rr->priority ); - result.append( ms ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - \class TQDns::Server - \brief The TQDns::Server class is described in TQDns::servers(). - - \internal -*/ - -/*! - Returns a list of servers if the record type is \c Srv. The class - TQDns::Server contains the following public variables: - \list - \i TQString TQDns::Server::name - \i Q_UINT16 TQDns::Server::priority - \i Q_UINT16 TQDns::Server::weight - \i Q_UINT16 TQDns::Server::port - \endlist - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 3 -*/ -TQValueList<TQDns::Server> TQDns::servers() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::servers (%s)", l.ascii() ); -#endif - TQValueList<TQDns::Server> result; - if ( t != Srv ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - Server s( rr->target, rr->priority, rr->weight, rr->port ); - result.append( s ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns a list of host names if the record type is \c Ptr. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 4 - -*/ -TQStringList TQDns::hostNames() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::hostNames (%s)", l.ascii() ); -#endif - TQStringList result; - if ( t != Ptr ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - TQString str( rr->target ); - result.append( str ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns a list of texts if the record type is \c Txt. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \snippet doc/src/snippets/code/src_qt3support_network_q3dns.cpp 5 -*/ -TQStringList TQDns::texts() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::texts (%s)", l.ascii() ); -#endif - TQStringList result; - if ( t != Txt ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - TQString str( rr->text ); - result.append( str ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns the canonical name for this DNS node. (This works - regardless of what recordType() is set to.) - - If the canonical name isn't known, this function returns a null - string. - - The canonical name of a DNS node is its full name, or the full - name of the target of its CNAME. For example, if l.trolltech.com - is a CNAME to lillian.troll.no, and the search path for TQDns is - "trolltech.com", then the canonical name for all of "lillian", - "l", "lillian.troll.no." and "l.trolltech.com" is - "lillian.troll.no.". -*/ - -TQString TQDns::canonicalName() const -{ - // the cname should work regardless of the recordType(), so set the record - // type temporarily to cname when you look at the cache - TQDns *that = (TQDns*) this; // mutable function - RecordType oldType = t; - that->t = Cname; - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( that ); - that->t = oldType; - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain && rr->domain ) { - delete cached; - return rr->target; - } - cached->next(); - } - delete cached; - return TQString(); -} - -#if defined(Q_DNS_SYNCHRONOUS) -/*! \reimp -*/ -void TQDns::connectNotify( const char *signal ) -{ - if ( d->noEventLoop && qstrcmp(signal,TQT_SIGNAL(resultsReady()) )==0 ) { - doSynchronousLookup(); - } -} -#endif - -#if defined(Q_OS_WIN32) || defined(Q_OS_CYGWIN) - -#if defined(Q_DNS_SYNCHRONOUS) -void TQDns::doSynchronousLookup() -{ - // ### not implemented yet -} -#endif - -// the following typedefs are needed for GetNetworkParams() API call -#ifndef IP_TYPES_INCLUDED -#define MAX_HOSTNAME_LEN 128 -#define MAX_DOMAIN_NAME_LEN 128 -#define MAX_SCOPE_ID_LEN 256 -typedef struct { - char String[4 * 4]; -} IP_ADDRESS_STRING, *PIP_ADDRESS_STRING, IP_MASK_STRING, *PIP_MASK_STRING; -typedef struct _IP_ADDR_STRING { - struct _IP_ADDR_STRING* Next; - IP_ADDRESS_STRING IpAddress; - IP_MASK_STRING IpMask; - DWORD Context; -} IP_ADDR_STRING, *PIP_ADDR_STRING; -typedef struct { - char HostName[MAX_HOSTNAME_LEN + 4] ; - char DomainName[MAX_DOMAIN_NAME_LEN + 4]; - PIP_ADDR_STRING CurrentDnsServer; - IP_ADDR_STRING DnsServerList; - UINT NodeType; - char ScopeId[MAX_SCOPE_ID_LEN + 4]; - UINT EnableRouting; - UINT EnableProxy; - UINT EnableDns; -} FIXED_INFO, *PFIXED_INFO; -#endif -typedef DWORD (WINAPI *GNP)( PFIXED_INFO, PULONG ); - -// ### FIXME: this code is duplicated in qfiledialog.cpp -static TQString getWindowsRegString(HKEY key, const TQString &subKey) -{ - TQString s; - - wchar_t buf[1024]; - DWORD bsz = sizeof(buf) / sizeof(wchar_t); - int r = RegQueryValueEx(key, (wchar_t*)subKey.utf16(), 0, 0, (LPBYTE)buf, &bsz); - if (r == ERROR_SUCCESS) { - s = TQString::fromWCharArray(buf); - } else if (r == ERROR_MORE_DATA) { - char *ptr = new char[bsz+1]; - r = RegQueryValueEx(key, (wchar_t*)subKey.utf16(), 0, 0, (LPBYTE)ptr, &bsz); - if (r == ERROR_SUCCESS) - s = TQLatin1String(ptr); - delete [] ptr; - } - - return s; -} - -static bool getDnsParamsFromRegistry( const TQString &path, - TQString *domainName, TQString *nameServer, TQString *searchList ) -{ - HKEY k; - int r = RegOpenKeyEx( HKEY_LOCAL_MACHINE, (wchar_t*)path.utf16(), 0, KEY_READ, &k ); - - if ( r == ERROR_SUCCESS ) { - *domainName = getWindowsRegString( k, TQLatin1String("DhcpDomain") ); - if ( domainName->isEmpty() ) - *domainName = getWindowsRegString( k, TQLatin1String("Domain") ); - - *nameServer = getWindowsRegString( k, TQLatin1String("DhcpNameServer") ); - if ( nameServer->isEmpty() ) - *nameServer = getWindowsRegString( k, TQLatin1String("NameServer") ); - - *searchList = getWindowsRegString( k, TQLatin1String("SearchList") ); - } - RegCloseKey( k ); - return r == ERROR_SUCCESS; -} - -void TQDns::doResInit() -{ - char separator = 0; - - if ( theNs ) - delete theNs; - theNs = new TQPtrList<TQHostAddress>; - theNs->setAutoDelete( true ); - theDomains = new TQStrList( true ); - theDomains->setAutoDelete( true ); - - TQString domainName, nameServer, searchList; - - bool gotNetworkParams = false; - // try the API call GetNetworkParams() first and use registry lookup only - // as a fallback - HINSTANCE hinstLib = LoadLibrary( L"iphlpapi" ); - if ( hinstLib != 0 ) { -#ifdef Q_OS_WINCE - GNP getNetworkParams = (GNP) GetProcAddress( hinstLib, L"GetNetworkParams" ); -#else - GNP getNetworkParams = (GNP) GetProcAddress( hinstLib, "GetNetworkParams" ); -#endif - if ( getNetworkParams != 0 ) { - ULONG l = 0; - DWORD res; - res = getNetworkParams( 0, &l ); - if ( res == ERROR_BUFFER_OVERFLOW ) { - FIXED_INFO *finfo = (FIXED_INFO*)new char[l]; - res = getNetworkParams( finfo, &l ); - if ( res == ERROR_SUCCESS ) { - domainName = TQLatin1String(finfo->DomainName); - nameServer = TQLatin1String(""); - IP_ADDR_STRING *dnsServer = &finfo->DnsServerList; - while ( dnsServer != 0 ) { - nameServer += TQLatin1String(dnsServer->IpAddress.String); - dnsServer = dnsServer->Next; - if ( dnsServer != 0 ) - nameServer += QLatin1Char(' '); - } - searchList = TQLatin1String(""); - separator = ' '; - gotNetworkParams = true; - } - delete[] finfo; - } - } - FreeLibrary( hinstLib ); - } - if ( !gotNetworkParams ) { - if ( getDnsParamsFromRegistry( - TQLatin1String("System\\CurrentControlSet\\Services\\Tcpip\\Parameters"), - &domainName, &nameServer, &searchList )) { - separator = ' '; - } else { - // Could not access the TCP/IP parameters - domainName = TQLatin1String(""); - nameServer = TQLatin1String("127.0.0.1"); - searchList = TQLatin1String(""); - separator = ' '; - } - } - - nameServer = nameServer.simplifyWhiteSpace(); - int first, last; - if ( !nameServer.isEmpty() ) { - first = 0; - do { - last = nameServer.find( QLatin1Char(separator), first ); - if ( last < 0 ) - last = nameServer.length(); - TQDns tmp( nameServer.mid( first, last-first ), TQDns::A ); - TQValueList<TQHostAddress> address = tmp.addresses(); - Q_LONG i = address.count(); - while( i ) - theNs->append( new TQHostAddress(address[--i]) ); - first = last+1; - } while( first < (int)nameServer.length() ); - } - - searchList += QLatin1Char(' ') + domainName; - searchList = searchList.simplifyWhiteSpace().lower(); - first = 0; - do { - last = searchList.find( QLatin1Char(separator), first ); - if ( last < 0 ) - last = searchList.length(); - theDomains->append( qstrdup( searchList.mid( first, last-first ).latin1() ) ); - first = last+1; - } while( first < (int)searchList.length() ); -} - -#elif defined(Q_OS_UNIX) - -#if defined(Q_DNS_SYNCHRONOUS) -void TQDns::doSynchronousLookup() -{ - if ( t!=None && !l.isEmpty() ) { - TQValueListIterator<TQString> it = n.begin(); - TQValueListIterator<TQString> end = n.end(); - int type; - switch( t ) { - case TQDns::A: - type = 1; - break; - case TQDns::Aaaa: - type = 28; - break; - case TQDns::Mx: - type = 15; - break; - case TQDns::Srv: - type = 33; - break; - case TQDns::Cname: - type = 5; - break; - case TQDns::Ptr: - type = 12; - break; - case TQDns::Txt: - type = 16; - break; - default: - type = (char)255; // any - break; - } - while( it != end ) { - TQString s = *it; - it++; - TQByteArray ba( 512 ); - int len = res_search( s.latin1(), 1, type, (uchar*)ba.data(), ba.size() ); - if ( len > 0 ) { - ba.resize( len ); - - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++theId; - query->t = t; - query->l = s; - TQDnsAnswer a( ba, query ); - a.parse(); - } else if ( len == -1 ) { - // res_search error - } - } - emit resultsReady(); - } -} -#endif - -#if defined(__GLIBC__) && ((__GLIBC__ > 2) || ((__GLIBC__ == 2) && (__GLIBC_MINOR__ >= 3))) -#define Q_MODERN_RES_API -#endif - -void TQDns::doResInit() -{ - if ( theNs ) - return; - theNs = new TQPtrList<TQHostAddress>; - theNs->setAutoDelete( true ); - theDomains = new TQStrList( true ); - theDomains->setAutoDelete( true ); - - // read resolv.conf manually. - QFile resolvConf(TQLatin1String("/etc/resolv.conf")); - if (resolvConf.open(QIODevice::ReadOnly)) { - QTextStream stream( &resolvConf ); - TQString line; - - while ( !stream.atEnd() ) { - line = stream.readLine(); - TQStringList list = TQStringList::split( TQLatin1String(" "), line ); - if( line.startsWith( QLatin1Char('#') ) || list.size() < 2 ) - continue; - const TQString type = list[0].lower(); - - if ( type == TQLatin1String("nameserver") ) { - TQHostAddress *address = new TQHostAddress(); - if ( address->setAddress( TQString(list[1]) ) ) { - // only add ipv6 addresses from resolv.conf if - // this host supports ipv6. - if ( address->isIPv4Address() || ipv6support ) - theNs->append( address ); - else - delete address; - } else { - delete address; - } - } else if ( type == TQLatin1String("search") ) { - TQStringList srch = TQStringList::split( TQLatin1String(" "), list[1] ); - for ( TQStringList::Iterator i = srch.begin(); i != srch.end(); ++i ) - theDomains->append( (*i).lower().local8Bit() ); - - } else if ( type == TQLatin1String("domain") ) { - theDomains->append( list[1].lower().local8Bit() ); - } - } - } - - if (theNs->isEmpty()) { -#if defined(Q_MODERN_RES_API) - struct __res_state res; - res_ninit( &res ); - int i; - // find the name servers to use - for( i=0; i < MAXNS && i < res.nscount; i++ ) - theNs->append( new TQHostAddress( ntohl( res.nsaddr_list[i].sin_addr.s_addr ) ) ); -# if defined(MAXDFLSRCH) - for( i=0; i < MAXDFLSRCH; i++ ) { - if ( res.dnsrch[i] && *(res.dnsrch[i]) ) - theDomains->append( TQT_TQSTRING(TQString::fromLatin1( res.dnsrch[i] )).lower().local8Bit() ); - else - break; - } -# endif - if ( *res.defdname ) - theDomains->append( TQT_TQSTRING(TQString::fromLatin1( res.defdname )).lower().local8Bit() ); -#else - res_init(); - int i; - // find the name servers to use - for( i=0; i < MAXNS && i < _res.nscount; i++ ) - theNs->append( new TQHostAddress( ntohl( _res.nsaddr_list[i].sin_addr.s_addr ) ) ); -# if defined(MAXDFLSRCH) - for( i=0; i < MAXDFLSRCH; i++ ) { - if ( _res.dnsrch[i] && *(_res.dnsrch[i]) ) - theDomains->append( TQString::fromLatin1( _res.dnsrch[i] ).lower().local8Bit() ); - else - break; - } -# endif - if ( *_res.defdname ) - theDomains->append( TQString::fromLatin1( _res.defdname ).lower().local8Bit() ); -#endif - - // the code above adds "0.0.0.0" as a name server at the slightest - // hint of trouble. so remove those again. - theNs->first(); - while( theNs->current() ) { - if ( theNs->current()->isNull() ) - delete theNs->take(); - else - theNs->next(); - } - } - - QFile hosts( TQString::fromLatin1( "/etc/hosts" ) ); - if ( hosts.open( QIODevice::ReadOnly ) ) { - // read the /etc/hosts file, creating long-life A and PTR RRs - // for the things we find. - QTextStream i( &hosts ); - TQString line; - while( !i.atEnd() ) { - line = TQT_TQSTRING(i.readLine()).simplifyWhiteSpace().lower(); - uint n = 0; - while( (int) n < line.length() && line[(int)n] != QLatin1Char('#') ) - n++; - line.truncate( n ); - n = 0; - while( (int) n < line.length() && !line[(int)n].isSpace() ) - n++; - TQString ip = line.left( n ); - TQHostAddress a; - a.setAddress( ip ); - if ( ( a.isIPv4Address() || a.isIPv6Address() ) && !a.isNull() ) { - bool first = true; - line = line.mid( n+1 ); - n = 0; - while( (int) n < line.length() && !line[(int)n].isSpace() ) - n++; - TQString hostname = line.left( n ); - // ### in case of bad syntax, hostname is invalid. do we care? - if ( n ) { - TQDnsRR * rr = new TQDnsRR( hostname ); - if ( a.isIPv4Address() ) - rr->t = TQDns::A; - else - rr->t = TQDns::Aaaa; - rr->address = a; - rr->deleteTime = UINT_MAX; - rr->expireTime = UINT_MAX; - rr->current = true; - if ( first ) { - first = false; - TQDnsRR * ptr = new TQDnsRR( TQDns::toInAddrArpaDomain( a ) ); - ptr->t = TQDns::Ptr; - ptr->target = hostname; - ptr->deleteTime = UINT_MAX; - ptr->expireTime = UINT_MAX; - ptr->current = true; - } - } - } - } - } -} - -#endif - -QT_END_NAMESPACE - -#endif // QT_NO_DNS - - -#else // USE_QT4 - -/**************************************************************************** -** -** Implementation of TQDns class. -** -** Created : 991122 -** -** Copyright (C) 1999-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqplatformdefs.h" - -// POSIX Large File Support redefines open -> open64 -#if defined(open) -# undef open -#endif - -// POSIX Large File Support redefines truncate -> truncate64 -#if defined(truncate) -# undef truncate -#endif - -// Solaris redefines connect -> __xnet_connect with _XOPEN_SOURCE_EXTENDED. -#if defined(connect) -# undef connect -#endif - -// UnixWare 7 redefines socket -> _socket -#if defined(socket) -# undef socket -#endif - -#include "tqdns.h" - -#ifndef TQT_NO_DNS - -#include "tqdatetime.h" -#include "tqdict.h" -#include "tqptrlist.h" -#include "tqstring.h" -#include "tqtimer.h" -#include "tqapplication.h" -#include "tqptrvector.h" -#include "tqstrlist.h" -#include "tqptrdict.h" -#include "tqfile.h" -#include "tqtextstream.h" -#include "tqsocketdevice.h" -#include "tqcleanuphandler.h" -#include <limits.h> -#ifdef TQ_OS_MAC -#include "../3rdparty/dlcompat/dlfcn.h" -#endif - -//#define TQDNS_DEBUG - -static TQ_UINT16 id; // ### seeded started by now() - - -static TQDateTime * originOfTime = 0; - -static TQCleanupHandler<TQDateTime> qdns_cleanup_time; - -static TQ_UINT32 now() -{ - if ( originOfTime ) - return originOfTime->secsTo( TQDateTime::tqcurrentDateTime() ); - - originOfTime = new TQDateTime( TQDateTime::tqcurrentDateTime() ); - ::id = originOfTime->time().msec() * 60 + originOfTime->time().second(); - qdns_cleanup_time.add( &originOfTime ); - return 0; -} - - -static TQPtrList<TQHostAddress> * ns = 0; -static TQStrList * domains = 0; -static bool ipv6support = FALSE; - -static int qdns_res_init() -{ -#ifdef TQ_OS_MAC - typedef int (*PtrRes_init)(); - static PtrRes_init ptrRes_init = 0; - if (!ptrRes_init) - ptrRes_init = (PtrRes_init)DL_PREFIX(dlsym)(RTLD_NEXT, "res_init"); - if (ptrRes_init) - return (*ptrRes_init)(); - else - return -1; -#elif defined(TQ_OS_UNIX) - return res_init(); -#else - return 0; // not called at all on Windows. -#endif -} - - -class TQDnsPrivate { -public: - TQDnsPrivate() : queryTimer( 0 ), noNames(FALSE) - { -#if defined(TQ_DNS_SYNCHRONOUS) -#if defined(TQ_OS_UNIX) - noEventLoop = tqApp==0 || tqApp->loopLevel()==0; -#else - noEventLoop = FALSE; -#endif -#endif - } - ~TQDnsPrivate() - { - delete queryTimer; - } -private: - TQTimer * queryTimer; - bool noNames; -#if defined(TQ_DNS_SYNCHRONOUS) - bool noEventLoop; -#endif - - friend class TQDns; - friend class TQDnsAnswer; -}; - - -class TQDnsRR; -class TQDnsDomain; - - - -// TQDnsRR is the class used to store a single RR. TQDnsRR can store -// all of the supported RR types. a TQDnsRR is always cached. - -// TQDnsRR is mostly constructed from the outside. a but hacky, but -// permissible since the entire class is internal. - -class TQDnsRR { -public: - TQDnsRR( const TQString & label ); - ~TQDnsRR(); - -public: - TQDnsDomain * domain; - TQDns::RecordType t; - bool nxdomain; - bool current; - TQ_UINT32 expireTime; - TQ_UINT32 deleteTime; - // somewhat space-wasting per-type data - // a / aaaa - TQHostAddress address; - // cname / mx / srv / ptr - TQString target; - // mx / srv - TQ_UINT16 priority; - // srv - TQ_UINT16 weight; - TQ_UINT16 port; - // txt - TQString text; // could be overloaded into target... -private: - -}; - - -class TQDnsDomain { -public: - TQDnsDomain( const TQString & label ); - ~TQDnsDomain(); - - static void add( const TQString & label, TQDnsRR * ); - static TQPtrList<TQDnsRR> * cached( const TQDns * ); - - void take( TQDnsRR * ); - - void sweep( TQ_UINT32 thisSweep ); - - bool isEmpty() const { return rrs == 0 || rrs->isEmpty(); } - - TQString name() const { return l; } - -public: - TQString l; - TQPtrList<TQDnsRR> * rrs; -}; - - -class TQDnsQuery: public TQTimer { // this inheritance is a very evil hack -public: - TQDnsQuery(): - id( 0 ), t( TQDns::None ), step(0), started(0), - dns( new TQPtrDict<void>(17) ) {} - ~TQDnsQuery() { delete dns; } - TQ_UINT16 id; - TQDns::RecordType t; - TQString l; - - uint step; - TQ_UINT32 started; - - TQPtrDict<void> * dns; -}; - - - -class TQDnsAnswer { -public: - TQDnsAnswer( TQDnsQuery * ); - TQDnsAnswer( const TQByteArray &, TQDnsQuery * ); - ~TQDnsAnswer(); - - void parse(); - void notify(); - - bool ok; - -private: - TQDnsQuery * query; - - TQ_UINT8 * answer; - int size; - int pp; - - TQPtrList<TQDnsRR> * rrs; - - // convenience - int next; - int ttl; - TQString label; - TQDnsRR * rr; - - TQString readString(bool multipleLabels = TRUE); - void parseA(); - void parseAaaa(); - void parseMx(); - void parseSrv(); - void parseCname(); - void parsePtr(); - void parseTxt(); - void parseNs(); -}; - - -TQDnsRR::TQDnsRR( const TQString & label ) - : domain( 0 ), t( TQDns::None ), - nxdomain( FALSE ), current( FALSE ), - expireTime( 0 ), deleteTime( 0 ), - priority( 0 ), weight( 0 ), port( 0 ) -{ - TQDnsDomain::add( label, this ); -} - - -// not supposed to be deleted except by TQDnsDomain -TQDnsRR::~TQDnsRR() -{ - // nothing is necessary -} - - -// this one just sticks in a NXDomain -TQDnsAnswer::TQDnsAnswer( TQDnsQuery * query_ ) -{ - ok = TRUE; - - answer = 0; - size = 0; - query = query_; - pp = 0; - rrs = new TQPtrList<TQDnsRR>; - rrs->setAutoDelete( FALSE ); - next = size; - ttl = 0; - label = TQString::null; - rr = 0; - - TQDnsRR * newrr = new TQDnsRR( query->l ); - newrr->t = query->t; - newrr->deleteTime = query->started + 10; - newrr->expireTime = query->started + 10; - newrr->nxdomain = TRUE; - newrr->current = TRUE; - rrs->append( newrr ); -} - - -TQDnsAnswer::TQDnsAnswer( const TQByteArray& answer_, - TQDnsQuery * query_ ) -{ - ok = TRUE; - - answer = (TQ_UINT8 *)(answer_.data()); - size = (int)answer_.size(); - query = query_; - pp = 0; - rrs = new TQPtrList<TQDnsRR>; - rrs->setAutoDelete( FALSE ); - next = size; - ttl = 0; - label = TQString::null; - rr = 0; -} - - -TQDnsAnswer::~TQDnsAnswer() -{ - if ( !ok && rrs ) { - TQPtrListIterator<TQDnsRR> it( *rrs ); - TQDnsRR * tmprr; - while( (tmprr=it.current()) != 0 ) { - ++it; - tmprr->t = TQDns::None; // will be deleted soonish - } - } - delete rrs; -} - - -TQString TQDnsAnswer::readString(bool multipleLabels) -{ - int p = pp; - TQString r = TQString::null; - TQ_UINT8 b; - for( ;; ) { - b = 128; - // Read one character - if ( p >= 0 && p < size ) - b = answer[p]; - - switch( b >> 6 ) { - case 0: - // b is less than 64 - p++; - - // Detect end of data - if ( b == 0 ) { - if ( p > pp ) - pp = p; - return r.isNull() ? TQString( "." ) : r; - } - - // Read a label of size 'b' characters - if ( !r.isNull() ) - r += '.'; - while( b-- > 0 ) { - r += TQChar( answer[p] ); - p++; - } - - // Return immediately if we were only supposed to read one - // label. - if (!multipleLabels) - return r; - - break; - default: - // Ignore unrecognized control character, or p was out of - // range. - goto not_ok; - case 3: - // Use the next character to determine the relative offset - // to jump to before continuing the packet parsing. - int q = ( (answer[p] & 0x3f) << 8 ) + answer[p+1]; - - if ( q >= pp || q >= p ) - goto not_ok; - if ( p >= pp ) - pp = p + 2; - p = q; - } - } -not_ok: - ok = FALSE; - return TQString::null; -} - - - -void TQDnsAnswer::parseA() -{ - if ( next != pp + 4 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN A for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::A; - rr->address = TQHostAddress( ( answer[pp+0] << 24 ) + - ( answer[pp+1] << 16 ) + - ( answer[pp+2] << 8 ) + - ( answer[pp+3] ) ); -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN A %s (ttl %d)", label.ascii(), - rr->address.toString().ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseAaaa() -{ - if ( next != pp + 16 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN Aaaa for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Aaaa; - rr->address = TQHostAddress( answer+pp ); -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN Aaaa %s (ttl %d)", label.ascii(), - rr->address.toString().ascii(), ttl ); -#endif -} - - - -void TQDnsAnswer::parseMx() -{ - if ( next < pp + 2 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN MX for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->priority = (answer[pp] << 8) + answer[pp+1]; - pp += 2; - rr->target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad string in MX for %s", label.ascii() ); -#endif - return; - } - rr->t = TQDns::Mx; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN MX %d %s (ttl %d)", label.ascii(), - rr->priority, rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseSrv() -{ - if ( next < pp + 6 ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %d bytes long IN SRV for %s", - next - pp, label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->priority = (answer[pp] << 8) + answer[pp+1]; - rr->weight = (answer[pp+2] << 8) + answer[pp+3]; - rr->port = (answer[pp+4] << 8) + answer[pp+5]; - pp += 6; - rr->target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad string in SRV for %s", label.ascii() ); -#endif - return; - } - rr->t = TQDns::Srv; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN SRV %d %d %d %s (ttl %d)", label.ascii(), - rr->priority, rr->weight, rr->port, rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseCname() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad cname for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Cname; - rr->target = target; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN CNAME %s (ttl %d)", label.ascii(), - rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseNs() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad cname for for %s", label.ascii() ); -#endif - return; - } - - // parse, but ignore - -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN NS %s (ttl %d)", label.ascii(), - target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parsePtr() -{ - TQString target = readString().lower(); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad PTR for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Ptr; - rr->target = target; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN PTR %s (ttl %d)", label.ascii(), - rr->target.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parseTxt() -{ - TQString text = readString(FALSE); - if ( !ok ) { -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw bad TXT for for %s", label.ascii() ); -#endif - return; - } - - rr = new TQDnsRR( label ); - rr->t = TQDns::Txt; - rr->text = text; -#if defined(TQDNS_DEBUG) - qDebug( "TQDns: saw %s IN TXT \"%s\" (ttl %d)", label.ascii(), - rr->text.ascii(), ttl ); -#endif -} - - -void TQDnsAnswer::parse() -{ - // okay, do the work... - if ( (answer[2] & 0x78) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: answer to wrong query type (%d)", answer[1] ); -#endif - ok = FALSE; - return; - } - - // AA - bool aa = (answer[2] & 4) != 0; - - // TC - if ( (answer[2] & 2) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: truncated answer; pressing on" ); -#endif - } - - // RD - bool rd = (answer[2] & 1) != 0; - - // we don't test RA - // we don't test the MBZ fields - - if ( (answer[3] & 0x0f) == 3 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: saw NXDomain for %s", query->l.ascii() ); -#endif - // NXDomain. cache that for one minute. - rr = new TQDnsRR( query->l ); - rr->t = query->t; - rr->deleteTime = query->started + 60; - rr->expireTime = query->started + 60; - rr->nxdomain = TRUE; - rr->current = TRUE; - rrs->append( rr ); - return; - } - - if ( (answer[3] & 0x0f) != 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: error code %d", answer[3] & 0x0f ); -#endif - ok = FALSE; - return; - } - - int qdcount = ( answer[4] << 8 ) + answer[5]; - int ancount = ( answer[6] << 8 ) + answer[7]; - int nscount = ( answer[8] << 8 ) + answer[9]; - int adcount = (answer[10] << 8 ) +answer[11]; - - pp = 12; - - // read query - while( qdcount > 0 && pp < size ) { - // should I compare the string against query->l? - (void)readString(); - if ( !ok ) - return; - pp += 4; - qdcount--; - } - - // answers and stuff - int rrno = 0; - // if we parse the answer completely, but there are no answers, - // ignore the entire thing. - int answers = 0; - while( ( rrno < ancount || - ( ok && answers >0 && rrno < ancount + nscount + adcount ) ) && - pp < size ) { - label = readString().lower(); - if ( !ok ) - return; - int rdlength = 0; - if ( pp + 10 <= size ) - rdlength = ( answer[pp+8] << 8 ) + answer[pp+9]; - if ( pp + 10 + rdlength > size ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: ran out of stuff to parse (%d+%d>%d (%d)", - pp, rdlength, size, rrno < ancount ); -#endif - // if we're still in the AN section, we should go back and - // at least down the TTLs. probably best to tqinvalidate - // the results. - // the rrs list is good for this - ok = ( rrno < ancount ); - return; - } - uint type, clas; - type = ( answer[pp+0] << 8 ) + answer[pp+1]; - clas = ( answer[pp+2] << 8 ) + answer[pp+3]; - ttl = ( answer[pp+4] << 24 ) + ( answer[pp+5] << 16 ) + - ( answer[pp+6] << 8 ) + answer[pp+7]; - pp = pp + 10; - if ( clas != 1 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: class %d (not internet) for %s", - clas, label.isNull() ? "." : label.ascii() ); -#endif - } else { - next = pp + rdlength; - rr = 0; - switch( type ) { - case 1: - parseA(); - break; - case 28: - parseAaaa(); - break; - case 15: - parseMx(); - break; - case 33: - parseSrv(); - break; - case 5: - parseCname(); - break; - case 12: - parsePtr(); - break; - case 16: - parseTxt(); - break; - case 2: - parseNs(); - break; - default: - // something we don't know -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: type %d for %s", type, - label.isNull() ? "." : label.ascii() ); -#endif - break; - } - if ( rr ) { - rr->deleteTime = 0; - if ( ttl > 0 ) - rr->expireTime = query->started + ttl; - else - rr->expireTime = query->started + 20; - if ( rrno < ancount ) { - answers++; - rr->deleteTime = rr->expireTime; - } - rr->current = TRUE; - rrs->append( rr ); - } - } - if ( !ok ) - return; - pp = next; - next = size; - rrno++; - } - if ( answers == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: answer contained no answers" ); -#endif - ok = ( aa && rd ); - } - - // now go through the list and mark all the As that are referenced - // by something we care about. we want to cache such As. - rrs->first(); - TQDict<void> used( 17 ); - used.setAutoDelete( FALSE ); - while( (rr=rrs->current()) != 0 ) { - rrs->next(); - if ( rr->target.length() && rr->deleteTime > 0 && rr->current ) - used.insert( rr->target, (void*)42 ); - if ( ( rr->t == TQDns::A || rr->t == TQDns::Aaaa ) && - used.find( rr->domain->name() ) != 0 ) - rr->deleteTime = rr->expireTime; - } - - // next, for each RR, delete any older RRs that are equal to it - rrs->first(); - while( (rr=rrs->current()) != 0 ) { - rrs->next(); - if ( rr && rr->domain && rr->domain->rrs ) { - TQPtrList<TQDnsRR> * drrs = rr->domain->rrs; - drrs->first(); - TQDnsRR * older; - while( (older=drrs->current()) != 0 ) { - if ( older != rr && - older->t == rr->t && - older->nxdomain == rr->nxdomain && - older->address == rr->address && - older->target == rr->target && - older->priority == rr->priority && - older->weight == rr->weight && - older->port == rr->port && - older->text == rr->text ) { - // well, it's equal, but it's not the same. so we kill it, - // but use its expiry time. -#if defined(TQDNS_DEBUG) - qDebug( "killing off old %d for %s, expire was %d", - older->t, older->domain->name().latin1(), - rr->expireTime ); -#endif - older->t = TQDns::None; - rr->expireTime = TQMAX( older->expireTime, rr->expireTime ); - rr->deleteTime = TQMAX( older->deleteTime, rr->deleteTime ); - older->deleteTime = 0; -#if defined(TQDNS_DEBUG) - qDebug( " adjusted expire is %d", rr->expireTime ); -#endif - } - drrs->next(); - } - } - } - -#if defined(TQDNS_DEBUG) - //qDebug( "DNS Manager: ()" ); -#endif -} - - -class TQDnsUgleHack: public TQDns { -public: - void ugle( bool emitAnyway=FALSE ); -}; - - -void TQDnsAnswer::notify() -{ - if ( !rrs || !ok || !query || !query->dns ) - return; - - TQPtrDict<void> notified; - notified.setAutoDelete( FALSE ); - - TQPtrDictIterator<void> it( *query->dns ); - TQDns * dns; - it.toFirst(); - while( (dns=(TQDns*)(it.current())) != 0 ) { - ++it; - if ( notified.find( (void*)dns ) == 0 ) { - notified.insert( (void*)dns, (void*)42 ); - if ( rrs->count() == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: found no answers!" ); -#endif - dns->d->noNames = TRUE; - ((TQDnsUgleHack*)dns)->ugle( TRUE ); - } else { - TQStringList n = dns->qualifiedNames(); - if ( query && n.contains(query->l) ) - ((TQDnsUgleHack*)dns)->ugle(); -#if defined(TQDNS_DEBUG) - else - qDebug( "DNS Manager: DNS thing %s not notified for %s", - dns->label().ascii(), query->l.ascii() ); -#endif - } - } - } -} - - -// -// -// TQDnsManager -// -// - - -class TQDnsManager: public TQDnsSocket { -private: -public: // just to silence the moronic g++. - TQDnsManager(); - ~TQDnsManager(); -public: - static TQDnsManager * manager(); - - TQDnsDomain * domain( const TQString & ); - - void transmitQuery( TQDnsQuery * ); - void transmitQuery( int ); - - // reimplementation of the Q_SLOTS - void cleanCache(); - void retransmit(); - void answer(); - -public: - TQPtrVector<TQDnsQuery> queries; - TQDict<TQDnsDomain> cache; - TQSocketDevice * ipv4Socket; -#if !defined (TQT_NO_IPV6) - TQSocketDevice * ipv6Socket; -#endif -}; - - - -static TQDnsManager * globalManager = 0; - -static void cleanupDns() -{ - delete globalManager; - globalManager = 0; -} - -TQDnsManager * TQDnsManager::manager() -{ - if ( !globalManager ) { - qAddPostRoutine(cleanupDns); - new TQDnsManager(); - } - return globalManager; -} - - -void TQDnsUgleHack::ugle( bool emitAnyway) -{ - if ( emitAnyway || !isWorking() ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: status change for %s (type %d)", - label().ascii(), recordType() ); -#endif - emit resultsReady(); - } -} - - -TQDnsManager::TQDnsManager() - : TQDnsSocket( tqApp, "Internal DNS manager" ), - queries( TQPtrVector<TQDnsQuery>( 0 ) ), - cache( TQDict<TQDnsDomain>( 83, FALSE ) ), - ipv4Socket( new TQSocketDevice( TQSocketDevice::Datagram, TQSocketDevice::IPv4, 0 ) ) -#if !defined (TQT_NO_IPV6) - , ipv6Socket( new TQSocketDevice( TQSocketDevice::Datagram, TQSocketDevice::IPv6, 0 ) ) -#endif -{ - cache.setAutoDelete( TRUE ); - globalManager = this; - - TQTimer * sweepTimer = new TQTimer( this ); - sweepTimer->start( 1000 * 60 * 3 ); - connect( sweepTimer, TQT_SIGNAL(timeout()), - this, TQT_SLOT(cleanCache()) ); - - TQSocketNotifier * rn4 = new TQSocketNotifier( ipv4Socket->socket(), - TQSocketNotifier::Read, - TQT_TQOBJECT(this), "dns IPv4 socket watcher" ); - ipv4Socket->setAddressReusable( FALSE ); - ipv4Socket->setBlocking( FALSE ); - connect( rn4, TQT_SIGNAL(activated(int)), TQT_SLOT(answer()) ); - -#if !defined (TQT_NO_IPV6) - // Don't connect the IPv6 socket notifier if the host does not - // support IPv6. - if ( ipv6Socket->socket() != -1 ) { - TQSocketNotifier * rn6 = new TQSocketNotifier( ipv6Socket->socket(), - TQSocketNotifier::Read, - TQT_TQOBJECT(this), "dns IPv6 socket watcher" ); - - ipv6support = TRUE; - ipv6Socket->setAddressReusable( FALSE ); - ipv6Socket->setBlocking( FALSE ); - connect( rn6, TQT_SIGNAL(activated(int)), TQT_SLOT(answer()) ); - } -#endif - - if ( !ns ) - TQDns::doResInit(); - - // O(n*n) stuff here. but for 3 and 6, O(n*n) with a low k should - // be perfect. the point is to eliminate any duplicates that - // might be hidden in the lists. - TQPtrList<TQHostAddress> * ns = new TQPtrList<TQHostAddress>; - - ::ns->first(); - TQHostAddress * h; - while( (h=::ns->current()) != 0 ) { - ns->first(); - while( ns->current() != 0 && !(*ns->current() == *h) ) - ns->next(); - if ( !ns->current() ) { - ns->append( new TQHostAddress(*h) ); -#if defined(TQDNS_DEBUG) - qDebug( "using name server %s", h->toString().latin1() ); - } else { - qDebug( "skipping address %s", h->toString().latin1() ); -#endif - } - ::ns->next(); - } - - delete ::ns; - ::ns = ns; - ::ns->setAutoDelete( TRUE ); - - TQStrList * domains = new TQStrList( TRUE ); - - ::domains->first(); - const char * s; - while( (s=::domains->current()) != 0 ) { - domains->first(); - while( domains->current() != 0 && qstrcmp( domains->current(), s ) ) - domains->next(); - if ( !domains->current() ) { - domains->append( s ); -#if defined(TQDNS_DEBUG) - qDebug( "searching domain %s", s ); - } else { - qDebug( "skipping domain %s", s ); -#endif - } - ::domains->next(); - } - - delete ::domains; - ::domains = domains; - ::domains->setAutoDelete( TRUE ); -} - - -TQDnsManager::~TQDnsManager() -{ - if ( globalManager ) - globalManager = 0; - queries.setAutoDelete( TRUE ); - cache.setAutoDelete( TRUE ); - delete ipv4Socket; -#if !defined (TQT_NO_IPV6) - delete ipv6Socket; -#endif -} - -static TQ_UINT32 lastSweep = 0; - -void TQDnsManager::cleanCache() -{ - bool again = FALSE; - TQDictIterator<TQDnsDomain> it( cache ); - TQDnsDomain * d; - TQ_UINT32 thisSweep = now(); -#if defined(TQDNS_DEBUG) - qDebug( "TQDnsManager::cleanCache(: Called, time is %u, last was %u", - thisSweep, lastSweep ); -#endif - - while( (d=it.current()) != 0 ) { - ++it; - d->sweep( thisSweep ); // after this, d may be empty - if ( !again ) - again = !d->isEmpty(); - } - if ( !again ) - delete this; - lastSweep = thisSweep; -} - - -void TQDnsManager::retransmit() -{ - const TQObject * o = sender(); - if ( o == 0 || globalManager == 0 || this != globalManager ) - return; - uint q = 0; - while( q < queries.size() && queries[q] != o ) - q++; - if ( q < queries.size() ) - transmitQuery( q ); -} - - -void TQDnsManager::answer() -{ - TQByteArray a( 16383 ); // large enough for anything, one suspects - - int r; -#if defined (TQT_NO_IPV6) - r = ipv4Socket->readBlock(a.data(), a.size()); -#else - if (((TQSocketNotifier *)sender())->socket() == ipv4Socket->socket()) - r = ipv4Socket->readBlock(a.data(), a.size()); - else - r = ipv6Socket->readBlock(a.data(), a.size()); -#endif -#if defined(TQDNS_DEBUG) -#if !defined (TQT_NO_IPV6) - qDebug("DNS Manager: answer arrived: %d bytes from %s:%d", r, - useIpv4Socket ? ipv4Socket->peerAddress().toString().ascii() - : ipv6Socket->peerAddress().toString().ascii(), - useIpv4Socket ? ipv4Socket->peerPort() : ipv6Socket->peerPort() ); -#else - qDebug("DNS Manager: answer arrived: %d bytes from %s:%d", r, - ipv4Socket->peerAddress().toString().ascii(), ipv4Socket->peerPort());; -#endif -#endif - if ( r < 12 ) - return; - // maybe we should check that the answer comes from port 53 on one - // of our name servers... - a.resize( r ); - - TQ_UINT16 aid = (((TQ_UINT8)a[0]) << 8) + ((TQ_UINT8)a[1]); - uint i = 0; - while( i < queries.size() && - !( queries[i] && queries[i]->id == aid ) ) - i++; - if ( i == queries.size() ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: bad id (0x%04x) %d", aid, i ); -#endif - return; - } - - // at this point queries[i] is whatever we asked for. - - if ( ( (TQ_UINT8)(a[2]) & 0x80 ) == 0 ) { -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: received a query" ); -#endif - return; - } - - TQDnsQuery * q = queries[i]; - TQDnsAnswer answer( a, q ); - answer.parse(); - if ( answer.ok ) { - queries.take( i ); - answer.notify(); - delete q; - } -} - - -void TQDnsManager::transmitQuery( TQDnsQuery * query_ ) -{ - if ( !query_ ) - return; - - uint i = 0; - while( i < queries.size() && queries[i] != 0 ) - i++; - if ( i == queries.size() ) - queries.resize( i+1 ); - queries.insert( i, query_ ); - transmitQuery( i ); -} - - -void TQDnsManager::transmitQuery( int i ) -{ - if ( i < 0 || i >= (int)queries.size() ) - return; - TQDnsQuery * q = queries[i]; - - if ( q && q->step > 8 ) { - // okay, we've run out of retransmissions. we fake an NXDomain - // with a very short life time... - TQDnsAnswer answer( q ); - answer.notify(); - - if (globalManager == 0) - return; - - // and then get rid of the query - queries.take( i ); -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: giving up on query 0x%04x", q->id ); -#endif - delete q; - TQTimer::singleShot( 0, TQDnsManager::manager(), TQT_SLOT(cleanCache()) ); - // and don't process anything more - return; - } - - if ( q && !q->dns || q->dns->isEmpty() ) - // noone currently wants the answer, so there's no point in - // retransmitting the query. we keep it, though. an answer may - // arrive for an earlier query transmission, and if it does we - // may benefit from caching the result. - return; - - TQByteArray p( 12 + q->l.length() + 2 + 4 ); - if ( p.size() > 500 ) - return; // way over the limit, so don't even try - - // header - // id - p[0] = (q->id & 0xff00) >> 8; - p[1] = q->id & 0x00ff; - p[2] = 1; // recursion desired, rest is 0 - p[3] = 0; // all is 0 - // one query - p[4] = 0; - p[5] = 1; - // no answers, name servers or additional data - p[6] = p[7] = p[8] = p[9] = p[10] = p[11] = 0; - - // the name is composed of several components. each needs to be - // written by itself... so we write... - // oh, and we assume that there's no funky characters in there. - int pp = 12; - uint lp = 0; - while( lp < q->l.length() ) { - int le = q->l.find( '.', lp ); - if ( le < 0 ) - le = q->l.length(); - TQString component = q->l.mid( lp, le-lp ); - p[pp++] = component.length(); - int cp; - for( cp=0; cp < (int)component.length(); cp++ ) - p[pp++] = component[cp].latin1(); - lp = le + 1; - } - // final null - p[pp++] = 0; - // query type - p[pp++] = 0; - switch( q->t ) { - case TQDns::A: - p[pp++] = 1; - break; - case TQDns::Aaaa: - p[pp++] = 28; - break; - case TQDns::Mx: - p[pp++] = 15; - break; - case TQDns::Srv: - p[pp++] = 33; - break; - case TQDns::Cname: - p[pp++] = 5; - break; - case TQDns::Ptr: - p[pp++] = 12; - break; - case TQDns::Txt: - p[pp++] = 16; - break; - default: - p[pp++] = (char)255; // any - break; - } - // query class (always internet) - p[pp++] = 0; - p[pp++] = 1; - - // if we have no name servers, we should regenerate ns in case - // name servers have recently been defined (like on windows, - // plugging/unplugging the network cable will change the name - // server entries) - if ( !ns || ns->isEmpty() ) - TQDns::doResInit(); - - if ( !ns || ns->isEmpty() ) { - // we don't find any name servers. We fake an NXDomain - // with a very short life time... - TQDnsAnswer answer( q ); - answer.notify(); - // and then get rid of the query - queries.take( i ); -#if defined(TQDNS_DEBUG) - qDebug( "DNS Manager: no DNS server found on query 0x%04x", q->id ); -#endif - delete q; - TQTimer::singleShot( 1000*10, TQDnsManager::manager(), TQT_SLOT(cleanCache()) ); - // and don't process anything more - return; - } - - TQHostAddress receiver = *ns->at( q->step % ns->count() ); - if (receiver.isIPv4Address()) - ipv4Socket->writeBlock( p.data(), pp, receiver, 53 ); -#if !defined (TQT_NO_IPV6) - else - ipv6Socket->writeBlock( p.data(), pp, receiver, 53 ); -#endif -#if defined(TQDNS_DEBUG) - qDebug( "issuing query 0x%04x (%d) about %s type %d to %s", - q->id, q->step, q->l.ascii(), q->t, - ns->at( q->step % ns->count() )->toString().ascii() ); -#endif - if ( ns->count() > 1 && q->step == 0 && queries.count() == 1 ) { - // if it's the first time, and we don't have any other - // outstanding queries, send nonrecursive queries to the other - // name servers too. - p[2] = 0; - TQHostAddress * server; - while( (server=ns->next()) != 0 ) { - if (server->isIPv4Address()) - ipv4Socket->writeBlock( p.data(), pp, *server, 53 ); -#if !defined (TQT_NO_IPV6) - else - ipv6Socket->writeBlock( p.data(), pp, *server, 53 ); -#endif -#if defined(TQDNS_DEBUG) - qDebug( "copying query to %s", server->toString().ascii() ); -#endif - } - } - q->step++; - // some testing indicates that normal dns queries take up to 0.6 - // seconds. the graph becomes steep around that point, and the - // number of errors rises... so it seems good to retry at that - // point. - q->start( q->step < ns->count() ? 800 : 1500, TRUE ); -} - - -TQDnsDomain * TQDnsManager::domain( const TQString & label ) -{ - TQDnsDomain * d = cache.find( label ); - if ( !d ) { - d = new TQDnsDomain( label ); - cache.insert( label, d ); - } - return d; -} - - -// -// -// the TQDnsDomain class looks after and coordinates queries for TQDnsRRs for -// each domain, and the cached TQDnsRRs. (A domain, in DNS terminology, is -// a node in the DNS. "no", "trolltech.com" and "lupinella.troll.no" are -// all domains.) -// -// - - -// this is ONLY to be called by TQDnsManager::domain(). noone else. -TQDnsDomain::TQDnsDomain( const TQString & label ) -{ - l = label; - rrs = 0; -} - - -TQDnsDomain::~TQDnsDomain() -{ - delete rrs; - rrs = 0; -} - - -void TQDnsDomain::add( const TQString & label, TQDnsRR * rr ) -{ - TQDnsDomain * d = TQDnsManager::manager()->domain( label ); - if ( !d->rrs ) { - d->rrs = new TQPtrList<TQDnsRR>; - d->rrs->setAutoDelete( TRUE ); - } - d->rrs->append( rr ); - rr->domain = d; -} - - -TQPtrList<TQDnsRR> * TQDnsDomain::cached( const TQDns * r ) -{ - TQPtrList<TQDnsRR> * l = new TQPtrList<TQDnsRR>; - - // test at first if you have to start a query at all - if ( r->recordType() == TQDns::A ) { - if ( r->label().lower() == "localhost" ) { - // undocumented hack. ipv4-specific. also, may be a memory - // leak? not sure. would be better to do this in doResInit(), - // anyway. - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - rrTmp->t = TQDns::A; - rrTmp->address = TQHostAddress( 0x7f000001 ); - rrTmp->current = TRUE; - l->append( rrTmp ); - return l; - } - TQHostAddress tmp; - if ( tmp.setAddress( r->label() ) ) { - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - if ( tmp.isIPv4Address() ) { - rrTmp->t = TQDns::A; - rrTmp->address = tmp; - rrTmp->current = TRUE; - l->append( rrTmp ); - } else { - rrTmp->nxdomain = TRUE; - } - return l; - } - } - if ( r->recordType() == TQDns::Aaaa ) { - TQHostAddress tmp; - if ( tmp.setAddress(r->label()) ) { - TQDnsRR *rrTmp = new TQDnsRR( r->label() ); - if ( tmp.isIPv6Address() ) { - rrTmp->t = TQDns::Aaaa; - rrTmp->address = tmp; - rrTmp->current = TRUE; - l->append( rrTmp ); - } else { - rrTmp->nxdomain = TRUE; - } - return l; - } - } - - // if you reach this point, you have to do the query - TQDnsManager * m = TQDnsManager::manager(); - TQStringList n = r->qualifiedNames(); - TQValueListIterator<TQString> it = n.begin(); - TQValueListIterator<TQString> end = n.end(); - bool nxdomain; - int cnamecount = 0; - while( it != end ) { - TQString s = *it++; - nxdomain = FALSE; -#if defined(TQDNS_DEBUG) - qDebug( "looking at cache for %s (%s %d)", - s.ascii(), r->label().ascii(), r->recordType() ); -#endif - TQDnsDomain * d = m->domain( s ); -#if defined(TQDNS_DEBUG) - qDebug( " - found %d RRs", d && d->rrs ? d->rrs->count() : 0 ); -#endif - if ( d->rrs ) - d->rrs->first(); - TQDnsRR * rr; - bool answer = FALSE; - while( d->rrs && (rr=d->rrs->current()) != 0 ) { - if ( rr->t == TQDns::Cname && r->recordType() != TQDns::Cname && - !rr->nxdomain && cnamecount < 16 ) { - // cname. if the code is ugly, that may just - // possibly be because the concept is. -#if defined(TQDNS_DEBUG) - qDebug( "found cname from %s to %s", - r->label().ascii(), rr->target.ascii() ); -#endif - s = rr->target; - d = m->domain( s ); - if ( d->rrs ) - d->rrs->first(); - it = end; - // we've elegantly moved over to whatever the cname - // pointed to. well, not elegantly. let's remember - // that we've done something, anyway, so we can't be - // fooled into an infinte loop as well. - cnamecount++; - } else { - if ( rr->t == r->recordType() ) { - if ( rr->nxdomain ) - nxdomain = TRUE; - else - answer = TRUE; - l->append( rr ); - if ( rr->deleteTime <= lastSweep ) { - // we're returning something that'll be - // deleted soon. we assume that if the client - // wanted it twice, it'll want it again, so we - // ask the name server again right now. - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++::id; - query->t = rr->t; - query->l = rr->domain->name(); - // note that here, we don't bother about - // notification. but we do bother about - // timeouts: we make sure to use high timeouts - // and few tramsissions. - query->step = ns->count(); - TQObject::connect( query, TQT_SIGNAL(timeout()), - TQDnsManager::manager(), - TQT_SLOT(retransmit()) ); - TQDnsManager::manager()->transmitQuery( query ); - } - } - d->rrs->next(); - } - } - // if we found a positive result, return quickly - if ( answer && l->count() ) { -#if defined(TQDNS_DEBUG) - qDebug( "found %d records for %s", - l->count(), r->label().ascii() ); - l->first(); - while( l->current() ) { - qDebug( " type %d target %s address %s", - l->current()->t, - l->current()->target.latin1(), - l->current()->address.toString().latin1() ); - l->next(); - } -#endif - l->first(); - return l; - } - -#if defined(TQDNS_DEBUG) - if ( nxdomain ) - qDebug( "found NXDomain %s", s.ascii() ); -#endif - - if ( !nxdomain ) { - // if we didn't, and not a negative result either, perhaps - // we need to transmit a query. - uint q = 0; - while ( q < m->queries.size() && - ( m->queries[q] == 0 || - m->queries[q]->t != r->recordType() || - m->queries[q]->l != s ) ) - q++; - // we haven't done it before, so maybe we should. but - // wait - if it's an unqualified name, only ask when all - // the other alternatives are exhausted. - if ( q == m->queries.size() && ( s.find( '.' ) >= 0 || - l->count() >= n.count()-1 ) ) { - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++::id; - query->t = r->recordType(); - query->l = s; - query->dns->replace( (void*)r, (void*)r ); - TQObject::connect( query, TQT_SIGNAL(timeout()), - TQDnsManager::manager(), TQT_SLOT(retransmit()) ); - TQDnsManager::manager()->transmitQuery( query ); - } else if ( q < m->queries.size() ) { - // if we've found an earlier query for the same - // domain/type, subscribe to its answer - m->queries[q]->dns->replace( (void*)r, (void*)r ); - } - } - } - l->first(); - return l; -} - - -void TQDnsDomain::sweep( TQ_UINT32 thisSweep ) -{ - if ( !rrs ) - return; - - TQDnsRR * rr; - rrs->first(); - while( (rr=rrs->current()) != 0 ) { - if ( !rr->deleteTime ) - rr->deleteTime = thisSweep; // will hit next time around - -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::sweep: %s type %d expires %u %u - %s / %s", - rr->domain->name().latin1(), rr->t, - rr->expireTime, rr->deleteTime, - rr->target.latin1(), rr->address.toString().latin1()); -#endif - if ( rr->current == FALSE || - rr->t == TQDns::None || - rr->deleteTime <= thisSweep || - rr->expireTime <= thisSweep ) - rrs->remove(); - else - rrs->next(); - } - - if ( rrs->isEmpty() ) { - delete rrs; - rrs = 0; - } -} - - - - -// the itsy-bitsy little socket class I don't really need except for -// so I can subclass and reimplement the Q_SLOTS. - - -TQDnsSocket::TQDnsSocket( TQObject * parent, const char * name ) - : TQObject( parent, name ) -{ - // nothing -} - - -TQDnsSocket::~TQDnsSocket() -{ - // nothing -} - - -void TQDnsSocket::cleanCache() -{ - // nothing -} - - -void TQDnsSocket::retransmit() -{ - // nothing -} - - -void TQDnsSocket::answer() -{ - // nothing -} - - -/*! - \class TQDns tqdns.h - \brief The TQDns class provides asynchronous DNS lookups. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \module network - \ingroup io - - Both Windows and Unix provide synchronous DNS lookups; Windows - provides some asynchronous support too. At the time of writing - neither operating system provides asynchronous support for - anything other than hostname-to-address mapping. - - TQDns rectifies this shortcoming, by providing asynchronous caching - lookups for the record types that we expect modern GUI - applications to need in the near future. - - The class is \e not straightforward to use (although it is much - simpler than the native APIs); TQSocket provides much easier to use - TCP connection facilities. The aim of TQDns is to provide a correct - and small API to the DNS and nothing more. (We use "correctness" - to mean that the DNS information is correctly cached, and - correctly timed out.) - - The API comprises a constructor, functions to set the DNS node - (the domain in DNS terminology) and record type (setLabel() and - setRecordType()), the corresponding get functions, an isWorking() - function to determine whether TQDns is working or reading, a - resultsReady() signal and query functions for the result. - - There is one query function for each RecordType, namely - addresses(), mailServers(), servers(), hostNames() and texts(). - There are also two generic query functions: canonicalName() - returns the name you'll presumably end up using (the exact meaning - of this depends on the record type) and qualifiedNames() returns a - list of the fully qualified names label() maps to. - - \sa TQSocket -*/ - -/*! - Constructs a DNS query object with invalid settings for both the - label and the search type. -*/ - -TQDns::TQDns() -{ - d = new TQDnsPrivate; - t = None; -} - - - - -/*! - Constructs a DNS query object that will return record type \a rr - information about \a label. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \a rr defaults to \c A, IPv4 addresses. -*/ - -TQDns::TQDns( const TQString & label, RecordType rr ) -{ - d = new TQDnsPrivate; - t = rr; - setLabel( label ); - setStartQueryTimer(); // start query the next time we enter event loop -} - - - -/*! - Constructs a DNS query object that will return record type \a rr - information about host address \a address. The label is set to the - IN-ADDR.ARPA domain name. This is useful in combination with the - \c Ptr record type (e.g. if you want to look up a hostname for a - given address). - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \a rr defaults to \c Ptr, that maps addresses to hostnames. -*/ - -TQDns::TQDns( const TQHostAddress & address, RecordType rr ) -{ - d = new TQDnsPrivate; - t = rr; - setLabel( address ); - setStartQueryTimer(); // start query the next time we enter event loop -} - - - - -/*! - Destroys the DNS query object and frees its allocated resources. -*/ - -TQDns::~TQDns() -{ - if ( globalManager ) { - uint q = 0; - TQDnsManager * m = globalManager; - while( q < m->queries.size() ) { - TQDnsQuery * query=m->queries[q]; - if ( query && query->dns ) - (void)query->dns->take( (void*) this ); - q++; - } - - } - - delete d; - d = 0; -} - - - - -/*! - Sets this DNS query object to query for information about \a - label. - - This does not change the recordType(), but its isWorking() status - will probably change as a result. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. -*/ - -void TQDns::setLabel( const TQString & label ) -{ - l = label; - d->noNames = FALSE; - - // construct a list of qualified names - n.clear(); - if ( l.length() > 1 && l[(int)l.length()-1] == '.' ) { - n.append( l.left( l.length()-1 ).lower() ); - } else { - int i = l.length(); - int dots = 0; - const int maxDots = 2; - while( i && dots < maxDots ) { - if ( l[--i] == '.' ) - dots++; - } - if ( dots < maxDots ) { - (void)TQDnsManager::manager(); // create a TQDnsManager, if it is not already there - TQStrListIterator it( *domains ); - const char * dom; - while( (dom=it.current()) != 0 ) { - ++it; - n.append( l.lower() + "." + dom ); - } - } - n.append( l.lower() ); - } - -#if defined(TQ_DNS_SYNCHRONOUS) - if ( d->noEventLoop ) { - doSynchronousLookup(); - } else { - setStartQueryTimer(); // start query the next time we enter event loop - } -#else - setStartQueryTimer(); // start query the next time we enter event loop -#endif -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::setLabel: %d address(es) for %s", n.count(), l.ascii() ); - int i = 0; - for( i = 0; i < (int)n.count(); i++ ) - qDebug( "TQDns::setLabel: %d: %s", i, n[i].ascii() ); -#endif -} - - -/*! - \overload - - Sets this DNS query object to query for information about the host - address \a address. The label is set to the IN-ADDR.ARPA domain - name. This is useful in combination with the \c Ptr record type - (e.g. if you want to look up a hostname for a given address). -*/ - -void TQDns::setLabel( const TQHostAddress & address ) -{ - setLabel( toInAddrArpaDomain( address ) ); -} - - -/*! - \fn TQStringList TQDns::qualifiedNames() const - - Returns a list of the fully qualified names label() maps to. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQStringList list = myDns.qualifiedNames(); - TQStringList::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode - -*/ - - -/*! - \fn TQString TQDns::label() const - - Returns the domain name for which this object returns information. - - \sa setLabel() -*/ - -/*! - \enum TQDns::RecordType - - This enum type defines the record types TQDns can handle. The DNS - provides many more; these are the ones we've judged to be in - current use, useful for GUI programs and important enough to - support right away: - - \value None No information. This exists only so that TQDns can - have a default. - - \value A IPv4 addresses. By far the most common type. - - \value Aaaa IPv6 addresses. So far mostly unused. - - \value Mx Mail eXchanger names. Used for mail delivery. - - \value Srv SeRVer names. Generic record type for finding - servers. So far mostly unused. - - \value Cname Canonical names. Maps from nicknames to the true - name (the canonical name) for a host. - - \value Ptr name PoinTeRs. Maps from IPv4 or IPv6 addresses to hostnames. - - \value Txt arbitrary TeXT for domains. - - We expect that some support for the - \link http://www.dns.net/dnsrd/rfc/rfc2535.html RFC-2535 \endlink - extensions will be added in future versions. -*/ - -/*! - Sets this object to query for record type \a rr records. - - The DNS lookup is started the next time the application enters the - event loop. When the result is found the signal resultsReady() is - emitted. - - \sa RecordType -*/ - -void TQDns::setRecordType( RecordType rr ) -{ - t = rr; - d->noNames = FALSE; - setStartQueryTimer(); // start query the next time we enter event loop -} - -/*! - \internal - - Private slot for starting the query. -*/ -void TQDns::startQuery() -{ - // isWorking() starts the query (if necessary) - if ( !isWorking() ) - emit resultsReady(); -} - -/*! - The three functions TQDns::TQDns(TQString, RecordType), - TQDns::setLabel() and TQDns::setRecordType() may start a DNS lookup. - This function handles setting up the single shot timer. -*/ -void TQDns::setStartQueryTimer() -{ -#if defined(TQ_DNS_SYNCHRONOUS) - if ( !d->queryTimer && !d->noEventLoop ) -#else - if ( !d->queryTimer ) -#endif - { - // start the query the next time we enter event loop - d->queryTimer = new TQTimer( this ); - connect( d->queryTimer, TQT_SIGNAL(timeout()), - this, TQT_SLOT(startQuery()) ); - d->queryTimer->start( 0, TRUE ); - } -} - -/* - Transforms the host address \a address to the IN-ADDR.ARPA domain - name. Returns something indeterminate if you're sloppy or - naughty. This function has an IPv4-specific name, but works for - IPv6 too. -*/ -TQString TQDns::toInAddrArpaDomain( const TQHostAddress &address ) -{ - TQString s; - if ( address.isNull() ) { - // if the address isn't valid, neither of the other two make - // cases make sense. better to just return. - } else if ( address.isIp4Addr() ) { - TQ_UINT32 i = address.ip4Addr(); - s.sprintf( "%d.%d.%d.%d.IN-ADDR.ARPA", - i & 0xff, (i >> 8) & 0xff, (i>>16) & 0xff, (i>>24) & 0xff ); - } else { - // RFC 3152. (1886 is deprecated, and clients no longer need to - // support it, in practice). - TQ_IPV6ADDR i = address.toIPv6Address(); - s = "ip6.arpa"; - uint b = 0; - while( b < 16 ) { - s = TQString::number( i.c[b]%16, 16 ) + "." + - TQString::number( i.c[b]/16, 16 ) + "." + s; - b++; - } - } - return s; -} - - -/*! - \fn TQDns::RecordType TQDns::recordType() const - - Returns the record type of this DNS query object. - - \sa setRecordType() RecordType -*/ - -/*! - \fn void TQDns::resultsReady() - - This signal is emitted when results are available for one of the - qualifiedNames(). -*/ - -/*! - Returns TRUE if TQDns is doing a lookup for this object (i.e. if it - does not already have the necessary information); otherwise - returns FALSE. - - TQDns emits the resultsReady() signal when the status changes to FALSE. -*/ - -bool TQDns::isWorking() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::isWorking (%s, %d)", l.ascii(), t ); -#endif - if ( t == None ) - return FALSE; - -#if defined(TQ_DNS_SYNCHRONOUS) - if ( d->noEventLoop ) - return TRUE; -#endif - - TQPtrList<TQDnsRR> * ll = TQDnsDomain::cached( this ); - TQ_LONG queries = n.count(); - while( ll->current() != 0 ) { - if ( ll->current()->nxdomain ) { - queries--; - } else { - delete ll; - return FALSE; - } - ll->next(); - } - delete ll; - - if ( queries <= 0 ) - return FALSE; - if ( d->noNames ) - return FALSE; - return TRUE; -} - - -/*! - Returns a list of the addresses for this name if this TQDns object - has a recordType() of \c TQDns::A or \c TQDns::Aaaa and the answer - is available; otherwise returns an empty list. - - As a special case, if label() is a valid numeric IP address, this - function returns that address. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQValueList<TQHostAddress> list = myDns.addresses(); - TQValueList<TQHostAddress>::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode - -*/ - -TQValueList<TQHostAddress> TQDns::addresses() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::addresses (%s)", l.ascii() ); -#endif - TQValueList<TQHostAddress> result; - if ( t != A && t != Aaaa ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) - result.append( rr->address ); - cached->next(); - } - delete cached; - return result; -} - - -/*! - \class TQDns::MailServer - \brief The TQDns::MailServer class is described in TQDns::mailServers(). -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - - \internal -*/ - -/*! - Returns a list of mail servers if the record type is \c Mx. The - class \c TQDns::MailServer contains the following public variables: - \list - \i TQString TQDns::MailServer::name - \i TQ_UINT16 TQDns::MailServer::priority - \endlist - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQValueList<TQDns::MailServer> list = myDns.mailServers(); - TQValueList<TQDns::MailServer>::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode - -*/ -TQValueList<TQDns::MailServer> TQDns::mailServers() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::mailServers (%s)", l.ascii() ); -#endif - TQValueList<TQDns::MailServer> result; - if ( t != Mx ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - MailServer ms( rr->target, rr->priority ); - result.append( ms ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - \class TQDns::Server - \brief The TQDns::Server class is described in TQDns::servers(). -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - - \internal -*/ - -/*! - Returns a list of servers if the record type is \c Srv. The class - \c TQDns::Server contains the following public variables: - \list - \i TQString TQDns::Server::name - \i TQ_UINT16 TQDns::Server::priority - \i TQ_UINT16 TQDns::Server::weight - \i TQ_UINT16 TQDns::Server::port - \endlist - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQValueList<TQDns::Server> list = myDns.servers(); - TQValueList<TQDns::Server>::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode -*/ -TQValueList<TQDns::Server> TQDns::servers() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::servers (%s)", l.ascii() ); -#endif - TQValueList<TQDns::Server> result; - if ( t != Srv ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - Server s( rr->target, rr->priority, rr->weight, rr->port ); - result.append( s ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns a list of host names if the record type is \c Ptr. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQStringList list = myDns.hostNames(); - TQStringList::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode - -*/ -TQStringList TQDns::hostNames() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::hostNames (%s)", l.ascii() ); -#endif - TQStringList result; - if ( t != Ptr ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - TQString str( rr->target ); - result.append( str ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns a list of texts if the record type is \c Txt. - - Note that if you want to iterate over the list, you should iterate - over a copy, e.g. - \code - TQStringList list = myDns.texts(); - TQStringList::Iterator it = list.begin(); - while( it != list.end() ) { - myProcessing( *it ); - ++it; - } - \endcode -*/ -TQStringList TQDns::texts() const -{ -#if defined(TQDNS_DEBUG) - qDebug( "TQDns::texts (%s)", l.ascii() ); -#endif - TQStringList result; - if ( t != Txt ) - return result; - - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( this ); - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain ) { - TQString str( rr->text ); - result.append( str ); - } - cached->next(); - } - delete cached; - return result; -} - - -/*! - Returns the canonical name for this DNS node. (This works - regardless of what recordType() is set to.) - - If the canonical name isn't known, this function returns a null - string. - - The canonical name of a DNS node is its full name, or the full - name of the target of its CNAME. For example, if l.trolltech.com - is a CNAME to lillian.troll.no, and the search path for TQDns is - "trolltech.com", then the canonical name for all of "lillian", - "l", "lillian.troll.no." and "l.trolltech.com" is - "lillian.troll.no.". -*/ - -TQString TQDns::canonicalName() const -{ - // the cname should work regardless of the recordType(), so set the record - // type temporarily to cname when you look at the cache - TQDns *that = (TQDns*) this; // mutable function - RecordType oldType = t; - that->t = Cname; - TQPtrList<TQDnsRR> * cached = TQDnsDomain::cached( that ); - that->t = oldType; - - TQDnsRR * rr; - while( (rr=cached->current()) != 0 ) { - if ( rr->current && !rr->nxdomain && rr->domain ) { - delete cached; - return rr->target; - } - cached->next(); - } - delete cached; - return TQString::null; -} - -#if defined(TQ_DNS_SYNCHRONOUS) -/*! \reimp -*/ -void TQDns::connectNotify( const char *signal ) -{ - if ( d->noEventLoop && qstrcmp(signal,TQT_SIGNAL(resultsReady()) )==0 ) { - doSynchronousLookup(); - } -} -#endif - -#if defined(TQ_OS_WIN32) || defined(TQ_OS_CYGWIN) - -#if defined(TQ_DNS_SYNCHRONOUS) -void TQDns::doSynchronousLookup() -{ - // ### not implemented yet -} -#endif - -// the following typedefs are needed for GetNetworkParams() API call -#ifndef IP_TYPES_INCLUDED -#define MAX_HOSTNAME_LEN 128 -#define MAX_DOMAIN_NAME_LEN 128 -#define MAX_SCOPE_ID_LEN 256 -typedef struct { - char String[4 * 4]; -} IP_ADDRESS_STRING, *PIP_ADDRESS_STRING, IP_MASK_STRING, *PIP_MASK_STRING; -typedef struct _IP_ADDR_STRING { - struct _IP_ADDR_STRING* Next; - IP_ADDRESS_STRING IpAddress; - IP_MASK_STRING IpMask; - DWORD Context; -} IP_ADDR_STRING, *PIP_ADDR_STRING; -typedef struct { - char HostName[MAX_HOSTNAME_LEN + 4] ; - char DomainName[MAX_DOMAIN_NAME_LEN + 4]; - PIP_ADDR_STRING CurrentDnsServer; - IP_ADDR_STRING DnsServerList; - UINT NodeType; - char ScopeId[MAX_SCOPE_ID_LEN + 4]; - UINT EnableRouting; - UINT EnableProxy; - UINT EnableDns; -} FIXED_INFO, *PFIXED_INFO; -#endif -typedef DWORD (WINAPI *GNP)( PFIXED_INFO, PULONG ); - -// ### FIXME: this code is duplicated in qfiledialog.cpp -static TQString getWindowsRegString( HKEY key, const TQString &subKey ) -{ - TQString s; - TQT_WA( { - char buf[1024]; - DWORD bsz = sizeof(buf); - int r = RegQueryValueEx( key, (TCHAR*)subKey.ucs2(), 0, 0, (LPBYTE)buf, &bsz ); - if ( r == ERROR_SUCCESS ) { - s = TQString::fromUcs2( (unsigned short *)buf ); - } else if ( r == ERROR_MORE_DATA ) { - char *ptr = new char[bsz+1]; - r = RegQueryValueEx( key, (TCHAR*)subKey.ucs2(), 0, 0, (LPBYTE)ptr, &bsz ); - if ( r == ERROR_SUCCESS ) - s = ptr; - delete [] ptr; - } - } , { - char buf[512]; - DWORD bsz = sizeof(buf); - int r = RegQueryValueExA( key, subKey.local8Bit(), 0, 0, (LPBYTE)buf, &bsz ); - if ( r == ERROR_SUCCESS ) { - s = buf; - } else if ( r == ERROR_MORE_DATA ) { - char *ptr = new char[bsz+1]; - r = RegQueryValueExA( key, subKey.local8Bit(), 0, 0, (LPBYTE)ptr, &bsz ); - if ( r == ERROR_SUCCESS ) - s = ptr; - delete [] ptr; - } - } ); - return s; -} - -static bool getDnsParamsFromRegistry( const TQString &path, - TQString *domainName, TQString *nameServer, TQString *searchList ) -{ - HKEY k; - int r; - TQT_WA( { - r = RegOpenKeyEx( HKEY_LOCAL_MACHINE, - (TCHAR*)path.ucs2(), - 0, KEY_READ, &k ); - } , { - r = RegOpenKeyExA( HKEY_LOCAL_MACHINE, - path, - 0, KEY_READ, &k ); - } ); - - if ( r == ERROR_SUCCESS ) { - *domainName = getWindowsRegString( k, "DhcpDomain" ); - if ( domainName->isEmpty() ) - *domainName = getWindowsRegString( k, "Domain" ); - - *nameServer = getWindowsRegString( k, "DhcpNameServer" ); - if ( nameServer->isEmpty() ) - *nameServer = getWindowsRegString( k, "NameServer" ); - - *searchList = getWindowsRegString( k, "SearchList" ); - } - RegCloseKey( k ); - return r == ERROR_SUCCESS; -} - -void TQDns::doResInit() -{ - char separator = 0; - - if ( ns ) - delete ns; - ns = new TQPtrList<TQHostAddress>; - ns->setAutoDelete( TRUE ); - domains = new TQStrList( TRUE ); - domains->setAutoDelete( TRUE ); - - TQString domainName, nameServer, searchList; - - bool gotNetworkParams = FALSE; - // try the API call GetNetworkParams() first and use registry lookup only - // as a fallback -#ifdef TQ_OS_TEMP - HINSTANCE hinstLib = LoadLibraryW( L"iphlpapi" ); -#else - HINSTANCE hinstLib = LoadLibraryA( "iphlpapi" ); -#endif - if ( hinstLib != 0 ) { -#ifdef TQ_OS_TEMP - GNP getNetworkParams = (GNP) GetProcAddressW( hinstLib, L"GetNetworkParams" ); -#else - GNP getNetworkParams = (GNP) GetProcAddress( hinstLib, "GetNetworkParams" ); -#endif - if ( getNetworkParams != 0 ) { - ULONG l = 0; - DWORD res; - res = getNetworkParams( 0, &l ); - if ( res == ERROR_BUFFER_OVERFLOW ) { - FIXED_INFO *finfo = (FIXED_INFO*)new char[l]; - res = getNetworkParams( finfo, &l ); - if ( res == ERROR_SUCCESS ) { - domainName = finfo->DomainName; - nameServer = ""; - IP_ADDR_STRING *dnsServer = &finfo->DnsServerList; - while ( dnsServer != 0 ) { - nameServer += dnsServer->IpAddress.String; - dnsServer = dnsServer->Next; - if ( dnsServer != 0 ) - nameServer += " "; - } - searchList = ""; - separator = ' '; - gotNetworkParams = TRUE; - } - delete[] finfo; - } - } - FreeLibrary( hinstLib ); - } - if ( !gotNetworkParams ) { - if ( getDnsParamsFromRegistry( - TQString( "System\\CurrentControlSet\\Services\\Tcpip\\Parameters" ), - &domainName, &nameServer, &searchList )) { - // for NT - separator = ' '; - } else if ( getDnsParamsFromRegistry( - TQString( "System\\CurrentControlSet\\Services\\VxD\\MSTCP" ), - &domainName, &nameServer, &searchList )) { - // for 95/98 - separator = ','; - } else { - // Could not access the TCP/IP parameters - domainName = ""; - nameServer = "127.0.0.1"; - searchList = ""; - separator = ' '; - } - } - - nameServer = nameServer.simplifyWhiteSpace(); - int first, last; - if ( !nameServer.isEmpty() ) { - first = 0; - do { - last = nameServer.find( separator, first ); - if ( last < 0 ) - last = nameServer.length(); - TQDns tmp( nameServer.mid( first, last-first ), TQDns::A ); - TQValueList<TQHostAddress> address = tmp.addresses(); - TQ_LONG i = address.count(); - while( i ) - ns->append( new TQHostAddress(address[--i]) ); - first = last+1; - } while( first < (int)nameServer.length() ); - } - - searchList = searchList + " " + domainName; - searchList = searchList.simplifyWhiteSpace().lower(); - first = 0; - do { - last = searchList.find( separator, first ); - if ( last < 0 ) - last = searchList.length(); - domains->append( qstrdup( searchList.mid( first, last-first ) ) ); - first = last+1; - } while( first < (int)searchList.length() ); -} - -#elif defined(TQ_OS_UNIX) - -#if defined(TQ_DNS_SYNCHRONOUS) -void TQDns::doSynchronousLookup() -{ - if ( t!=None && !l.isEmpty() ) { - TQValueListIterator<TQString> it = n.begin(); - TQValueListIterator<TQString> end = n.end(); - int type; - switch( t ) { - case TQDns::A: - type = 1; - break; - case TQDns::Aaaa: - type = 28; - break; - case TQDns::Mx: - type = 15; - break; - case TQDns::Srv: - type = 33; - break; - case TQDns::Cname: - type = 5; - break; - case TQDns::Ptr: - type = 12; - break; - case TQDns::Txt: - type = 16; - break; - default: - type = (char)255; // any - break; - } - while( it != end ) { - TQString s = *it; - it++; - TQByteArray ba( 512 ); - int len = res_search( s.latin1(), 1, type, (uchar*)ba.data(), ba.size() ); - if ( len > 0 ) { - ba.resize( len ); - - TQDnsQuery * query = new TQDnsQuery; - query->started = now(); - query->id = ++::id; - query->t = t; - query->l = s; - TQDnsAnswer a( ba, query ); - a.parse(); - } else if ( len == -1 ) { - // res_search error - } - } - emit resultsReady(); - } -} -#endif - -#if defined(__GLIBC__) && ((__GLIBC__ > 2) || ((__GLIBC__ == 2) && (__GLIBC_MINOR__ >= 3))) -#define TQ_MODERN_RES_API -#else -#endif - -void TQDns::doResInit() -{ - if ( ns ) - return; - ns = new TQPtrList<TQHostAddress>; - ns->setAutoDelete( TRUE ); - domains = new TQStrList( TRUE ); - domains->setAutoDelete( TRUE ); - - // read resolv.conf manually. - TQFile resolvConf("/etc/resolv.conf"); - if (resolvConf.open(IO_ReadOnly)) { - TQTextStream stream( &resolvConf ); - TQString line; - - while ( !stream.atEnd() ) { - line = stream.readLine(); - TQStringList list = TQStringList::split( " ", line ); - if ( line.startsWith( "#" ) || list.count() < 2 ) - continue; - const TQString type = list[0].lower(); - - if ( type == "nameserver" ) { - TQHostAddress *address = new TQHostAddress(); - if ( address->setAddress( TQString(list[1]) ) ) { - // only add ipv6 addresses from resolv.conf if - // this host supports ipv6. - if ( address->isIPv4Address() || ipv6support ) - ns->append( address ); - else - delete address; - } else { - delete address; - } - } else if ( type == "search" ) { - TQStringList srch = TQStringList::split( " ", list[1] ); - for ( TQStringList::Iterator i = srch.begin(); i != srch.end(); ++i ) - domains->append( (*i).lower() ); - - } else if ( type == "domain" ) { - domains->append( list[1].lower() ); - } - } - } - - if (ns->isEmpty()) { -#if defined(TQ_MODERN_RES_API) - struct __res_state res; - res_ninit( &res ); - int i; - // find the name servers to use - for( i=0; i < MAXNS && i < res.nscount; i++ ) - ns->append( new TQHostAddress( ntohl( res.nsaddr_list[i].sin_addr.s_addr ) ) ); -# if defined(MAXDFLSRCH) - for( i=0; i < MAXDFLSRCH; i++ ) { - if ( res.dnsrch[i] && *(res.dnsrch[i]) ) - domains->append( TQString::tqfromLatin1( res.dnsrch[i] ).lower() ); - else - break; - } -# endif - if ( *res.defdname ) - domains->append( TQString::tqfromLatin1( res.defdname ).lower() ); -#else - qdns_res_init(); - int i; - // find the name servers to use - for( i=0; i < MAXNS && i < _res.nscount; i++ ) - ns->append( new TQHostAddress( ntohl( _res.nsaddr_list[i].sin_addr.s_addr ) ) ); -# if defined(MAXDFLSRCH) - for( i=0; i < MAXDFLSRCH; i++ ) { - if ( _res.dnsrch[i] && *(_res.dnsrch[i]) ) - domains->append( TQString::tqfromLatin1( _res.dnsrch[i] ).lower() ); - else - break; - } -# endif - if ( *_res.defdname ) - domains->append( TQString::tqfromLatin1( _res.defdname ).lower() ); -#endif - - // the code above adds "0.0.0.0" as a name server at the slightest - // hint of trouble. so remove those again. - ns->first(); - while( ns->current() ) { - if ( ns->current()->isNull() ) - delete ns->take(); - else - ns->next(); - } - } - - TQFile hosts( TQString::tqfromLatin1( "/etc/hosts" ) ); - if ( hosts.open( IO_ReadOnly ) ) { - // read the /etc/hosts file, creating long-life A and PTR RRs - // for the things we find. - TQTextStream i( &hosts ); - TQString line; - while( !i.atEnd() ) { - line = i.readLine().simplifyWhiteSpace().lower(); - uint n = 0; - while( n < line.length() && line[(int)n] != '#' ) - n++; - line.truncate( n ); - n = 0; - while( n < line.length() && !line[(int)n].isSpace() ) - n++; - TQString ip = line.left( n ); - TQHostAddress a; - a.setAddress( ip ); - if ( ( a.isIPv4Address() || a.isIPv6Address() ) && !a.isNull() ) { - bool first = TRUE; - line = line.mid( n+1 ); - n = 0; - while( n < line.length() && !line[(int)n].isSpace() ) - n++; - TQString hostname = line.left( n ); - // ### in case of bad syntax, hostname is invalid. do we care? - if ( n ) { - TQDnsRR * rr = new TQDnsRR( hostname ); - if ( a.isIPv4Address() ) - rr->t = TQDns::A; - else - rr->t = TQDns::Aaaa; - rr->address = a; - rr->deleteTime = UINT_MAX; - rr->expireTime = UINT_MAX; - rr->current = TRUE; - if ( first ) { - first = FALSE; - TQDnsRR * ptr = new TQDnsRR( TQDns::toInAddrArpaDomain( a ) ); - ptr->t = TQDns::Ptr; - ptr->target = hostname; - ptr->deleteTime = UINT_MAX; - ptr->expireTime = UINT_MAX; - ptr->current = TRUE; - } - } - } - } - } -} - -#endif - -#endif // TQT_NO_DNS - -#endif // USE_QT4
\ No newline at end of file diff --git a/tqtinterface/qt4/src/network/tqdns.h b/tqtinterface/qt4/src/network/tqdns.h deleted file mode 100644 index 23c84ee..0000000 --- a/tqtinterface/qt4/src/network/tqdns.h +++ /dev/null @@ -1,382 +0,0 @@ -#include "tqtglobaldefines.h" - -#ifdef USE_QT4 - -/**************************************************************************** -** -** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). -** All rights reserved. -** Contact: Nokia Corporation ([email protected]) -** -** This file is part of the Qt3Support module of the Qt Toolkit. -** -** $QT_BEGIN_LICENSE:LGPL$ -** Commercial Usage -** Licensees holding valid Qt Commercial licenses may use this file in -** accordance with the Qt Commercial License Agreement provided with the -** Software or, alternatively, in accordance with the terms contained in -** a written agreement between you and Nokia. -** -** GNU Lesser General Public License Usage -** Alternatively, this file may be used under the terms of the GNU Lesser -** General Public License version 2.1 as published by the Free Software -** Foundation and appearing in the file LICENSE.LGPL included in the -** packaging of this file. Please review the following information to -** ensure the GNU Lesser General Public License version 2.1 requirements -** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. -** -** In addition, as a special exception, Nokia gives you certain additional -** rights. These rights are described in the Nokia Qt LGPL Exception -** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. -** -** GNU General Public License Usage -** Alternatively, this file may be used under the terms of the GNU -** General Public License version 3.0 as published by the Free Software -** Foundation and appearing in the file LICENSE.GPL included in the -** packaging of this file. Please review the following information to -** ensure the GNU General Public License version 3.0 requirements will be -** met: http://www.gnu.org/copyleft/gpl.html. -** -** If you have questions regarding the use of this file, please contact -** Nokia at [email protected]. -** $QT_END_LICENSE$ -** -****************************************************************************/ - -#ifndef TQDNS_H -#define TQDNS_H - -#include <Qt/qobject.h> -#include <Qt/qhostaddress.h> -#include <Qt/qsocketnotifier.h> -#include <tqstringlist.h> -#include <tqvaluelist.h> - -#include "tqtimer.h" -#include "tqobject.h" -#include "tqhostaddress.h" -#include "tqsocketnotifier.h" - -#define TQLatin1String TQString -#define TQLatin1Char TQChar - -typedef qint8 Q_INT8; -typedef quint8 Q_UINT8; -typedef qint16 Q_INT16; -typedef quint16 Q_UINT16; -typedef qint32 Q_INT32; -typedef quint32 Q_UINT32; -typedef qint64 Q_INT64; -typedef quint64 Q_UINT64; -typedef qint64 Q_LLONG; -typedef quint64 Q_ULLONG; -#if defined(Q_OS_WIN64) -typedef __int64 Q_LONG; /* word up to 64 bit signed */ -typedef unsigned __int64 Q_ULONG; /* word up to 64 bit unsigned */ -#else -typedef long Q_LONG; /* word up to 64 bit signed */ -typedef unsigned long Q_ULONG; /* word up to 64 bit unsigned */ -#endif - -QT_BEGIN_HEADER - -QT_BEGIN_NAMESPACE - -QT_MODULE(Qt3Support) - -#ifndef QT_NO_DNS - -//#define Q_DNS_SYNCHRONOUS - -class TQDnsPrivate; - -class Q_COMPAT_EXPORT TQDns: public TQObject { - Q_OBJECT - TQ_OBJECT -public: - enum RecordType { - None, - A, Aaaa, - Mx, Srv, - Cname, - Ptr, - Txt - }; - - TQDns(); - TQDns( const TQString & label, RecordType rr = A ); - TQDns( const TQHostAddress & address, RecordType rr = Ptr ); - virtual ~TQDns(); - - // to set/change the query - virtual void setLabel( const TQString & label ); - virtual void setLabel( const TQHostAddress & address ); - TQString label() const { return l; } - - virtual void setRecordType( RecordType rr = A ); - RecordType recordType() const { return t; } - - // whether something is happening behind the scenes - bool isWorking() const; - - // to query for replies - TQValueList<TQHostAddress> addresses() const; - - class Q_COMPAT_EXPORT MailServer { - public: - MailServer( const TQString & n=TQString(), Q_UINT16 p=0 ) - :name(n), priority(p) {} - TQString name; - Q_UINT16 priority; - Q_DUMMY_COMPARISON_OPERATOR(MailServer) - }; - TQValueList<MailServer> mailServers() const; - - class Q_COMPAT_EXPORT Server { - public: - Server(const TQString & n=TQString(), Q_UINT16 p=0, Q_UINT16 w=0, Q_UINT16 po=0 ) - : name(n), priority(p), weight(w), port(po) {} - TQString name; - Q_UINT16 priority; - Q_UINT16 weight; - Q_UINT16 port; - Q_DUMMY_COMPARISON_OPERATOR(Server) - }; - TQValueList<Server> servers() const; - - TQStringList hostNames() const; - - TQStringList texts() const; - - TQString canonicalName() const; // ### real-world but uncommon: TQStringList - - TQStringList qualifiedNames() const { return n; } - -#if defined(Q_DNS_SYNCHRONOUS) -protected: - void connectNotify( const char *signal ); -#endif - -Q_SIGNALS: - void resultsReady(); - -private Q_SLOTS: - void startQuery(); - -private: - static void doResInit(); - void setStartQueryTimer(); - static TQString toInAddrArpaDomain( const TQHostAddress &address ); -#if defined(Q_DNS_SYNCHRONOUS) - void doSynchronousLookup(); -#endif - - TQString l; - TQStringList n; - RecordType t; - TQDnsPrivate * d; - - friend class TQDnsAnswer; - friend class TQDnsManager; -}; - - -// TQDnsSocket are sockets that are used for DNS lookup - -class TQDnsSocket: public TQObject { - Q_OBJECT - // note: Private not public. This class contains NO public API. -protected: - TQDnsSocket( TQT_BASE_OBJECT_NAME *, const char * ); - virtual ~TQDnsSocket(); - -private Q_SLOTS: - virtual void cleanCache(); - virtual void retransmit(); - virtual void answer(); -}; - -#endif // QT_NO_DNS - -QT_END_NAMESPACE - -QT_END_HEADER - -#endif // TQDNS_H - -#else // USE_QT4 - -/**************************************************************************** -** -** Definition of TQDns class. -** -** Created : 991122 -** -** Copyright (C) 1999-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQDNS_H -#define TQDNS_H - -#ifndef TQT_H -#include "tqobject.h" -#include "tqhostaddress.h" -#include "tqsocketnotifier.h" -#include "tqstringlist.h" -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_DNS -#else -#define TQM_EXPORT_DNS TQ_EXPORT -#endif - -#ifndef TQT_NO_DNS - -//#define TQ_DNS_SYNCHRONOUS - -class TQDnsPrivate; - -class TQM_EXPORT_DNS TQDns: public TQObject { - Q_OBJECT - TQ_OBJECT -public: - enum RecordType { - None, - A, Aaaa, - Mx, Srv, - Cname, - Ptr, - Txt - }; - - TQDns(); - TQDns( const TQString & label, RecordType rr = A ); - TQDns( const TQHostAddress & address, RecordType rr = Ptr ); - virtual ~TQDns(); - - // to set/change the query - virtual void setLabel( const TQString & label ); - virtual void setLabel( const TQHostAddress & address ); - TQString label() const { return l; } - - virtual void setRecordType( RecordType rr = A ); - RecordType recordType() const { return t; } - - // whether something is happening behind the scenes - bool isWorking() const; - - // to query for replies - TQValueList<TQHostAddress> addresses() const; - - class TQM_EXPORT_DNS MailServer { - public: - MailServer( const TQString & n=TQString::null, TQ_UINT16 p=0 ) - :name(n), priority(p) {} - TQString name; - TQ_UINT16 priority; - TQ_DUMMY_COMPARISON_OPERATOR(MailServer) - }; - TQValueList<MailServer> mailServers() const; - - class TQM_EXPORT_DNS Server { - public: - Server(const TQString & n=TQString::null, TQ_UINT16 p=0, TQ_UINT16 w=0, TQ_UINT16 po=0 ) - : name(n), priority(p), weight(w), port(po) {} - TQString name; - TQ_UINT16 priority; - TQ_UINT16 weight; - TQ_UINT16 port; - TQ_DUMMY_COMPARISON_OPERATOR(Server) - }; - TQValueList<Server> servers() const; - - TQStringList hostNames() const; - - TQStringList texts() const; - - TQString canonicalName() const; // ### real-world but uncommon: TQStringList - - TQStringList qualifiedNames() const { return n; } - -#if defined(TQ_DNS_SYNCHRONOUS) -protected: - void connectNotify( const char *signal ); -#endif - -Q_SIGNALS: - void resultsReady(); - -private Q_SLOTS: - void startQuery(); - -private: - static void doResInit(); - void setStartQueryTimer(); - static TQString toInAddrArpaDomain( const TQHostAddress &address ); -#if defined(TQ_DNS_SYNCHRONOUS) - void doSynchronousLookup(); -#endif - - TQString l; - TQStringList n; - RecordType t; - TQDnsPrivate * d; - - friend class TQDnsAnswer; - friend class TQDnsManager; -}; - - -// TQDnsSocket are sockets that are used for DNS lookup - -class TQDnsSocket: public TQObject { - Q_OBJECT - TQ_OBJECT - // note: Private not public. This class contains NO public API. -protected: - TQDnsSocket( TQObject *, const char * ); - virtual ~TQDnsSocket(); - -private Q_SLOTS: - virtual void cleanCache(); - virtual void retransmit(); - virtual void answer(); -}; - -#endif // TQT_NO_DNS - -#endif // TQDNS_H - -#endif // USE_QT4
\ No newline at end of file diff --git a/tqtinterface/qt4/src/network/tqftp.cpp b/tqtinterface/qt4/src/network/tqftp.cpp deleted file mode 100644 index 78f1ce8..0000000 --- a/tqtinterface/qt4/src/network/tqftp.cpp +++ /dev/null @@ -1,2421 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQFtp class. -** -** Created : 970521 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqftp.h" - -#ifndef TQT_NO_NETWORKPROTOCOL_FTP - -#include "tqsocket.h" -#include "tqsocketdevice.h" -#include "tqurlinfo.h" -#include "tqurloperator.h" -#include "tqstringlist.h" -#include "tqregexp.h" -#include "tqtimer.h" -#include "tqfileinfo.h" -#include "tqptrdict.h" // binary compatibility - -#ifndef TQT_NO_TEXTCODEC -#include "tqtextcodec.h" -#endif - -//#define TQFTPPI_DEBUG -//#define TQFTPDTP_DEBUG - - -class TQFtpPI; - -class TQFtpDTP : public TQObject -{ - Q_OBJECT - TQ_OBJECT - -public: - enum ConnectState { - CsHostFound, - CsConnected, - CsClosed, - CsHostNotFound, - CsConnectionRefused - }; - - TQFtpDTP( TQFtpPI *p, TQObject *parent=0, const char *name=0 ); - - void setData( TQByteArray * ); - void setDevice( TQIODevice * ); - void writeData(); - - void setBytesTotal( int bytes ) - { - bytesTotal = bytes; - bytesDone = 0; - emit dataTransferProgress( bytesDone, bytesTotal ); - } - - bool hasError() const; - TQString errorMessage() const; - void clearError(); - - void connectToHost( const TQString & host, TQ_UINT16 port ) - { socket.connectToHost( host, port ); } - - TQSocket::State socketState() const - { return socket.state(); } - - TQ_ULONG bytesAvailable() const - { return socket.bytesAvailable(); } - - TQ_LONG readBlock( char *data, TQ_ULONG maxlen ) - { - TQ_LONG read = socket.readBlock( data, maxlen ); - bytesDone += read; - return read; - } - - TQByteArray readAll() - { - TQByteArray tmp = TQT_TQBYTEARRAY_OBJECT(socket.readAll()); - bytesDone += tmp.size(); - return tmp; - } - - void abortConnection(); - - static bool parseDir( const TQString &buffer, const TQString &userName, TQUrlInfo *info ); - -Q_SIGNALS: - void listInfo( const TQUrlInfo& ); - void readyRead(); - void dataTransferProgress( int, int ); - - void connectState( int ); - -private Q_SLOTS: - void socketConnected(); - void socketReadyRead(); - void socketError( int ); - void socketConnectionClosed(); - void socketBytesWritten( int ); - -private: - void clearData() - { - is_ba = FALSE; - data.dev = 0; - } - - TQSocket socket; - TQFtpPI *pi; - TQString err; - int bytesDone; - int bytesTotal; - bool callWriteData; - - // If is_ba is TRUE, ba is used; ba is never 0. - // Otherwise dev is used; dev can be 0 or not. - union { - TQByteArray *ba; - TQIODevice *dev; - } data; - bool is_ba; -}; - -class TQFtpPI : public TQObject -{ - Q_OBJECT - TQ_OBJECT - -public: - TQFtpPI( TQObject *parent = 0 ); - - void connectToHost( const TQString &host, TQ_UINT16 port ); - - bool sendCommands( const TQStringList &cmds ); - bool sendCommand( const TQString &cmd ) - { return sendCommands( TQStringList( cmd ) ); } - - void clearPendingCommands(); - void abort(); - - TQString currentCommand() const - { return currentCmd; } - - bool rawCommand; - - TQFtpDTP dtp; // the PI has a DTP which is not the design of RFC 959, but it - // makes the design simpler this way -Q_SIGNALS: - void connectState( int ); - void finished( const TQString& ); - void error( int, const TQString& ); - void rawFtpReply( int, const TQString& ); - -private Q_SLOTS: - void hostFound(); - void connected(); - void connectionClosed(); - void delayedCloseFinished(); - void readyRead(); - void error( int ); - - void dtpConnectState( int ); - -private: - // the states are modelled after the generalized state diagram of RFC 959, - // page 58 - enum State { - Begin, - Idle, - Waiting, - Success, - Failure - }; - - enum AbortState { - None, - AbortStarted, - WaitForAbortToFinish - }; - - bool processReply(); - bool startNextCmd(); - - TQSocket commandSocket; - TQString replyText; - char replyCode[3]; - State state; - AbortState abortState; - TQStringList pendingCommands; - TQString currentCmd; - - bool waitForDtpToConnect; - bool waitForDtpToClose; -}; - -/********************************************************************** - * - * TQFtpCommand implemenatation - * - *********************************************************************/ -class TQFtpCommand -{ -public: - TQFtpCommand( TQFtp::Command cmd, TQStringList raw ); - TQFtpCommand( TQFtp::Command cmd, TQStringList raw, const TQByteArray &ba ); - TQFtpCommand( TQFtp::Command cmd, TQStringList raw, TQIODevice *dev ); - ~TQFtpCommand(); - - int id; - TQFtp::Command command; - TQStringList rawCmds; - - // If is_ba is TRUE, ba is used; ba is never 0. - // Otherwise dev is used; dev can be 0 or not. - union { - TQByteArray *ba; - TQIODevice *dev; - } data; - bool is_ba; - - static int idCounter; -}; - -int TQFtpCommand::idCounter = 0; - -TQFtpCommand::TQFtpCommand( TQFtp::Command cmd, TQStringList raw ) - : command(cmd), rawCmds(raw), is_ba(FALSE) -{ - id = ++idCounter; - data.dev = 0; -} - -TQFtpCommand::TQFtpCommand( TQFtp::Command cmd, TQStringList raw, const TQByteArray &ba ) - : command(cmd), rawCmds(raw), is_ba(TRUE) -{ - id = ++idCounter; - data.ba = new TQByteArray( ba ); -} - -TQFtpCommand::TQFtpCommand( TQFtp::Command cmd, TQStringList raw, TQIODevice *dev ) - : command(cmd), rawCmds(raw), is_ba(FALSE) -{ - id = ++idCounter; - data.dev = dev; -} - -TQFtpCommand::~TQFtpCommand() -{ - if ( is_ba ) - delete data.ba; -} - -/********************************************************************** - * - * TQFtpDTP implemenatation - * - *********************************************************************/ -TQFtpDTP::TQFtpDTP( TQFtpPI *p, TQObject *parent, const char *name ) : - TQObject( parent, name ), - socket( 0, "TQFtpDTP_socket" ), - pi( p ), - callWriteData( FALSE ) -{ - clearData(); - - connect( &socket, TQT_SIGNAL( connected() ), - TQT_SLOT( socketConnected() ) ); - connect( &socket, TQT_SIGNAL( readyRead() ), - TQT_SLOT( socketReadyRead() ) ); - connect( &socket, TQT_SIGNAL( error(int) ), - TQT_SLOT( socketError(int) ) ); - connect( &socket, TQT_SIGNAL( connectionClosed() ), - TQT_SLOT( socketConnectionClosed() ) ); - connect( &socket, TQT_SIGNAL( bytesWritten(int) ), - TQT_SLOT( socketBytesWritten(int) ) ); -} - -void TQFtpDTP::setData( TQByteArray *ba ) -{ - is_ba = TRUE; - data.ba = ba; -} - -void TQFtpDTP::setDevice( TQIODevice *dev ) -{ - is_ba = FALSE; - data.dev = dev; -} - -void TQFtpDTP::writeData() -{ - if ( is_ba ) { -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::writeData: write %d bytes", data.ba->size() ); -#endif - if ( data.ba->size() == 0 ) - emit dataTransferProgress( 0, bytesTotal ); - else - socket.writeBlock( data.ba->data(), data.ba->size() ); - socket.close(); - clearData(); - } else if ( data.dev ) { - callWriteData = FALSE; - const int blockSize = 16*1024; - char buf[blockSize]; - while ( !data.dev->atEnd() && socket.bytesToWrite()==0 ) { - TQ_LONG read = data.dev->readBlock( buf, blockSize ); -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::writeData: writeBlock() of size %d bytes", (int)read ); -#endif - socket.writeBlock( buf, read ); - if ( !data.dev ) - return; // this can happen when a command is aborted - } - if ( data.dev->atEnd() ) { - if ( bytesDone==0 && socket.bytesToWrite()==0 ) - emit dataTransferProgress( 0, bytesTotal ); - socket.close(); - clearData(); - } else { - callWriteData = TRUE; - } - } -} - -inline bool TQFtpDTP::hasError() const -{ - return !err.isNull(); -} - -inline TQString TQFtpDTP::errorMessage() const -{ - return err; -} - -inline void TQFtpDTP::clearError() -{ - err = TQString::null; -} - -void TQFtpDTP::abortConnection() -{ -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::abortConnection" ); -#endif - callWriteData = FALSE; - clearData(); - - socket.clearPendingData(); - socket.close(); -} - -bool TQFtpDTP::parseDir( const TQString &buffer, const TQString &userName, TQUrlInfo *info ) -{ - TQStringList lst = TQStringList::split( " ", buffer ); - - if ( lst.count() < 9 ) - return FALSE; - - TQString tmp; - - // permissions - tmp = lst[ 0 ]; - - if ( tmp[ 0 ] == TQChar( 'd' ) ) { - info->setDir( TRUE ); - info->setFile( FALSE ); - info->setSymLink( FALSE ); - } else if ( tmp[ 0 ] == TQChar( '-' ) ) { - info->setDir( FALSE ); - info->setFile( TRUE ); - info->setSymLink( FALSE ); - } else if ( tmp[ 0 ] == TQChar( 'l' ) ) { - info->setDir( TRUE ); // #### todo - info->setFile( FALSE ); - info->setSymLink( TRUE ); - } else { - return FALSE; - } - - static int user = 0; - static int group = 1; - static int other = 2; - static int readable = 0; - static int writable = 1; - static int executable = 2; - - bool perms[ 3 ][ 3 ]; - perms[0][0] = (tmp[ 1 ] == 'r'); - perms[0][1] = (tmp[ 2 ] == 'w'); - perms[0][2] = (tmp[ 3 ] == 'x'); - perms[1][0] = (tmp[ 4 ] == 'r'); - perms[1][1] = (tmp[ 5 ] == 'w'); - perms[1][2] = (tmp[ 6 ] == 'x'); - perms[2][0] = (tmp[ 7 ] == 'r'); - perms[2][1] = (tmp[ 8 ] == 'w'); - perms[2][2] = (tmp[ 9 ] == 'x'); - - // owner - tmp = lst[ 2 ]; - info->setOwner( tmp ); - - // group - tmp = lst[ 3 ]; - info->setGroup( tmp ); - - // ### not correct - info->setWritable( ( userName == info->owner() && perms[ user ][ writable ] ) || - perms[ other ][ writable ] ); - info->setReadable( ( userName == info->owner() && perms[ user ][ readable ] ) || - perms[ other ][ readable ] ); - - int p = 0; - if ( perms[ user ][ readable ] ) - p |= TQUrlInfo::ReadOwner; - if ( perms[ user ][ writable ] ) - p |= TQUrlInfo::WriteOwner; - if ( perms[ user ][ executable ] ) - p |= TQUrlInfo::ExeOwner; - if ( perms[ group ][ readable ] ) - p |= TQUrlInfo::ReadGroup; - if ( perms[ group ][ writable ] ) - p |= TQUrlInfo::WriteGroup; - if ( perms[ group ][ executable ] ) - p |= TQUrlInfo::ExeGroup; - if ( perms[ other ][ readable ] ) - p |= TQUrlInfo::ReadOther; - if ( perms[ other ][ writable ] ) - p |= TQUrlInfo::WriteOther; - if ( perms[ other ][ executable ] ) - p |= TQUrlInfo::ExeOther; - info->setPermissions( p ); - - // size - tmp = lst[ 4 ]; - info->setSize( tmp.toInt() ); - - // date and time - TQTime time; - TQString dateStr; - dateStr += "Sun "; - lst[ 5 ][ 0 ] = lst[ 5 ][ 0 ].upper(); - dateStr += lst[ 5 ]; - dateStr += ' '; - dateStr += lst[ 6 ]; - dateStr += ' '; - - if ( lst[ 7 ].contains( ":" ) ) { - time = TQTime( lst[ 7 ].left( 2 ).toInt(), lst[ 7 ].right( 2 ).toInt() ); - dateStr += TQString::number( TQDate::currentDate().year() ); - } else { - dateStr += lst[ 7 ]; - } - - TQDate date = TQT_TQDATE_OBJECT(TQDate::fromString( dateStr )); - info->setLastModified( TQDateTime( date, time ) ); - - if ( lst[ 7 ].contains( ":" ) ) { - const int futureTolerance = 600; - if( info->lastModified().secsTo( TQDateTime::tqcurrentDateTime() ) < -futureTolerance ) { - TQDateTime dt = info->lastModified(); - TQDate d = TQT_TQDATE_OBJECT(dt.date()); - d.setYMD(d.year()-1, d.month(), d.day()); - dt.setDate(d); - info->setLastModified(dt); - } - } - - // name - if ( info->isSymLink() ) - info->setName( lst[ 8 ].stripWhiteSpace() ); - else { - TQString n; - for ( uint i = 8; i < lst.count(); ++i ) - n += lst[ i ] + " "; - n = n.stripWhiteSpace(); - info->setName( n ); - } - return TRUE; -} - -void TQFtpDTP::socketConnected() -{ -#if !defined (TQ_WS_TQWS) - // Use a large send buffer to reduce the number - // of writeBlocks when download and uploading files. - // The actual size used here (128k) is default on most - // Unixes. - socket.socketDevice()->setSendBufferSize(128 * 1024); - socket.socketDevice()->setReceiveBufferSize(128 * 1024); -#endif - - bytesDone = 0; -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::connectState( CsConnected )" ); -#endif - emit connectState( TQFtpDTP::CsConnected ); -} - -void TQFtpDTP::socketReadyRead() -{ - if ( pi->currentCommand().isEmpty() ) { - socket.close(); -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::connectState( CsClosed )" ); -#endif - emit connectState( TQFtpDTP::CsClosed ); - return; - } - - if ( pi->currentCommand().startsWith("LIST") ) { - while ( socket.canReadLine() ) { - TQUrlInfo i; - TQString line = socket.readLine(); -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP read (list): '%s'", line.latin1() ); -#endif - if ( parseDir( line, "", &i ) ) { - emit listInfo( i ); - } else { - // some FTP servers don't return a 550 if the file or directory - // does not exist, but rather write a text to the data socket - // -- try to catch these cases - if ( line.endsWith( "No such file or directory\r\n" ) ) - err = line; - } - } - } else { - if ( !is_ba && data.dev ) { - TQByteArray ba( socket.bytesAvailable() ); - TQ_LONG bytesRead = socket.readBlock( ba.data(), ba.size() ); - if ( bytesRead < 0 ) { - // ### error handling - return; - } - ba.resize( bytesRead ); - bytesDone += bytesRead; -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP read: %d bytes (total %d bytes)", (int)bytesRead, bytesDone ); -#endif - emit dataTransferProgress( bytesDone, bytesTotal ); - if (data.dev) // make sure it wasn't deleted in the slot - data.dev->writeBlock( ba ); - } else { -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP readyRead: %d bytes available (total %d bytes read)", (int)bytesAvailable(), bytesDone ); -#endif - emit dataTransferProgress( bytesDone+socket.bytesAvailable(), bytesTotal ); - emit readyRead(); - } - } -} - -void TQFtpDTP::socketError( int e ) -{ - if ( e == TQSocket::ErrHostNotFound ) { -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::connectState( CsHostNotFound )" ); -#endif - emit connectState( TQFtpDTP::CsHostNotFound ); - } else if ( e == TQSocket::ErrConnectionRefused ) { -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::connectState( CsConnectionRefused )" ); -#endif - emit connectState( TQFtpDTP::CsConnectionRefused ); - } -} - -void TQFtpDTP::socketConnectionClosed() -{ - if ( !is_ba && data.dev ) { - clearData(); - } -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::connectState( CsClosed )" ); -#endif - emit connectState( TQFtpDTP::CsClosed ); -} - -void TQFtpDTP::socketBytesWritten( int bytes ) -{ - bytesDone += bytes; -#if defined(TQFTPDTP_DEBUG) - qDebug( "TQFtpDTP::bytesWritten( %d )", bytesDone ); -#endif - emit dataTransferProgress( bytesDone, bytesTotal ); - if ( callWriteData ) - writeData(); -} - -/********************************************************************** - * - * TQFtpPI implemenatation - * - *********************************************************************/ -TQFtpPI::TQFtpPI( TQObject *parent ) : - TQObject( parent ), - rawCommand(FALSE), - dtp( this ), - commandSocket( 0, "TQFtpPI_socket" ), - state( Begin ), abortState( None ), - currentCmd( TQString::null ), - waitForDtpToConnect( FALSE ), - waitForDtpToClose( FALSE ) -{ - connect( &commandSocket, TQT_SIGNAL(hostFound()), - TQT_SLOT(hostFound()) ); - connect( &commandSocket, TQT_SIGNAL(connected()), - TQT_SLOT(connected()) ); - connect( &commandSocket, TQT_SIGNAL(connectionClosed()), - TQT_SLOT(connectionClosed()) ); - connect( &commandSocket, TQT_SIGNAL(delayedCloseFinished()), - TQT_SLOT(delayedCloseFinished()) ); - connect( &commandSocket, TQT_SIGNAL(readyRead()), - TQT_SLOT(readyRead()) ); - connect( &commandSocket, TQT_SIGNAL(error(int)), - TQT_SLOT(error(int)) ); - - connect( &dtp, TQT_SIGNAL(connectState(int)), - TQT_SLOT(dtpConnectState(int)) ); -} - -void TQFtpPI::connectToHost( const TQString &host, TQ_UINT16 port ) -{ - emit connectState( TQFtp::HostLookup ); - commandSocket.connectToHost( host, port ); -} - -/* - Sends the sequence of commands \a cmds to the FTP server. When the commands - are all done the finished() signal is emitted. When an error occurs, the - error() signal is emitted. - - If there are pending commands in the queue this functions returns FALSE and - the \a cmds are not added to the queue; otherwise it returns TRUE. -*/ -bool TQFtpPI::sendCommands( const TQStringList &cmds ) -{ - if ( !pendingCommands.isEmpty() ) - return FALSE; - - if ( commandSocket.state()!=TQSocket::Connected || state!=Idle ) { - emit error( TQFtp::NotConnected, TQFtp::tr( "Not connected" ) ); - return TRUE; // there are no pending commands - } - - pendingCommands = cmds; - startNextCmd(); - return TRUE; -} - -void TQFtpPI::clearPendingCommands() -{ - pendingCommands.clear(); - dtp.abortConnection(); - currentCmd = TQString::null; - state = Idle; -} - -void TQFtpPI::abort() -{ - pendingCommands.clear(); - - if ( abortState != None ) - // ABOR already sent - return; - - abortState = AbortStarted; -#if defined(TQFTPPI_DEBUG) - qDebug( "TQFtpPI send: ABOR" ); -#endif - commandSocket.writeBlock( "ABOR\r\n", 6 ); - - if ( currentCmd.startsWith("STOR ") ) - dtp.abortConnection(); -} - -void TQFtpPI::hostFound() -{ - emit connectState( TQFtp::Connecting ); -} - -void TQFtpPI::connected() -{ - state = Begin; -#if defined(TQFTPPI_DEBUG) -// qDebug( "TQFtpPI state: %d [connected()]", state ); -#endif - emit connectState( TQFtp::Connected ); -} - -void TQFtpPI::connectionClosed() -{ - commandSocket.close(); - emit connectState( TQFtp::Unconnected ); -} - -void TQFtpPI::delayedCloseFinished() -{ - emit connectState( TQFtp::Unconnected ); -} - -void TQFtpPI::error( int e ) -{ - if ( e == TQSocket::ErrHostNotFound ) { - emit connectState( TQFtp::Unconnected ); - emit error( TQFtp::HostNotFound, - TQFtp::tr( "Host %1 not found" ).arg( commandSocket.peerName() ) ); - } else if ( e == TQSocket::ErrConnectionRefused ) { - emit connectState( TQFtp::Unconnected ); - emit error( TQFtp::ConnectionRefused, - TQFtp::tr( "Connection refused to host %1" ).arg( commandSocket.peerName() ) ); - } -} - -void TQFtpPI::readyRead() -{ - if ( waitForDtpToClose ) - return; - - while ( commandSocket.canReadLine() ) { - // read line with respect to line continuation - TQString line = commandSocket.readLine(); - if ( replyText.isEmpty() ) { - if ( line.length() < 3 ) { - // ### protocol error - return; - } - const int lowerLimit[3] = {1,0,0}; - const int upperLimit[3] = {5,5,9}; - for ( int i=0; i<3; i++ ) { - replyCode[i] = line[i].digitValue(); - if ( replyCode[i]<lowerLimit[i] || replyCode[i]>upperLimit[i] ) { - // ### protocol error - return; - } - } - } - TQString endOfMultiLine; - endOfMultiLine[0] = '0' + replyCode[0]; - endOfMultiLine[1] = '0' + replyCode[1]; - endOfMultiLine[2] = '0' + replyCode[2]; - endOfMultiLine[3] = ' '; - TQString lineCont( endOfMultiLine ); - lineCont[3] = '-'; - TQString lineLeft4 = line.left(4); - - while ( lineLeft4 != endOfMultiLine ) { - if ( lineLeft4 == lineCont ) - replyText += line.mid( 4 ); // strip 'xyz-' - else - replyText += line; - if ( !commandSocket.canReadLine() ) - return; - line = commandSocket.readLine(); - lineLeft4 = line.left(4); - } - replyText += line.mid( 4 ); // strip reply code 'xyz ' - if ( replyText.endsWith("\r\n") ) - replyText.truncate( replyText.length()-2 ); - - if ( processReply() ) - replyText = ""; - } -} - -/* - Process a reply from the FTP server. - - Returns TRUE if the reply was processed or FALSE if the reply has to be - processed at a later point. -*/ -bool TQFtpPI::processReply() -{ -#if defined(TQFTPPI_DEBUG) -// qDebug( "TQFtpPI state: %d [processReply() begin]", state ); - if ( replyText.length() < 400 ) - qDebug( "TQFtpPI recv: %d %s", 100*replyCode[0]+10*replyCode[1]+replyCode[2], replyText.latin1() ); - else - qDebug( "TQFtpPI recv: %d (text skipped)", 100*replyCode[0]+10*replyCode[1]+replyCode[2] ); -#endif - - // process 226 replies ("Closing Data Connection") only when the data - // connection is really closed to avoid short reads of the DTP - if ( 100*replyCode[0]+10*replyCode[1]+replyCode[2] == 226 ) { - if ( dtp.socketState() != TQSocket::Idle ) { - waitForDtpToClose = TRUE; - return FALSE; - } - } - - switch ( abortState ) { - case AbortStarted: - abortState = WaitForAbortToFinish; - break; - case WaitForAbortToFinish: - abortState = None; - return TRUE; - default: - break; - } - - // get new state - static const State table[5] = { - /* 1yz 2yz 3yz 4yz 5yz */ - Waiting, Success, Idle, Failure, Failure - }; - switch ( state ) { - case Begin: - if ( replyCode[0] == 1 ) { - return TRUE; - } else if ( replyCode[0] == 2 ) { - state = Idle; - emit finished( TQFtp::tr( "Connected to host %1" ).arg( commandSocket.peerName() ) ); - break; - } - // ### error handling - return TRUE; - case Waiting: - if ( replyCode[0]<0 || replyCode[0]>5 ) - state = Failure; - else - state = table[ replyCode[0] - 1 ]; - break; - default: - // ### spontaneous message - return TRUE; - } -#if defined(TQFTPPI_DEBUG) -// qDebug( "TQFtpPI state: %d [processReply() intermediate]", state ); -#endif - - // special actions on certain replies - int replyCodeInt = 100*replyCode[0] + 10*replyCode[1] + replyCode[2]; - emit rawFtpReply( replyCodeInt, replyText ); - if ( rawCommand ) { - rawCommand = FALSE; - } else if ( replyCodeInt == 227 ) { - // 227 Entering Passive Mode (h1,h2,h3,h4,p1,p2) - // rfc959 does not define this response precisely, and gives - // both examples where the parenthesis are used, and where - // they are missing. We need to scan for the address and host - // info. - TQRegExp addrPortPattern("(\\d+),(\\d+),(\\d+),(\\d+),(\\d+),(\\d+)"); - if (addrPortPattern.search(replyText) == -1) { -#if defined(TQFTPPI_DEBUG) - qDebug( "TQFtp: bad 227 response -- address and port information missing" ); -#endif - // ### error handling - } else { - QStringList lst = addrPortPattern.capturedTexts(); - TQString host = lst[1] + "." + lst[2] + "." + lst[3] + "." + lst[4]; - TQ_UINT16 port = ( lst[5].toUInt() << 8 ) + lst[6].toUInt(); - waitForDtpToConnect = TRUE; - dtp.connectToHost( host, port ); - } - } else if ( replyCodeInt == 230 ) { - if ( currentCmd.startsWith("USER ") && pendingCommands.count()>0 && - pendingCommands.first().startsWith("PASS ") ) { - // no need to send the PASS -- we are already logged in - pendingCommands.pop_front(); - } - // 230 User logged in, proceed. - emit connectState( TQFtp::LoggedIn ); - } else if ( replyCodeInt == 213 ) { - // 213 File status. - if ( currentCmd.startsWith("SIZE ") ) - dtp.setBytesTotal( replyText.simplifyWhiteSpace().toInt() ); - } else if ( replyCode[0]==1 && currentCmd.startsWith("STOR ") ) { - dtp.writeData(); - } - - // react on new state - switch ( state ) { - case Begin: - // ### should never happen - break; - case Success: - // ### success handling - state = Idle; - // no break! - case Idle: - if ( dtp.hasError() ) { - emit error( TQFtp::UnknownError, dtp.errorMessage() ); - dtp.clearError(); - } - startNextCmd(); - break; - case Waiting: - // ### do nothing - break; - case Failure: - emit error( TQFtp::UnknownError, replyText ); - state = Idle; - startNextCmd(); - break; - } -#if defined(TQFTPPI_DEBUG) -// qDebug( "TQFtpPI state: %d [processReply() end]", state ); -#endif - return TRUE; -} - -#ifndef TQT_NO_TEXTCODEC -TQM_EXPORT_FTP TQTextCodec *qt_ftp_filename_codec = 0; -#endif - -/* - Starts next pending command. Returns FALSE if there are no pending commands, - otherwise it returns TRUE. -*/ -bool TQFtpPI::startNextCmd() -{ - if ( waitForDtpToConnect ) - // don't process any new commands until we are connected - return TRUE; - -#if defined(TQFTPPI_DEBUG) - if ( state != Idle ) - qDebug( "TQFtpPI startNextCmd: Internal error! TQFtpPI called in non-Idle state %d", state ); -#endif - if ( pendingCommands.isEmpty() ) { - currentCmd = TQString::null; - emit finished( replyText ); - return FALSE; - } - currentCmd = pendingCommands.first(); - pendingCommands.pop_front(); -#if defined(TQFTPPI_DEBUG) - qDebug( "TQFtpPI send: %s", currentCmd.left( currentCmd.length()-2 ).latin1() ); -#endif - state = Waiting; -#ifndef TQT_NO_TEXTCODEC - if ( qt_ftp_filename_codec ) { - int len = 0; - TQCString enc = qt_ftp_filename_codec->fromUnicode(currentCmd,len); - commandSocket.writeBlock( enc.data(), len ); - } else -#endif - { - commandSocket.writeBlock( currentCmd.latin1(), currentCmd.length() ); - } - return TRUE; -} - -void TQFtpPI::dtpConnectState( int s ) -{ - switch ( s ) { - case TQFtpDTP::CsClosed: - if ( waitForDtpToClose ) { - // there is an unprocessed reply - if ( processReply() ) - replyText = ""; - else - return; - } - waitForDtpToClose = FALSE; - readyRead(); - return; - case TQFtpDTP::CsConnected: - waitForDtpToConnect = FALSE; - startNextCmd(); - return; - case TQFtpDTP::CsHostNotFound: - case TQFtpDTP::CsConnectionRefused: - emit error( TQFtp::ConnectionRefused, - TQFtp::tr( "Connection refused for data connection" ) ); - startNextCmd(); - return; - default: - return; - } -} - -/********************************************************************** - * - * TQFtpPrivate - * - *********************************************************************/ -class TQFtpPrivate -{ -public: - TQFtpPrivate() : - close_waitForStateChange(FALSE), - state( TQFtp::Unconnected ), - error( TQFtp::NoError ), - npWaitForLoginDone( FALSE ) - { pending.setAutoDelete( TRUE ); } - - TQFtpPI pi; - TQPtrList<TQFtpCommand> pending; - bool close_waitForStateChange; - TQFtp::State state; - TQFtp::Error error; - TQString errorString; - - bool npWaitForLoginDone; -}; - -static TQPtrDict<TQFtpPrivate> *d_ptr = 0; -static void cleanup_d_ptr() -{ - delete d_ptr; - d_ptr = 0; -} -static TQFtpPrivate* d( const TQFtp* foo ) -{ - if ( !d_ptr ) { - d_ptr = new TQPtrDict<TQFtpPrivate>; - d_ptr->setAutoDelete( TRUE ); - qAddPostRoutine( cleanup_d_ptr ); - } - TQFtpPrivate* ret = d_ptr->find( (void*)foo ); - if ( ! ret ) { - ret = new TQFtpPrivate; - d_ptr->replace( (void*) foo, ret ); - } - return ret; -} - -static void delete_d( const TQFtp* foo ) -{ - if ( d_ptr ) - d_ptr->remove( (void*) foo ); -} - -/********************************************************************** - * - * TQFtp implementation - * - *********************************************************************/ -/*! - \class TQFtp tqftp.h - \brief The TQFtp class provides an implementation of the FTP protocol. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class provides two different interfaces: one is the - TQNetworkProtocol interface that allows you to use FTP through the - TQUrlOperator abstraction. The other is a direct interface to FTP - that gives you lower-level access to the FTP protocol for finer - control. Using the direct interface you can also execute arbitrary - FTP commands. - - Don't mix the two interfaces, since the behavior is not - well-defined. - - If you want to use TQFtp with the TQNetworkProtocol interface, you - do not use it directly, but rather through a TQUrlOperator, for - example: - - \code - TQUrlOperator op( "ftp://ftp.trolltech.com" ); - op.listChildren(); // Asks the server to provide a directory listing - \endcode - - This code will only work if the TQFtp class is registered; to - register the class, you must call qInitNetworkProtocols() before - using a TQUrlOperator with TQFtp. - - The rest of this descrption describes the direct interface to FTP. - - The class works asynchronously, so there are no blocking - functions. If an operation cannot be executed immediately, the - function will still return straight away and the operation will be - scheduled for later execution. The results of scheduled operations - are reported via Q_SIGNALS. This approach depends on the event loop - being in operation. - - The operations that can be scheduled (they are called "commands" - in the rest of the documentation) are the following: - connectToHost(), login(), close(), list(), cd(), get(), put(), - remove(), mkdir(), rmdir(), rename() and rawCommand(). - - All of these commands return a unique identifier that allows you - to keep track of the command that is currently being executed. - When the execution of a command starts, the commandStarted() - signal with the command's identifier is emitted. When the command - is finished, the commandFinished() signal is emitted with the - command's identifier and a bool that indicates whether the command - finished with an error. - - In some cases, you might want to execute a sequence of commands, - e.g. if you want to connect and login to a FTP server. This is - simply achieved: - - \code - TQFtp *ftp = new TQFtp( this ); // this is an optional TQObject parent - ftp->connectToHost( "ftp.trolltech.com" ); - ftp->login(); - \endcode - - In this case two FTP commands have been scheduled. When the last - scheduled command has finished, a done() signal is emitted with - a bool argument that tells you whether the sequence finished with - an error. - - If an error occurs during the execution of one of the commands in - a sequence of commands, all the pending commands (i.e. scheduled, - but not yet executed commands) are cleared and no Q_SIGNALS are - emitted for them. - - Some commands, e.g. list(), emit additional Q_SIGNALS to report - their results. - - Example: If you want to download the INSTALL file from Trolltech's - FTP server, you would write this: - - \code - ftp->connectToHost( "ftp.trolltech.com" ); // id == 1 - ftp->login(); // id == 2 - ftp->cd( "qt" ); // id == 3 - ftp->get( "INSTALL" ); // id == 4 - ftp->close(); // id == 5 - \endcode - - For this example the following sequence of Q_SIGNALS is emitted - (with small variations, depending on network traffic, etc.): - - \code - start( 1 ) - stateChanged( HostLookup ) - stateChanged( Connecting ) - stateChanged( Connected ) - finished( 1, FALSE ) - - start( 2 ) - stateChanged( LoggedIn ) - finished( 2, FALSE ) - - start( 3 ) - finished( 3, FALSE ) - - start( 4 ) - dataTransferProgress( 0, 3798 ) - dataTransferProgress( 2896, 3798 ) - readyRead() - dataTransferProgress( 3798, 3798 ) - readyRead() - finished( 4, FALSE ) - - start( 5 ) - stateChanged( Closing ) - stateChanged( Unconnected ) - finished( 5, FALSE ) - - done( FALSE ) - \endcode - - The dataTransferProgress() signal in the above example is useful - if you want to show a \link TQProgressBar progressbar \endlink to - inform the user about the progress of the download. The - readyRead() signal tells you that there is data ready to be read. - The amount of data can be queried then with the bytesAvailable() - function and it can be read with the readBlock() or readAll() - function. - - If the login fails for the above example, the Q_SIGNALS would look - like this: - - \code - start( 1 ) - stateChanged( HostLookup ) - stateChanged( Connecting ) - stateChanged( Connected ) - finished( 1, FALSE ) - - start( 2 ) - finished( 2, TRUE ) - - done( TRUE ) - \endcode - - You can then get details about the error with the error() and - errorString() functions. - - The functions currentId() and currentCommand() provide more - information about the currently executing command. - - The functions hasPendingCommands() and clearPendingCommands() - allow you to query and clear the list of pending commands. - - The safest and easiest way to use the FTP protocol is to use - TQUrlOperator() or the FTP commands described above. If you are an - experienced network programmer and want to have complete control - you can use rawCommand() to execute arbitrary FTP commands. - - \sa \link network.html TQt Network Documentation \endlink TQNetworkProtocol, TQUrlOperator TQHttp -*/ - -/*! - Constructs a TQFtp object. -*/ -TQFtp::TQFtp() : TQNetworkProtocol() -{ - init(); -} - -/*! - Constructs a TQFtp object. The \a parent and \a name parameters - are passed to the TQObject constructor. -*/ -TQFtp::TQFtp( TQObject *parent, const char *name ) : TQNetworkProtocol() -{ - if ( parent ) - parent->insertChild( this ); - TQT_TQOBJECT(this)->setName( name ); - init(); -} - -void TQFtp::init() -{ - TQFtpPrivate *d = ::d( this ); - d->errorString = tr( "Unknown error" ); - - connect( &d->pi, TQT_SIGNAL(connectState(int)), - TQT_SLOT(piConnectState(int)) ); - connect( &d->pi, TQT_SIGNAL(finished(const TQString&)), - TQT_SLOT(piFinished(const TQString&)) ); - connect( &d->pi, TQT_SIGNAL(error(int,const TQString&)), - TQT_SLOT(piError(int,const TQString&)) ); - connect( &d->pi, TQT_SIGNAL(rawFtpReply(int,const TQString&)), - TQT_SLOT(piFtpReply(int,const TQString&)) ); - - connect( &d->pi.dtp, TQT_SIGNAL(readyRead()), - TQT_SIGNAL(readyRead()) ); - connect( &d->pi.dtp, TQT_SIGNAL(dataTransferProgress(int,int)), - TQT_SIGNAL(dataTransferProgress(int,int)) ); - connect( &d->pi.dtp, TQT_SIGNAL(listInfo(const TQUrlInfo&)), - TQT_SIGNAL(listInfo(const TQUrlInfo&)) ); -} - -/*! - \enum TQFtp::State - - This enum defines the connection state: - - \value Unconnected There is no connection to the host. - \value HostLookup A host name lookup is in progress. - \value Connecting An attempt to connect to the host is in progress. - \value Connected Connection to the host has been achieved. - \value LoggedIn Connection and user login have been achieved. - \value Closing The connection is closing down, but it is not yet - closed. (The state will be \c Unconnected when the connection is - closed.) - - \sa stateChanged() state() -*/ -/*! - \enum TQFtp::Error - - This enum identifies the error that occurred. - - \value NoError No error occurred. - \value HostNotFound The host name lookup failed. - \value ConnectionRefused The server refused the connection. - \value NotConnected Tried to send a command, but there is no connection to - a server. - \value UnknownError An error other than those specified above - occurred. - - \sa error() -*/ - -/*! - \enum TQFtp::Command - - This enum is used as the return value for the currentCommand() function. - This allows you to perform specific actions for particular - commands, e.g. in a FTP client, you might want to clear the - directory view when a list() command is started; in this case you - can simply check in the slot connected to the start() signal if - the currentCommand() is \c List. - - \value None No command is being executed. - \value ConnectToHost connectToHost() is being executed. - \value Login login() is being executed. - \value Close close() is being executed. - \value List list() is being executed. - \value Cd cd() is being executed. - \value Get get() is being executed. - \value Put put() is being executed. - \value Remove remove() is being executed. - \value Mkdir mkdir() is being executed. - \value Rmdir rmdir() is being executed. - \value Rename rename() is being executed. - \value RawCommand rawCommand() is being executed. - - \sa currentCommand() -*/ - -/*! - \fn void TQFtp::stateChanged( int state ) - - This signal is emitted when the state of the connection changes. - The argument \a state is the new state of the connection; it is - one of the \l State values. - - It is usually emitted in response to a connectToHost() or close() - command, but it can also be emitted "spontaneously", e.g. when the - server closes the connection unexpectedly. - - \sa connectToHost() close() state() State -*/ - -/*! - \fn void TQFtp::listInfo( const TQUrlInfo &i ); - - This signal is emitted for each directory entry the list() command - finds. The details of the entry are stored in \a i. - - \sa list() -*/ - -/*! - \fn void TQFtp::commandStarted( int id ) - - This signal is emitted when processing the command identified by - \a id starts. - - \sa commandFinished() done() -*/ - -/*! - \fn void TQFtp::commandFinished( int id, bool error ) - - This signal is emitted when processing the command identified by - \a id has finished. \a error is TRUE if an error occurred during - the processing; otherwise \a error is FALSE. - - \sa commandStarted() done() error() errorString() -*/ - -/*! - \fn void TQFtp::done( bool error ) - - This signal is emitted when the last pending command has finished; - (it is emitted after the last command's commandFinished() signal). - \a error is TRUE if an error occurred during the processing; - otherwise \a error is FALSE. - - \sa commandFinished() error() errorString() -*/ - -/*! - \fn void TQFtp::readyRead() - - This signal is emitted in response to a get() command when there - is new data to read. - - If you specify a tqdevice as the second argument in the get() - command, this signal is \e not emitted; instead the data is - written directly to the tqdevice. - - You can read the data with the readAll() or readBlock() functions. - - This signal is useful if you want to process the data in chunks as - soon as it becomes available. If you are only interested in the - complete data, just connect to the commandFinished() signal and - read the data then instead. - - \sa get() readBlock() readAll() bytesAvailable() -*/ - -/*! - \fn void TQFtp::dataTransferProgress( int done, int total ) - - This signal is emitted in response to a get() or put() request to - indicate the current progress of the download or upload. - - \a done is the amount of data that has already been transferred - and \a total is the total amount of data to be read or written. It - is possible that the TQFtp class is not able to determine the total - amount of data that should be transferred, in which case \a total - is 0. (If you connect this signal to a TQProgressBar, the progress - bar shows a busy indicator if the total is 0). - - \warning \a done and \a total are not necessarily the size in - bytes, since for large files these values might need to be - "scaled" to avoid overflow. - - \sa get() put() TQProgressBar::setProgress() -*/ - -/*! - \fn void TQFtp::rawCommandReply( int replyCode, const TQString &detail ); - - This signal is emitted in response to the rawCommand() function. - \a replyCode is the 3 digit reply code and \a detail is the text - that follows the reply code. - - \sa rawCommand() -*/ - -/*! - Connects to the FTP server \a host using port \a port. - - The stateChanged() signal is emitted when the state of the - connecting process changes, e.g. to \c HostLookup, then \c - Connecting, then \c Connected. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa stateChanged() commandStarted() commandFinished() -*/ -int TQFtp::connectToHost( const TQString &host, TQ_UINT16 port ) -{ - TQStringList cmds; - cmds << host; - cmds << TQString::number( (uint)port ); - return addCommand( new TQFtpCommand( ConnectToHost, cmds ) ); -} - -/*! - Logs in to the FTP server with the username \a user and the - password \a password. - - The stateChanged() signal is emitted when the state of the - connecting process changes, e.g. to \c LoggedIn. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::login( const TQString &user, const TQString &password ) -{ - TQStringList cmds; - cmds << ( TQString("USER ") + ( user.isNull() ? TQString("anonymous") : user ) + "\r\n" ); - cmds << ( TQString("PASS ") + ( password.isNull() ? TQString("anonymous@") : password ) + "\r\n" ); - return addCommand( new TQFtpCommand( Login, cmds ) ); -} - -/*! - Closes the connection to the FTP server. - - The stateChanged() signal is emitted when the state of the - connecting process changes, e.g. to \c Closing, then \c - Unconnected. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa stateChanged() commandStarted() commandFinished() -*/ -int TQFtp::close() -{ - return addCommand( new TQFtpCommand( Close, TQStringList("TQUIT\r\n") ) ); -} - -/*! - Lists the contents of directory \a dir on the FTP server. If \a - dir is empty, it lists the contents of the current directory. - - The listInfo() signal is emitted for each directory entry found. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa listInfo() commandStarted() commandFinished() -*/ -int TQFtp::list( const TQString &dir ) -{ - TQStringList cmds; - cmds << "TYPE A\r\n"; - cmds << "PASV\r\n"; - if ( dir.isEmpty() ) - cmds << "LIST\r\n"; - else - cmds << ( "LIST " + dir + "\r\n" ); - return addCommand( new TQFtpCommand( List, cmds ) ); -} - -/*! - Changes the working directory of the server to \a dir. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::cd( const TQString &dir ) -{ - return addCommand( new TQFtpCommand( Cd, TQStringList("CWD "+dir+"\r\n") ) ); -} - -/*! - Downloads the file \a file from the server. - - If \a dev is 0, then the readyRead() signal is emitted when there - is data available to read. You can then read the data with the - readBlock() or readAll() functions. - - If \a dev is not 0, the data is written directly to the tqdevice \a - dev. Make sure that the \a dev pointer is valid for the duration - of the operation (it is safe to delete it when the - commandFinished() signal is emitted). In this case the readyRead() - signal is \e not emitted and you cannot read data with the - readBlock() or readAll() functions. - - If you don't read the data immediately it becomes available, i.e. - when the readyRead() signal is emitted, it is still available - until the next command is started. - - For example, if you want to present the data to the user as soon - as there is something available, connect to the readyRead() signal - and read the data immediately. On the other hand, if you only want - to work with the complete data, you can connect to the - commandFinished() signal and read the data when the get() command - is finished. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa readyRead() dataTransferProgress() commandStarted() - commandFinished() -*/ -int TQFtp::get( const TQString &file, TQIODevice *dev ) -{ - TQStringList cmds; - cmds << ( "SIZE " + file + "\r\n" ); - cmds << "TYPE I\r\n"; - cmds << "PASV\r\n"; - cmds << ( "RETR " + file + "\r\n" ); - if ( dev ) - return addCommand( new TQFtpCommand( Get, cmds, dev ) ); - return addCommand( new TQFtpCommand( Get, cmds ) ); -} - -/*! - \overload - - Writes the data \a data to the file called \a file on the server. - The progress of the upload is reported by the - dataTransferProgress() signal. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa dataTransferProgress() commandStarted() commandFinished() -*/ -int TQFtp::put( const TQByteArray &data, const TQString &file ) -{ - TQStringList cmds; - cmds << "TYPE I\r\n"; - cmds << "PASV\r\n"; - cmds << ( "ALLO " + TQString::number(data.size()) + "\r\n" ); - cmds << ( "STOR " + file + "\r\n" ); - return addCommand( new TQFtpCommand( Put, cmds, data ) ); -} - -/*! - Reads the data from the IO tqdevice \a dev, and writes it to the - file called \a file on the server. The data is read in chunks from - the IO tqdevice, so this overload allows you to transmit large - amounts of data without the need to read all the data into memory - at once. - - Make sure that the \a dev pointer is valid for the duration of the - operation (it is safe to delete it when the commandFinished() is - emitted). -*/ -int TQFtp::put( TQIODevice *dev, const TQString &file ) -{ - TQStringList cmds; - cmds << "TYPE I\r\n"; - cmds << "PASV\r\n"; - if ( !dev->isSequentialAccess() ) - cmds << ( "ALLO " + TQString::number(dev->size()) + "\r\n" ); - cmds << ( "STOR " + file + "\r\n" ); - return addCommand( new TQFtpCommand( Put, cmds, dev ) ); -} - -/*! - Deletes the file called \a file from the server. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::remove( const TQString &file ) -{ - return addCommand( new TQFtpCommand( Remove, TQStringList("DELE "+file+"\r\n") ) ); -} - -/*! - Creates a directory called \a dir on the server. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::mkdir( const TQString &dir ) -{ - return addCommand( new TQFtpCommand( Mkdir, TQStringList("MKD "+dir+"\r\n") ) ); -} - -/*! - Removes the directory called \a dir from the server. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::rmdir( const TQString &dir ) -{ - return addCommand( new TQFtpCommand( Rmdir, TQStringList("RMD "+dir+"\r\n") ) ); -} - -/*! - Renames the file called \a oldname to \a newname on the server. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa commandStarted() commandFinished() -*/ -int TQFtp::rename( const TQString &oldname, const TQString &newname ) -{ - TQStringList cmds; - cmds << ( "RNFR " + oldname + "\r\n" ); - cmds << ( "RNTO " + newname + "\r\n" ); - return addCommand( new TQFtpCommand( Rename, cmds ) ); -} - -/*! - Sends the raw FTP command \a command to the FTP server. This is - useful for low-level FTP access. If the operation you wish to - perform has an equivalent TQFtp function, we recommend using the - function instead of raw FTP commands since the functions are - easier and safer. - - The function does not block and returns immediately. The command - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - commandStarted() and commandFinished(). - - When the command is started the commandStarted() signal is - emitted. When it is finished the commandFinished() signal is - emitted. - - \sa rawCommandReply() commandStarted() commandFinished() -*/ -int TQFtp::rawCommand( const TQString &command ) -{ - TQString cmd = command.stripWhiteSpace() + "\r\n"; - return addCommand( new TQFtpCommand( RawCommand, TQStringList(cmd) ) ); -} - -/*! - Returns the number of bytes that can be read from the data socket - at the moment. - - \sa get() readyRead() readBlock() readAll() -*/ -TQ_ULONG TQFtp::bytesAvailable() const -{ - TQFtpPrivate *d = ::d( this ); - return d->pi.dtp.bytesAvailable(); -} - -/*! - Reads \a maxlen bytes from the data socket into \a data and - returns the number of bytes read. Returns -1 if an error occurred. - - \sa get() readyRead() bytesAvailable() readAll() -*/ -TQ_LONG TQFtp::readBlock( char *data, TQ_ULONG maxlen ) -{ - TQFtpPrivate *d = ::d( this ); - return d->pi.dtp.readBlock( data, maxlen ); -} - -/*! - Reads all the bytes available from the data socket and returns - them. - - \sa get() readyRead() bytesAvailable() readBlock() -*/ -TQByteArray TQFtp::readAll() -{ - TQFtpPrivate *d = ::d( this ); - return d->pi.dtp.readAll(); -} - -/*! - Aborts the current command and deletes all scheduled commands. - - If there is an unfinished command (i.e. a command for which the - commandStarted() signal has been emitted, but for which the - commandFinished() signal has not been emitted), this function - sends an \c ABORT command to the server. When the server replies - that the command is aborted, the commandFinished() signal with the - \c error argument set to \c TRUE is emitted for the command. Due - to timing issues, it is possible that the command had already - finished before the abort request reached the server, in which - case, the commandFinished() signal is emitted with the \c error - argument set to \c FALSE. - - For all other commands that are affected by the abort(), no - Q_SIGNALS are emitted. - - If you don't start further FTP commands directly after the - abort(), there won't be any scheduled commands and the done() - signal is emitted. - - \warning Some FTP servers, for example the BSD FTP daemon (version - 0.3), wrongly return a positive reply even when an abort has - occurred. For these servers the commandFinished() signal has its - error flag set to \c FALSE, even though the command did not - complete successfully. - - \sa clearPendingCommands() -*/ -void TQFtp::abort() -{ - TQFtpPrivate *d = ::d( this ); - if ( d->pending.isEmpty() ) - return; - - clearPendingCommands(); - d->pi.abort(); -} - -/*! - Returns the identifier of the FTP command that is being executed - or 0 if there is no command being executed. - - \sa currentCommand() -*/ -int TQFtp::currentId() const -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = d->pending.getFirst(); - if ( c == 0 ) - return 0; - return c->id; -} - -/*! - Returns the command type of the FTP command being executed or \c - None if there is no command being executed. - - \sa currentId() -*/ -TQFtp::Command TQFtp::currentCommand() const -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = d->pending.getFirst(); - if ( c == 0 ) - return None; - return c->command; -} - -/*! - Returns the TQIODevice pointer that is used by the FTP command to read data - from or store data to. If there is no current FTP command being executed or - if the command does not use an IO tqdevice, this function returns 0. - - This function can be used to delete the TQIODevice in the slot connected to - the commandFinished() signal. - - \sa get() put() -*/ -TQIODevice* TQFtp::currentDevice() const -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = d->pending.getFirst(); - if ( !c ) - return 0; - if ( c->is_ba ) - return 0; - return c->data.dev; -} - -/*! - Returns TRUE if there are any commands scheduled that have not yet - been executed; otherwise returns FALSE. - - The command that is being executed is \e not considered as a - scheduled command. - - \sa clearPendingCommands() currentId() currentCommand() -*/ -bool TQFtp::hasPendingCommands() const -{ - TQFtpPrivate *d = ::d( this ); - return d->pending.count() > 1; -} - -/*! - Deletes all pending commands from the list of scheduled commands. - This does not affect the command that is being executed. If you - want to stop this this as well, use abort(). - - \sa hasPendingCommands() abort() -*/ -void TQFtp::clearPendingCommands() -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = 0; - if ( d->pending.count() > 0 ) - c = d->pending.take( 0 ); - d->pending.clear(); - if ( c ) - d->pending.append( c ); -} - -/*! - Returns the current state of the object. When the state changes, - the stateChanged() signal is emitted. - - \sa State stateChanged() -*/ -TQFtp::State TQFtp::state() const -{ - TQFtpPrivate *d = ::d( this ); - return d->state; -} - -/*! - Returns the last error that occurred. This is useful to find out - what when wrong when receiving a commandFinished() or a done() - signal with the \c error argument set to \c TRUE. - - If you start a new command, the error status is reset to \c NoError. -*/ -TQFtp::Error TQFtp::error() const -{ - TQFtpPrivate *d = ::d( this ); - return d->error; -} - -/*! - Returns a human-readable description of the last error that - occurred. This is useful for presenting a error message to the - user when receiving a commandFinished() or a done() signal with - the \c error argument set to \c TRUE. - - The error string is often (but not always) the reply from the - server, so it is not always possible to translate the string. If - the message comes from TQt, the string has already passed through - tr(). -*/ -TQString TQFtp::errorString() const -{ - TQFtpPrivate *d = ::d( this ); - return d->errorString; -} - -int TQFtp::addCommand( TQFtpCommand *cmd ) -{ - TQFtpPrivate *d = ::d( this ); - d->pending.append( cmd ); - - if ( d->pending.count() == 1 ) - // don't emit the commandStarted() signal before the id is returned - TQTimer::singleShot( 0, this, TQT_SLOT(startNextCommand()) ); - - return cmd->id; -} - -void TQFtp::startNextCommand() -{ - TQFtpPrivate *d = ::d( this ); - - TQFtpCommand *c = d->pending.getFirst(); - if ( c == 0 ) - return; - - d->error = NoError; - d->errorString = tr( "Unknown error" ); - - if ( bytesAvailable() ) - readAll(); // clear the data - emit commandStarted( c->id ); - - if ( c->command == ConnectToHost ) { - d->pi.connectToHost( c->rawCmds[0], c->rawCmds[1].toUInt() ); - } else { - if ( c->command == Put ) { - if ( c->is_ba ) { - d->pi.dtp.setData( c->data.ba ); - d->pi.dtp.setBytesTotal( c->data.ba->size() ); - } else if ( c->data.dev && (c->data.dev->isOpen() || c->data.dev->open(IO_ReadOnly) ) ) { - d->pi.dtp.setDevice( c->data.dev ); - if ( c->data.dev->isSequentialAccess() ) - d->pi.dtp.setBytesTotal( 0 ); - else - d->pi.dtp.setBytesTotal( c->data.dev->size() ); - } - } else if ( c->command == Get ) { - if ( !c->is_ba && c->data.dev ) { - d->pi.dtp.setDevice( c->data.dev ); - } - } else if ( c->command == Close ) { - d->state = TQFtp::Closing; - emit stateChanged( d->state ); - } - if ( !d->pi.sendCommands( c->rawCmds ) ) { - // ### error handling (this case should not happen) - } - } -} - -void TQFtp::piFinished( const TQString& ) -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = d->pending.getFirst(); - if ( c == 0 ) - return; - - if ( c->command == Close ) { - // The order of in which the Q_SLOTS are called is arbitrary, so - // disconnect the SIGNAL-TQT_SIGNAL temporary to make sure that we - // don't get the commandFinished() signal before the stateChanged() - // signal. - if ( d->state != TQFtp::Unconnected ) { - d->close_waitForStateChange = TRUE; - return; - } - } - emit commandFinished( c->id, FALSE ); - - d->pending.removeFirst(); - if ( d->pending.isEmpty() ) { - emit done( FALSE ); - } else { - startNextCommand(); - } -} - -void TQFtp::piError( int errorCode, const TQString &text ) -{ - TQFtpPrivate *d = ::d( this ); - TQFtpCommand *c = d->pending.getFirst(); - - // non-fatal errors - if ( c->command==Get && d->pi.currentCommand().startsWith("SIZE ") ) { - d->pi.dtp.setBytesTotal( -1 ); - return; - } else if ( c->command==Put && d->pi.currentCommand().startsWith("ALLO ") ) { - return; - } - - d->error = (Error)errorCode; - switch ( currentCommand() ) { - case ConnectToHost: - d->errorString = tr( "Connecting to host failed:\n%1" ).arg( text ); - break; - case Login: - d->errorString = tr( "Login failed:\n%1" ).arg( text ); - break; - case List: - d->errorString = tr( "Listing directory failed:\n%1" ).arg( text ); - break; - case Cd: - d->errorString = tr( "Changing directory failed:\n%1" ).arg( text ); - break; - case Get: - d->errorString = tr( "Downloading file failed:\n%1" ).arg( text ); - break; - case Put: - d->errorString = tr( "Uploading file failed:\n%1" ).arg( text ); - break; - case Remove: - d->errorString = tr( "Removing file failed:\n%1" ).arg( text ); - break; - case Mkdir: - d->errorString = tr( "Creating directory failed:\n%1" ).arg( text ); - break; - case Rmdir: - d->errorString = tr( "Removing directory failed:\n%1" ).arg( text ); - break; - default: - d->errorString = text; - break; - } - - d->pi.clearPendingCommands(); - clearPendingCommands(); - emit commandFinished( c->id, TRUE ); - - d->pending.removeFirst(); - if ( d->pending.isEmpty() ) - emit done( TRUE ); - else - startNextCommand(); -} - -void TQFtp::piConnectState( int state ) -{ - TQFtpPrivate *d = ::d( this ); - d->state = (State)state; - emit stateChanged( d->state ); - if ( d->close_waitForStateChange ) { - d->close_waitForStateChange = FALSE; - piFinished( tr( "Connection closed" ) ); - } -} - -void TQFtp::piFtpReply( int code, const TQString &text ) -{ - if ( currentCommand() == RawCommand ) { - TQFtpPrivate *d = ::d( this ); - d->pi.rawCommand = TRUE; - emit rawCommandReply( code, text ); - } -} - -/*! - Destructor. -*/ -TQFtp::~TQFtp() -{ - abort(); - close(); - delete_d( this ); -} - -/********************************************************************** - * - * TQFtp implementation of the TQNetworkProtocol interface - * - *********************************************************************/ -/*! \reimp -*/ -void TQFtp::operationListChildren( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - cd( ( url()->path().isEmpty() ? TQString( "/" ) : url()->path() ) ); - list(); - emit start( op ); -} - -/*! \reimp -*/ -void TQFtp::operationMkDir( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - mkdir( op->arg( 0 ) ); -} - -/*! \reimp -*/ -void TQFtp::operationRemove( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - cd( ( url()->path().isEmpty() ? TQString( "/" ) : url()->path() ) ); - remove( TQUrl( op->arg( 0 ) ).path() ); -} - -/*! \reimp -*/ -void TQFtp::operationRename( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - cd( ( url()->path().isEmpty() ? TQString( "/" ) : url()->path() ) ); - rename( op->arg( 0 ), op->arg( 1 )); -} - -/*! \reimp -*/ -void TQFtp::operationGet( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - TQUrl u( op->arg( 0 ) ); - get( u.path() ); -} - -/*! \reimp -*/ -void TQFtp::operationPut( TQNetworkOperation *op ) -{ - op->setState( StInProgress ); - - TQUrl u( op->arg( 0 ) ); - put( op->rawArg(1), u.path() ); -} - -/*! \reimp -*/ -bool TQFtp::checkConnection( TQNetworkOperation *op ) -{ - TQFtpPrivate *d = ::d( this ); - if ( state() == Unconnected && !d->npWaitForLoginDone ) { - connect( this, TQT_SIGNAL(listInfo(const TQUrlInfo&)), - this, TQT_SLOT(npListInfo(const TQUrlInfo&)) ); - connect( this, TQT_SIGNAL(done(bool)), - this, TQT_SLOT(npDone(bool)) ); - connect( this, TQT_SIGNAL(stateChanged(int)), - this, TQT_SLOT(npStateChanged(int)) ); - connect( this, TQT_SIGNAL(dataTransferProgress(int,int)), - this, TQT_SLOT(npDataTransferProgress(int,int)) ); - connect( this, TQT_SIGNAL(readyRead()), - this, TQT_SLOT(npReadyRead()) ); - - d->npWaitForLoginDone = TRUE; - switch ( op->operation() ) { - case OpGet: - case OpPut: - { - TQUrl u( op->arg( 0 ) ); - connectToHost( u.host(), u.port() != -1 ? u.port() : 21 ); - } - break; - default: - connectToHost( url()->host(), url()->port() != -1 ? url()->port() : 21 ); - break; - } - TQString user = url()->user().isEmpty() ? TQString( "anonymous" ) : url()->user(); - TQString pass = url()->password().isEmpty() ? TQString( "anonymous@" ) : url()->password(); - login( user, pass ); - } - - if ( state() == LoggedIn ) - return TRUE; - return FALSE; -} - -/*! \reimp -*/ -int TQFtp::supportedOperations() const -{ - return OpListChildren | OpMkDir | OpRemove | OpRename | OpGet | OpPut; -} - -/*! \internal - Parses the string, \a buffer, which is one line of a directory - listing which came from the FTP server, and sets the values which - have been parsed to the url info object, \a info. -*/ -void TQFtp::parseDir( const TQString &buffer, TQUrlInfo &info ) -{ - TQFtpDTP::parseDir( buffer, url()->user(), &info ); -} - -void TQFtp::npListInfo( const TQUrlInfo & i ) -{ - if ( url() ) { - TQRegExp filt( url()->nameFilter(), FALSE, TRUE ); - if ( i.isDir() || filt.search( i.name() ) != -1 ) { - emit newChild( i, operationInProgress() ); - } - } else { - emit newChild( i, operationInProgress() ); - } -} - -void TQFtp::npDone( bool err ) -{ - TQFtpPrivate *d = ::d( this ); - - bool emitFinishedSignal = FALSE; - TQNetworkOperation *op = operationInProgress(); - if ( op ) { - if ( err ) { - op->setProtocolDetail( errorString() ); - op->setState( StFailed ); - if ( error() == HostNotFound ) { - op->setErrorCode( (int)ErrHostNotFound ); - } else { - switch ( op->operation() ) { - case OpListChildren: - op->setErrorCode( (int)ErrListChildren ); - break; - case OpMkDir: - op->setErrorCode( (int)ErrMkDir ); - break; - case OpRemove: - op->setErrorCode( (int)ErrRemove ); - break; - case OpRename: - op->setErrorCode( (int)ErrRename ); - break; - case OpGet: - op->setErrorCode( (int)ErrGet ); - break; - case OpPut: - op->setErrorCode( (int)ErrPut ); - break; - } - } - emitFinishedSignal = TRUE; - } else if ( !d->npWaitForLoginDone ) { - switch ( op->operation() ) { - case OpRemove: - emit removed( op ); - break; - case OpMkDir: - { - TQUrlInfo inf( op->arg( 0 ), 0, "", "", 0, TQDateTime(), - TQDateTime(), TRUE, FALSE, FALSE, TRUE, TRUE, TRUE ); - emit newChild( inf, op ); - emit createdDirectory( inf, op ); - } - break; - case OpRename: - emit itemChanged( operationInProgress() ); - break; - default: - break; - } - op->setState( StDone ); - emitFinishedSignal = TRUE; - } - } - d->npWaitForLoginDone = FALSE; - - if ( state() == Unconnected ) { - disconnect( this, TQT_SIGNAL(listInfo(const TQUrlInfo&)), - this, TQT_SLOT(npListInfo(const TQUrlInfo&)) ); - disconnect( this, TQT_SIGNAL(done(bool)), - this, TQT_SLOT(npDone(bool)) ); - disconnect( this, TQT_SIGNAL(stateChanged(int)), - this, TQT_SLOT(npStateChanged(int)) ); - disconnect( this, TQT_SIGNAL(dataTransferProgress(int,int)), - this, TQT_SLOT(npDataTransferProgress(int,int)) ); - disconnect( this, TQT_SIGNAL(readyRead()), - this, TQT_SLOT(npReadyRead()) ); - } - - // emit the finished() signal at the very end to avoid reentrance problems - if ( emitFinishedSignal ) - emit finished( op ); -} - -void TQFtp::npStateChanged( int state ) -{ - if ( url() ) { - if ( state == Connecting ) - emit connectionStateChanged( ConHostFound, tr( "Host %1 found" ).arg( url()->host() ) ); - else if ( state == Connected ) - emit connectionStateChanged( ConConnected, tr( "Connected to host %1" ).arg( url()->host() ) ); - else if ( state == Unconnected ) - emit connectionStateChanged( ConClosed, tr( "Connection to %1 closed" ).arg( url()->host() ) ); - } else { - if ( state == Connecting ) - emit connectionStateChanged( ConHostFound, tr( "Host found" ) ); - else if ( state == Connected ) - emit connectionStateChanged( ConConnected, tr( "Connected to host" ) ); - else if ( state == Unconnected ) - emit connectionStateChanged( ConClosed, tr( "Connection closed" ) ); - } -} - -void TQFtp::npDataTransferProgress( int bDone, int bTotal ) -{ - emit TQNetworkProtocol::dataTransferProgress( bDone, bTotal, operationInProgress() ); -} - -void TQFtp::npReadyRead() -{ - emit data( readAll(), operationInProgress() ); -} - -// ### unused -- delete in TQt 4.0 -/*! \internal -*/ -void TQFtp::hostFound() -{ -} -/*! \internal -*/ -void TQFtp::connected() -{ -} -/*! \internal -*/ -void TQFtp::closed() -{ -} -/*! \internal -*/ -void TQFtp::dataHostFound() -{ -} -/*! \internal -*/ -void TQFtp::dataConnected() -{ -} -/*! \internal -*/ -void TQFtp::dataClosed() -{ -} -/*! \internal -*/ -void TQFtp::dataReadyRead() -{ -} -/*! \internal -*/ -void TQFtp::dataBytesWritten( int ) -{ -} -/*! \internal -*/ -void TQFtp::error( int ) -{ -} - -#include "tqftp.tqmoc" - -#endif // TQT_NO_NETWORKPROTOCOL_FTP diff --git a/tqtinterface/qt4/src/network/tqftp.h b/tqtinterface/qt4/src/network/tqftp.h deleted file mode 100644 index afa7461..0000000 --- a/tqtinterface/qt4/src/network/tqftp.h +++ /dev/null @@ -1,204 +0,0 @@ -/**************************************************************************** -** -** Definition of TQFtp class. -** -** Created : 970521 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQFTP_H -#define TQFTP_H - -#ifndef TQT_H -#include "tqstring.h" // char*->TQString conversion -#include "tqurlinfo.h" -#include "tqcstring.h" -#include "tqnetworkprotocol.h" -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_FTP -#else -#define TQM_EXPORT_FTP TQ_EXPORT -#endif - -#ifndef TQT_NO_NETWORKPROTOCOL_FTP - - -class TQSocket; -class TQFtpCommand; - -class TQM_EXPORT_FTP TQFtp : public TQNetworkProtocol -{ - Q_OBJECT - TQ_OBJECT - -public: - TQFtp(); // ### TQt 4.0: get rid of this overload - TQFtp( TQObject *parent, const char *name=0 ); - virtual ~TQFtp(); - - int supportedOperations() const; - - // non-TQNetworkProtocol functions: - enum State { - Unconnected, - HostLookup, - Connecting, - Connected, - LoggedIn, - Closing - }; - enum Error { - NoError, - UnknownError, - HostNotFound, - ConnectionRefused, - NotConnected - }; - enum Command { - None, - ConnectToHost, - Login, - Close, - List, - Cd, - Get, - Put, - Remove, - Mkdir, - Rmdir, - Rename, - RawCommand - }; - - int connectToHost( const TQString &host, TQ_UINT16 port=21 ); - int login( const TQString &user=TQString::null, const TQString &password=TQString::null ); - int close(); - int list( const TQString &dir=TQString::null ); - int cd( const TQString &dir ); - int get( const TQString &file, TQIODevice *dev=0 ); - int put( const TQByteArray &data, const TQString &file ); - int put( TQIODevice *dev, const TQString &file ); - int remove( const TQString &file ); - int mkdir( const TQString &dir ); - int rmdir( const TQString &dir ); - int rename( const TQString &oldname, const TQString &newname ); - - int rawCommand( const TQString &command ); - - TQ_ULONG bytesAvailable() const; - TQ_LONG readBlock( char *data, TQ_ULONG maxlen ); - TQByteArray readAll(); - - int currentId() const; - TQIODevice* currentDevice() const; - Command currentCommand() const; - bool hasPendingCommands() const; - void clearPendingCommands(); - - State state() const; - - Error error() const; - TQString errorString() const; - -public Q_SLOTS: - void abort(); - -Q_SIGNALS: - void stateChanged( int ); - void listInfo( const TQUrlInfo& ); - void readyRead(); - void dataTransferProgress( int, int ); - void rawCommandReply( int, const TQString& ); - - void commandStarted( int ); - void commandFinished( int, bool ); - void done( bool ); - -protected: - void parseDir( const TQString &buffer, TQUrlInfo &info ); // ### TQt 4.0: delete this? (not public API) - void operationListChildren( TQNetworkOperation *op ); - void operationMkDir( TQNetworkOperation *op ); - void operationRemove( TQNetworkOperation *op ); - void operationRename( TQNetworkOperation *op ); - void operationGet( TQNetworkOperation *op ); - void operationPut( TQNetworkOperation *op ); - - // ### TQt 4.0: delete these - // unused variables: - TQSocket *commandSocket, *dataSocket; - bool connectionReady, passiveMode; - int getTotalSize, getDoneSize; - bool startGetOnFail; - int putToWrite, putWritten; - bool errorInListChildren; - -private: - void init(); - int addCommand( TQFtpCommand * ); - - bool checkConnection( TQNetworkOperation *op ); - -private Q_SLOTS: - void startNextCommand(); - void piFinished( const TQString& ); - void piError( int, const TQString& ); - void piConnectState( int ); - void piFtpReply( int, const TQString& ); - -private Q_SLOTS: - void npListInfo( const TQUrlInfo & ); - void npDone( bool ); - void npStateChanged( int ); - void npDataTransferProgress( int, int ); - void npReadyRead(); - -protected Q_SLOTS: - // ### TQt 4.0: delete these - void hostFound(); - void connected(); - void closed(); - void dataHostFound(); - void dataConnected(); - void dataClosed(); - void dataReadyRead(); - void dataBytesWritten( int nbytes ); - void error( int ); -}; - -#endif // TQT_NO_NETWORKPROTOCOL_FTP - -#endif // TQFTP_H diff --git a/tqtinterface/qt4/src/network/tqhostaddress.cpp b/tqtinterface/qt4/src/network/tqhostaddress.cpp deleted file mode 100644 index 5a4e5cf..0000000 --- a/tqtinterface/qt4/src/network/tqhostaddress.cpp +++ /dev/null @@ -1,459 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQHostAddress class. -** -** Created : 979899 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqhostaddress.h" -#include "tqstringlist.h" - -#ifdef USE_QT4 - -#else // USE_QT4 - -#ifndef TQT_NO_NETWORK -class TQHostAddressPrivate -{ -public: - TQHostAddressPrivate( TQ_UINT32 a_=0 ) : a(a_), isIp4(TRUE) - { - } - TQHostAddressPrivate( TQ_UINT8 *a_ ); - TQHostAddressPrivate(const TQ_IPV6ADDR &a_); - ~TQHostAddressPrivate() - { - } - - TQHostAddressPrivate & operator=( const TQHostAddressPrivate &from ) - { - a = from.a; - isIp4 = from.isIp4; - a6 = from.a6; - return *this; - } - -private: - TQ_UINT32 a; // ip 4 address - TQ_IPV6ADDR a6; // ip 6 address - bool isIp4; - - friend class TQHostAddress; -}; - -TQHostAddressPrivate::TQHostAddressPrivate(TQ_UINT8 *a_) : a(0), isIp4(FALSE) -{ - for ( int i=0; i<16; i++ ) { - a6.c[i] = a_[i]; - } -} - -TQHostAddressPrivate::TQHostAddressPrivate(const TQ_IPV6ADDR &a_) : a(0), isIp4(FALSE) -{ - a6 = a_; -} - -/*! - \class TQHostAddress tqhostaddress.h - \brief The TQHostAddress class provides an IP address. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class contains an IP address in a platform and protocol - independent manner. It stores both IPv4 and IPv6 addresses in a - way that you can easily access on any platform. - - TQHostAddress is normally used with the classes TQSocket, - TQServerSocket and TQSocketDevice to set up a server or to connect - to a host. - - Host addresses may be set with setAddress() and retrieved with - ip4Addr() or toString(). - - \sa TQSocket, TQServerSocket, TQSocketDevice -*/ - - -/*! - Creates a host address object with the IP address 0.0.0.0. -*/ -TQHostAddress::TQHostAddress() - : d( new TQHostAddressPrivate ) -{ -} - - -/*! - Creates a host address object for the IPv4 address \a ip4Addr. -*/ -TQHostAddress::TQHostAddress( TQ_UINT32 ip4Addr ) - : d( new TQHostAddressPrivate( ip4Addr ) ) -{ -} - - -/*! - Creates a host address object with the specified IPv6 address. - - \a ip6Addr must be a 16 byte array in network byte order - (high-order byte first). -*/ -TQHostAddress::TQHostAddress( TQ_UINT8 *ip6Addr ) - : d( new TQHostAddressPrivate( ip6Addr ) ) -{ -} - -/*! - Creates a host address object with the IPv6 address, \a ip6Addr. -*/ -TQHostAddress::TQHostAddress(const TQ_IPV6ADDR &ip6Addr) - : d(new TQHostAddressPrivate(ip6Addr)) -{ -} - -// ### DOC: Can only make this public if we specify precisely the -// format of the address string. -/*! - \internal -*/ -TQHostAddress::TQHostAddress(const TQString &address) - : d( new TQHostAddressPrivate ) -{ - setAddress( address ); -} - -/*! - Creates a copy of \a address. -*/ -TQHostAddress::TQHostAddress( const TQHostAddress &address ) - : d( new TQHostAddressPrivate ) -{ - *d = *(address.d); -} - - -/*! - Destroys the host address object. -*/ -TQHostAddress::~TQHostAddress() -{ - delete d; -} - - -/*! - Assigns another host address object \a address to this object and - returns a reference to this object. -*/ -TQHostAddress & TQHostAddress::operator=( const TQHostAddress & address ) -{ - *d = *(address.d); - return *this; -} - - -/*! - Set the IPv4 address specified by \a ip4Addr. -*/ -void TQHostAddress::setAddress( TQ_UINT32 ip4Addr ) -{ - delete d; - d = new TQHostAddressPrivate( ip4Addr ); -} - - -/*! - \overload - - Set the IPv6 address specified by \a ip6Addr. - - \a ip6Addr must be a 16 byte array in network byte order - (high-order byte first). -*/ -void TQHostAddress::setAddress( TQ_UINT8 *ip6Addr ) -{ - delete d; - d = new TQHostAddressPrivate( ip6Addr ); -} - -#ifndef TQT_NO_STRINGLIST -static bool parseIp4(const TQString& address, TQ_UINT32 *addr) -{ - TQStringList ipv4 = TQStringList::split(".", address, FALSE); - if (ipv4.count() == 4) { - int i = 0; - bool ok = TRUE; - while(ok && i < 4) { - uint byteValue = ipv4[i].toUInt(&ok); - if (byteValue > 255) - ok = FALSE; - if (ok) - *addr = (*addr << 8) + byteValue; - ++i; - } - if (ok) - return TRUE; - } - return FALSE; -} - -/*! - \overload - - Sets the IPv4 or IPv6 address specified by the string - representation \a address (e.g. "127.0.0.1"). Returns TRUE and - sets the address if the address was successfully parsed; otherwise - returns FALSE and leaves the address unchanged. -*/ -bool TQHostAddress::setAddress(const TQString& address) -{ - TQString a = address.simplifyWhiteSpace(); - - // try ipv4 - TQ_UINT32 maybeIp4 = 0; - if (parseIp4(address, &maybeIp4)) { - setAddress(maybeIp4); - return TRUE; - } - - // try ipv6 - TQStringList ipv6 = TQStringList::split(":", a, TRUE); - int count = (int)ipv6.count(); - if (count < 3) - return FALSE; // there must be at least two ":" - if (count > 8) - return FALSE; // maximum of seven ":" exceeded - TQ_UINT8 maybeIp6[16]; - int mc = 16; - int fillCount = 9 - count; - for (int i=count-1; i>=0; --i) { - if ( mc <= 0 ) - return FALSE; - - if (ipv6[i].isEmpty()) { - if (i==count-1) { - // special case: ":" is last character - if (!ipv6[i-1].isEmpty()) - return FALSE; - maybeIp6[--mc] = 0; - maybeIp6[--mc] = 0; - } else if (i==0) { - // special case: ":" is first character - if (!ipv6[i+1].isEmpty()) - return FALSE; - maybeIp6[--mc] = 0; - maybeIp6[--mc] = 0; - } else { - for (int j=0; j<fillCount; ++j) { - if ( mc <= 0 ) - return FALSE; - maybeIp6[--mc] = 0; - maybeIp6[--mc] = 0; - } - } - } else { - bool ok = FALSE; - uint byteValue = ipv6[i].toUInt(&ok, 16); - if (ok && byteValue <= 0xffff) { - maybeIp6[--mc] = byteValue & 0xff; - maybeIp6[--mc] = (byteValue >> 8) & 0xff; - } else { - if (i == count-1) { - // parse the ipv4 part of a mixed type - if (!parseIp4(ipv6[i], &maybeIp4)) - return FALSE; - maybeIp6[--mc] = maybeIp4 & 0xff; - maybeIp6[--mc] = (maybeIp4 >> 8) & 0xff; - maybeIp6[--mc] = (maybeIp4 >> 16) & 0xff; - maybeIp6[--mc] = (maybeIp4 >> 24) & 0xff; - --fillCount; - } else { - return FALSE; - } - } - } - } - if (mc == 0) { - setAddress(maybeIp6); - return TRUE; - } - - return FALSE; -} -#endif - -/*! - \obsolete - - Use isIPv4Address() instead. -*/ -bool TQHostAddress::isIp4Addr() const -{ - return isIPv4Address(); -} - -/*! - Returns TRUE if the host address represents an IPv4 address; - otherwise returns FALSE. -*/ -bool TQHostAddress::isIPv4Address() const -{ - return d->isIp4; -} - -/*! - \obsolete - - Use toIPv4Address() instead. -*/ -TQ_UINT32 TQHostAddress::ip4Addr() const -{ - return toIPv4Address(); -} - -/*! - Returns the IPv4 address as a number. - - For example, if the address is 127.0.0.1, the returned value is - 2130706433 (i.e. 0x7f000001). - - This value is only valid when isIp4Addr() returns TRUE. - - \sa toString() -*/ -TQ_UINT32 TQHostAddress::toIPv4Address() const -{ - return d->a; -} - -/*! - Returns TRUE if the host address represents an IPv6 address; - otherwise returns FALSE. -*/ -bool TQHostAddress::isIPv6Address() const -{ - return !d->isIp4; -} - -/*! - Returns the IPv6 address as a TQ_IPV6ADDR structure. The structure - consists of 16 unsigned characters. - - \code - TQ_IPV6ADDR addr = hostAddr.ip6Addr(); - // addr.c[] contains 16 unsigned characters - - for (int i = 0; i < 16; ++i) { - // process addr.c[i] - } - \endcode - - This value is only valid when isIPv6Address() returns TRUE. - - \sa toString() -*/ -TQ_IPV6ADDR TQHostAddress::toIPv6Address() const -{ - return d->a6; -} - -#ifndef TQT_NO_SPRINTF -/*! - Returns the address as a string. - - For example, if the address is the IPv4 address 127.0.0.1, the - returned string is "127.0.0.1". - - \sa ip4Addr() -*/ -TQString TQHostAddress::toString() const -{ - if ( d->isIp4 ) { - TQ_UINT32 i = ip4Addr(); - TQString s; - s.sprintf( "%d.%d.%d.%d", (i>>24) & 0xff, (i>>16) & 0xff, - (i >> 8) & 0xff, i & 0xff ); - return s; - } else { - TQ_UINT16 ugle[8]; - for ( int i=0; i<8; i++ ) { - ugle[i] = ( (TQ_UINT16)( d->a6.c[2*i] ) << 8 ) | - ( (TQ_UINT16)( d->a6.c[2*i+1] ) ); - } - TQString s; - s.sprintf( "%X:%X:%X:%X:%X:%X:%X:%X", - ugle[0], ugle[1], ugle[2], ugle[3], - ugle[4], ugle[5], ugle[6], ugle[7] ); - return s; - } -} -#endif - - -/*! - Returns TRUE if this host address is the same as \a other; - otherwise returns FALSE. -*/ -bool TQHostAddress::operator==( const TQHostAddress & other ) const -{ - return d->a == other.d->a; -} - - -/*! - Returns TRUE if this host address is null (INADDR_ANY or in6addr_any). The - default constructor creates a null address, and that address isn't valid - for any particular host or interface. -*/ -bool TQHostAddress::isNull() const -{ - if ( d->isIp4 ) - return d->a == 0; - int i = 0; - while( i < 16 ) { - if ( d->a6.c[i++] != 0 ) - return FALSE; - } - return TRUE; -} - -#endif //TQT_NO_NETWORK - -#endif // USE_QT4
\ No newline at end of file diff --git a/tqtinterface/qt4/src/network/tqhostaddress.h b/tqtinterface/qt4/src/network/tqhostaddress.h deleted file mode 100644 index 692bfd8..0000000 --- a/tqtinterface/qt4/src/network/tqhostaddress.h +++ /dev/null @@ -1,131 +0,0 @@ -/**************************************************************************** -** -** Definition of TQHostAddress class. -** -** Created : 979899 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQHOSTADDRESS_H -#define TQHOSTADDRESS_H - -#include "tqtglobaldefines.h" - -#ifndef TQT_H -#include "tqstring.h" -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_NETWORK -#else -#define TQM_EXPORT_NETWORK TQ_EXPORT -#endif - -#ifdef USE_QT4 - -#include <Qt/qhostaddress.h> - -#endif // USE_QT4 - -#ifndef TQT_NO_NETWORK -class TQHostAddressPrivate; - -typedef Q_IPV6ADDR TQ_IPV6ADDR; - -#ifdef USE_QT4 - -class TQM_EXPORT_NETWORK TQHostAddress : public QHostAddress -{ -public: - TQHostAddress() : QHostAddress() {} - TQHostAddress( TQ_UINT32 ip4Addr ) : QHostAddress( ip4Addr ) {} - TQHostAddress( TQ_UINT8 *ip6Addr ) : QHostAddress( ip6Addr ) {} - TQHostAddress(const TQ_IPV6ADDR &ip6Addr) : QHostAddress( ip6Addr ) {} -#ifndef TQT_NO_STRINGLIST - TQHostAddress(const QString &address) : QHostAddress( address ) {} -#endif - TQHostAddress( const QHostAddress &tqha ) : QHostAddress( tqha ) {} - - inline quint32 ip4Addr() const { return toIPv4Address(); } - inline bool isIPv4Address() const { return protocol() == QAbstractSocket::IPv4Protocol || protocol() == QAbstractSocket::UnknownNetworkLayerProtocol; } - inline bool isIp4Addr() const { return protocol() == QAbstractSocket::IPv4Protocol || protocol() == QAbstractSocket::UnknownNetworkLayerProtocol; } - inline bool isIPv6Address() const { return protocol() == QAbstractSocket::IPv6Protocol; } -}; - -#else // USE_QT4 - -class TQM_EXPORT_NETWORK TQHostAddress -{ -public: - TQHostAddress(); - TQHostAddress( TQ_UINT32 ip4Addr ); - TQHostAddress( TQ_UINT8 *ip6Addr ); - TQHostAddress(const TQ_IPV6ADDR &ip6Addr); -#ifndef TQT_NO_STRINGLIST - TQHostAddress(const TQString &address); -#endif - TQHostAddress( const TQHostAddress & ); - virtual ~TQHostAddress(); - - TQHostAddress & operator=( const TQHostAddress & ); - - void setAddress( TQ_UINT32 ip4Addr ); - void setAddress( TQ_UINT8 *ip6Addr ); -#ifndef TQT_NO_STRINGLIST - bool setAddress( const TQString& address ); -#endif - bool isIp4Addr() const; // obsolete - TQ_UINT32 ip4Addr() const; // obsolete - - bool isIPv4Address() const; - TQ_UINT32 toIPv4Address() const; - bool isIPv6Address() const; - TQ_IPV6ADDR toIPv6Address() const; - -#ifndef TQT_NO_SPRINTF - TQString toString() const; -#endif - - bool operator==( const TQHostAddress & ) const; - bool isNull() const; - -private: - TQHostAddressPrivate* d; -}; - -#endif // USE_QT4 - -#endif //TQT_NO_NETWORK -#endif diff --git a/tqtinterface/qt4/src/network/tqhttp.cpp b/tqtinterface/qt4/src/network/tqhttp.cpp deleted file mode 100644 index 6b1f061..0000000 --- a/tqtinterface/qt4/src/network/tqhttp.cpp +++ /dev/null @@ -1,2384 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQHttp and related classes. -** -** Created : 970521 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqhttp.h" - -#ifndef TQT_NO_NETWORKPROTOCOL_HTTP - -#include "tqsocket.h" -#include "tqtextstream.h" -#include "tqmap.h" -#include "tqstring.h" -#include "tqstringlist.h" -#include "tqcstring.h" -#include "tqbuffer.h" -#include "tqurloperator.h" -#include "tqtimer.h" -#include "private/tqinternal_p.h" - -//#define TQHTTP_DEBUG - -class TQHttpPrivate -{ -public: - TQHttpPrivate() : - state( TQHttp::Unconnected ), - error( TQHttp::NoError ), - hostname( TQString::null ), - port( 0 ), - toDevice( 0 ), - postDevice( 0 ), - bytesDone( 0 ), - chunkedSize( -1 ), - idleTimer( 0 ) - { - pending.setAutoDelete( TRUE ); - } - - TQSocket socket; - TQPtrList<TQHttpRequest> pending; - - TQHttp::State state; - TQHttp::Error error; - TQString errorString; - - TQString hostname; - TQ_UINT16 port; - - TQByteArray buffer; - TQIODevice* toDevice; - TQIODevice* postDevice; - - uint bytesDone; - uint bytesTotal; - TQ_LONG chunkedSize; - - TQHttpRequestHeader header; - - bool readHeader; - TQString headerStr; - TQHttpResponseHeader response; - - int idleTimer; - - TQMembuf rba; -}; - -class TQHttpRequest -{ -public: - TQHttpRequest() - { - id = ++idCounter; - } - virtual ~TQHttpRequest() - { } - - virtual void start( TQHttp * ) = 0; - virtual bool hasRequestHeader(); - virtual TQHttpRequestHeader requestHeader(); - - virtual TQIODevice* sourceDevice() = 0; - virtual TQIODevice* destinationDevice() = 0; - - int id; - -private: - static int idCounter; -}; - -int TQHttpRequest::idCounter = 0; - -bool TQHttpRequest::hasRequestHeader() -{ - return FALSE; -} - -TQHttpRequestHeader TQHttpRequest::requestHeader() -{ - return TQHttpRequestHeader(); -} - -/**************************************************** - * - * TQHttpNormalRequest - * - ****************************************************/ - -class TQHttpNormalRequest : public TQHttpRequest -{ -public: - TQHttpNormalRequest( const TQHttpRequestHeader &h, TQIODevice *d, TQIODevice *t ) : - header(h), to(t) - { - is_ba = FALSE; - data.dev = d; - } - - TQHttpNormalRequest( const TQHttpRequestHeader &h, TQByteArray *d, TQIODevice *t ) : - header(h), to(t) - { - is_ba = TRUE; - data.ba = d; - } - - ~TQHttpNormalRequest() - { - if ( is_ba ) - delete data.ba; - } - - void start( TQHttp * ); - bool hasRequestHeader(); - TQHttpRequestHeader requestHeader(); - - TQIODevice* sourceDevice(); - TQIODevice* destinationDevice(); - -protected: - TQHttpRequestHeader header; - -private: - union { - TQByteArray *ba; - TQIODevice *dev; - } data; - bool is_ba; - TQIODevice *to; -}; - -void TQHttpNormalRequest::start( TQHttp *http ) -{ - http->d->header = header; - - if ( is_ba ) { - http->d->buffer = *data.ba; - if ( http->d->buffer.size() > 0 ) - http->d->header.setContentLength( http->d->buffer.size() ); - - http->d->postDevice = 0; - } else { - http->d->buffer = TQByteArray(); - - if ( data.dev && ( data.dev->isOpen() || data.dev->open(IO_ReadOnly) ) ) { - http->d->postDevice = data.dev; - if ( http->d->postDevice->size() > 0 ) - http->d->header.setContentLength( http->d->postDevice->size() ); - } else { - http->d->postDevice = 0; - } - } - - if ( to && ( to->isOpen() || to->open(IO_WriteOnly) ) ) - http->d->toDevice = to; - else - http->d->toDevice = 0; - - http->sendRequest(); -} - -bool TQHttpNormalRequest::hasRequestHeader() -{ - return TRUE; -} - -TQHttpRequestHeader TQHttpNormalRequest::requestHeader() -{ - return header; -} - -TQIODevice* TQHttpNormalRequest::sourceDevice() -{ - if ( is_ba ) - return 0; - return data.dev; -} - -TQIODevice* TQHttpNormalRequest::destinationDevice() -{ - return to; -} - -/**************************************************** - * - * TQHttpPGHRequest - * (like a TQHttpNormalRequest, but for the convenience - * functions put(), get() and head() -- i.e. set the - * host header field correctly before sending the - * request) - * - ****************************************************/ - -class TQHttpPGHRequest : public TQHttpNormalRequest -{ -public: - TQHttpPGHRequest( const TQHttpRequestHeader &h, TQIODevice *d, TQIODevice *t ) : - TQHttpNormalRequest( h, d, t ) - { } - - TQHttpPGHRequest( const TQHttpRequestHeader &h, TQByteArray *d, TQIODevice *t ) : - TQHttpNormalRequest( h, d, t ) - { } - - ~TQHttpPGHRequest() - { } - - void start( TQHttp * ); -}; - -void TQHttpPGHRequest::start( TQHttp *http ) -{ - header.setValue( "Host", http->d->hostname ); - TQHttpNormalRequest::start( http ); -} - -/**************************************************** - * - * TQHttpSetHostRequest - * - ****************************************************/ - -class TQHttpSetHostRequest : public TQHttpRequest -{ -public: - TQHttpSetHostRequest( const TQString &h, TQ_UINT16 p ) : - hostname(h), port(p) - { } - - void start( TQHttp * ); - - TQIODevice* sourceDevice() - { return 0; } - TQIODevice* destinationDevice() - { return 0; } - -private: - TQString hostname; - TQ_UINT16 port; -}; - -void TQHttpSetHostRequest::start( TQHttp *http ) -{ - http->d->hostname = hostname; - http->d->port = port; - http->finishedWithSuccess(); -} - -/**************************************************** - * - * TQHttpCloseRequest - * - ****************************************************/ - -class TQHttpCloseRequest : public TQHttpRequest -{ -public: - TQHttpCloseRequest() - { } - void start( TQHttp * ); - - TQIODevice* sourceDevice() - { return 0; } - TQIODevice* destinationDevice() - { return 0; } -}; - -void TQHttpCloseRequest::start( TQHttp *http ) -{ - http->close(); -} - -/**************************************************** - * - * TQHttpHeader - * - ****************************************************/ - -/*! - \class TQHttpHeader tqhttp.h - \brief The TQHttpHeader class contains header information for HTTP. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - In most cases you should use the more specialized derivatives of - this class, TQHttpResponseHeader and TQHttpRequestHeader, rather - than directly using TQHttpHeader. - - TQHttpHeader provides the HTTP header fields. A HTTP header field - consists of a name followed by a colon, a single space, and the - field value. (See RFC 1945.) Field names are case-insensitive. A - typical header field looks like this: - \code - content-type: text/html - \endcode - - In the API the header field name is called the "key" and the - content is called the "value". You can get and set a header - field's value by using its key with value() and setValue(), e.g. - \code - header.setValue( "content-type", "text/html" ); - TQString contentType = header.value( "content-type" ); - \endcode - - Some fields are so common that getters and setters are provided - for them as a convenient alternative to using \l value() and - \l setValue(), e.g. contentLength() and contentType(), - setContentLength() and setContentType(). - - Each header key has a \e single value associated with it. If you - set the value for a key which already exists the previous value - will be discarded. - - \sa TQHttpRequestHeader TQHttpResponseHeader -*/ - -/*! - \fn int TQHttpHeader::majorVersion() const - - Returns the major protocol-version of the HTTP header. -*/ - -/*! - \fn int TQHttpHeader::minorVersion() const - - Returns the minor protocol-version of the HTTP header. -*/ - -/*! - Constructs an empty HTTP header. -*/ -TQHttpHeader::TQHttpHeader() - : valid( TRUE ) -{ -} - -/*! - Constructs a copy of \a header. -*/ -TQHttpHeader::TQHttpHeader( const TQHttpHeader& header ) - : valid( header.valid ) -{ - values = header.values; -} - -/*! - Constructs a HTTP header for \a str. - - This constructor parses the string \a str for header fields and - adds this information. The \a str should consist of one or more - "\r\n" delimited lines; each of these lines should have the format - key, colon, space, value. -*/ -TQHttpHeader::TQHttpHeader( const TQString& str ) - : valid( TRUE ) -{ - parse( str ); -} - -/*! - Destructor. -*/ -TQHttpHeader::~TQHttpHeader() -{ -} - -/*! - Assigns \a h and returns a reference to this http header. -*/ -TQHttpHeader& TQHttpHeader::operator=( const TQHttpHeader& h ) -{ - values = h.values; - valid = h.valid; - return *this; -} - -/*! - Returns TRUE if the HTTP header is valid; otherwise returns FALSE. - - A TQHttpHeader is invalid if it was created by parsing a malformed string. -*/ -bool TQHttpHeader::isValid() const -{ - return valid; -} - -/*! \internal - Parses the HTTP header string \a str for header fields and adds - the keys/values it finds. If the string is not parsed successfully - the TQHttpHeader becomes \link isValid() invalid\endlink. - - Returns TRUE if \a str was successfully parsed; otherwise returns FALSE. - - \sa toString() -*/ -bool TQHttpHeader::parse( const TQString& str ) -{ - TQStringList lst; - int pos = str.find( '\n' ); - if ( pos > 0 && str.at( pos - 1 ) == '\r' ) - lst = TQStringList::split( "\r\n", str.stripWhiteSpace(), FALSE ); - else - lst = TQStringList::split( "\n", str.stripWhiteSpace(), FALSE ); - - if ( lst.isEmpty() ) - return TRUE; - - TQStringList lines; - TQStringList::Iterator it = lst.begin(); - for( ; it != lst.end(); ++it ) { - if ( !(*it).isEmpty() ) { - if ( (*it)[0].isSpace() ) { - if ( !lines.isEmpty() ) { - lines.last() += " "; - lines.last() += (*it).stripWhiteSpace(); - } - } else { - lines.append( (*it) ); - } - } - } - - int number = 0; - it = lines.begin(); - for( ; it != lines.end(); ++it ) { - if ( !parseLine( *it, number++ ) ) { - valid = FALSE; - return FALSE; - } - } - return TRUE; -} - -/*! \internal -*/ -void TQHttpHeader::setValid( bool v ) -{ - valid = v; -} - -/*! - Returns the value for the entry with the given \a key. If no entry - has this \a key, an empty string is returned. - - \sa setValue() removeValue() hasKey() keys() -*/ -TQString TQHttpHeader::value( const TQString& key ) const -{ - return values[ key.lower() ]; -} - -/*! - Returns a list of the keys in the HTTP header. - - \sa hasKey() -*/ -TQStringList TQHttpHeader::keys() const -{ - return values.keys(); -} - -/*! - Returns TRUE if the HTTP header has an entry with the given \a - key; otherwise returns FALSE. - - \sa value() setValue() keys() -*/ -bool TQHttpHeader::hasKey( const TQString& key ) const -{ - return values.contains( key.lower() ); -} - -/*! - Sets the value of the entry with the \a key to \a value. - - If no entry with \a key exists, a new entry with the given \a key - and \a value is created. If an entry with the \a key already - exists, its value is discarded and replaced with the given \a - value. - - \sa value() hasKey() removeValue() -*/ -void TQHttpHeader::setValue( const TQString& key, const TQString& value ) -{ - values[ key.lower() ] = value; -} - -/*! - Removes the entry with the key \a key from the HTTP header. - - \sa value() setValue() -*/ -void TQHttpHeader::removeValue( const TQString& key ) -{ - values.remove( key.lower() ); -} - -/*! \internal - Parses the single HTTP header line \a line which has the format - key, colon, space, value, and adds key/value to the headers. The - linenumber is \a number. Returns TRUE if the line was successfully - parsed and the key/value added; otherwise returns FALSE. - - \sa parse() -*/ -bool TQHttpHeader::parseLine( const TQString& line, int ) -{ - int i = line.find( ":" ); - if ( i == -1 ) - return FALSE; - - values.insert( TQT_TQSTRING(line.left( i )).stripWhiteSpace().lower(), TQT_TQSTRING(line.mid( i + 1 )).stripWhiteSpace() ); - - return TRUE; -} - -/*! - Returns a string representation of the HTTP header. - - The string is suitable for use by the constructor that takes a - TQString. It consists of lines with the format: key, colon, space, - value, "\r\n". -*/ -TQString TQHttpHeader::toString() const -{ - if ( !isValid() ) - return ""; - - TQString ret = ""; - - TQMap<TQString,TQString>::ConstIterator it = values.begin(); - for( ; it != values.end(); ++it ) - ret += it.key() + ": " + it.data() + "\r\n"; - - return ret; -} - -/*! - Returns TRUE if the header has an entry for the special HTTP - header field \c content-length; otherwise returns FALSE. - - \sa contentLength() setContentLength() -*/ -bool TQHttpHeader::hasContentLength() const -{ - return hasKey( "content-length" ); -} - -/*! - Returns the value of the special HTTP header field \c - content-length. - - \sa setContentLength() hasContentLength() -*/ -uint TQHttpHeader::contentLength() const -{ - return values[ "content-length" ].toUInt(); -} - -/*! - Sets the value of the special HTTP header field \c content-length - to \a len. - - \sa contentLength() hasContentLength() -*/ -void TQHttpHeader::setContentLength( int len ) -{ - values[ "content-length" ] = TQString::number( len ); -} - -/*! - Returns TRUE if the header has an entry for the the special HTTP - header field \c content-type; otherwise returns FALSE. - - \sa contentType() setContentType() -*/ -bool TQHttpHeader::hasContentType() const -{ - return hasKey( "content-type" ); -} - -/*! - Returns the value of the special HTTP header field \c content-type. - - \sa setContentType() hasContentType() -*/ -TQString TQHttpHeader::contentType() const -{ - TQString type = values[ "content-type" ]; - if ( type.isEmpty() ) - return TQString::null; - - int pos = type.find( ";" ); - if ( pos == -1 ) - return type; - - return TQT_TQSTRING(type.left( pos )).stripWhiteSpace(); -} - -/*! - Sets the value of the special HTTP header field \c content-type to - \a type. - - \sa contentType() hasContentType() -*/ -void TQHttpHeader::setContentType( const TQString& type ) -{ - values[ "content-type" ] = type; -} - -/**************************************************** - * - * TQHttpResponseHeader - * - ****************************************************/ - -/*! - \class TQHttpResponseHeader tqhttp.h - \brief The TQHttpResponseHeader class contains response header information for HTTP. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class is used by the TQHttp class to report the header - information that the client received from the server. - - HTTP responses have a status code that indicates the status of the - response. This code is a 3-digit integer result code (for details - see to RFC 1945). In addition to the status code, you can also - specify a human-readable text that describes the reason for the - code ("reason phrase"). This class allows you to get the status - code and the reason phrase. - - \sa TQHttpRequestHeader TQHttp -*/ - -/*! - Constructs an empty HTTP response header. -*/ -TQHttpResponseHeader::TQHttpResponseHeader() -{ - setValid( FALSE ); -} - -/*! - Constructs a HTTP response header with the status code \a code, - the reason phrase \a text and the protocol-version \a majorVer and - \a minorVer. -*/ -TQHttpResponseHeader::TQHttpResponseHeader( int code, const TQString& text, int majorVer, int minorVer ) - : TQHttpHeader(), statCode( code ), reasonPhr( text ), majVer( majorVer ), minVer( minorVer ) -{ -} - -/*! - Constructs a copy of \a header. -*/ -TQHttpResponseHeader::TQHttpResponseHeader( const TQHttpResponseHeader& header ) - : TQHttpHeader( header ), statCode( header.statCode ), reasonPhr( header.reasonPhr ), majVer( header.majVer ), minVer( header.minVer ) -{ -} - -/*! - Constructs a HTTP response header from the string \a str. The - string is parsed and the information is set. The \a str should - consist of one or more "\r\n" delimited lines; the first line should be the - status-line (format: HTTP-version, space, status-code, space, - reason-phrase); each of remaining lines should have the format key, colon, - space, value. -*/ -TQHttpResponseHeader::TQHttpResponseHeader( const TQString& str ) - : TQHttpHeader() -{ - parse( str ); -} - -/*! - Sets the status code to \a code, the reason phrase to \a text and - the protocol-version to \a majorVer and \a minorVer. - - \sa statusCode() reasonPhrase() majorVersion() minorVersion() -*/ -void TQHttpResponseHeader::setqStatusLine( int code, const TQString& text, int majorVer, int minorVer ) -{ - setValid( TRUE ); - statCode = code; - reasonPhr = text; - majVer = majorVer; - minVer = minorVer; -} - -/*! - Returns the status code of the HTTP response header. - - \sa reasonPhrase() majorVersion() minorVersion() -*/ -int TQHttpResponseHeader::statusCode() const -{ - return statCode; -} - -/*! - Returns the reason phrase of the HTTP response header. - - \sa statusCode() majorVersion() minorVersion() -*/ -TQString TQHttpResponseHeader::reasonPhrase() const -{ - return reasonPhr; -} - -/*! - Returns the major protocol-version of the HTTP response header. - - \sa minorVersion() statusCode() reasonPhrase() -*/ -int TQHttpResponseHeader::majorVersion() const -{ - return majVer; -} - -/*! - Returns the minor protocol-version of the HTTP response header. - - \sa majorVersion() statusCode() reasonPhrase() -*/ -int TQHttpResponseHeader::minorVersion() const -{ - return minVer; -} - -/*! \reimp -*/ -bool TQHttpResponseHeader::parseLine( const TQString& line, int number ) -{ - if ( number != 0 ) - return TQHttpHeader::parseLine( line, number ); - - TQString l = line.simplifyWhiteSpace(); - if ( l.length() < 10 ) - return FALSE; - - if ( l.left( 5 ) == "HTTP/" && l[5].isDigit() && l[6] == '.' && - l[7].isDigit() && l[8] == ' ' && l[9].isDigit() ) { - majVer = l[5].latin1() - '0'; - minVer = l[7].latin1() - '0'; - - int pos = l.find( ' ', 9 ); - if ( pos != -1 ) { - reasonPhr = l.mid( pos + 1 ); - statCode = l.mid( 9, pos - 9 ).toInt(); - } else { - statCode = l.mid( 9 ).toInt(); - reasonPhr = TQString::null; - } - } else { - return FALSE; - } - - return TRUE; -} - -/*! \reimp -*/ -TQString TQHttpResponseHeader::toString() const -{ - TQString ret( "HTTP/%1.%2 %3 %4\r\n%5\r\n" ); - return ret.arg( majVer ).arg ( minVer ).arg( statCode ).arg( reasonPhr ).arg( TQHttpHeader::toString() ); -} - -/**************************************************** - * - * TQHttpRequestHeader - * - ****************************************************/ - -/*! - \class TQHttpRequestHeader tqhttp.h - \brief The TQHttpRequestHeader class contains request header information for -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - HTTP. -\if defined(commercial_edition) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class is used in the TQHttp class to report the header - information if the client requests something from the server. - - HTTP requests have a method which describes the request's action. - The most common requests are "GET" and "POST". In addition to the - request method the header also includes a request-URI to specify - the location for the method to use. - - The method, request-URI and protocol-version can be set using a - constructor or later using setRequest(). The values can be - obtained using method(), path(), majorVersion() and - minorVersion(). - - This class is a TQHttpHeader subclass so that class's functions, - e.g. \link TQHttpHeader::setValue() setValue()\endlink, \link - TQHttpHeader::value() value()\endlink, etc. are also available. - - \sa TQHttpResponseHeader TQHttp - - \important value() setValue() -*/ - -/*! - Constructs an empty HTTP request header. -*/ -TQHttpRequestHeader::TQHttpRequestHeader() - : TQHttpHeader() -{ - setValid( FALSE ); -} - -/*! - Constructs a HTTP request header for the method \a method, the - request-URI \a path and the protocol-version \a majorVer and \a minorVer. -*/ -TQHttpRequestHeader::TQHttpRequestHeader( const TQString& method, const TQString& path, int majorVer, int minorVer ) - : TQHttpHeader(), m( method ), p( path ), majVer( majorVer ), minVer( minorVer ) -{ -} - -/*! - Constructs a copy of \a header. -*/ -TQHttpRequestHeader::TQHttpRequestHeader( const TQHttpRequestHeader& header ) - : TQHttpHeader( header ), m( header.m ), p( header.p ), majVer( header.majVer ), minVer( header.minVer ) -{ -} - -/*! - Constructs a HTTP request header from the string \a str. The \a - str should consist of one or more "\r\n" delimited lines; the first line - should be the request-line (format: method, space, request-URI, space - HTTP-version); each of the remaining lines should have the format key, - colon, space, value. -*/ -TQHttpRequestHeader::TQHttpRequestHeader( const TQString& str ) - : TQHttpHeader() -{ - parse( str ); -} - -/*! - This function sets the request method to \a method, the - request-URI to \a path and the protocol-version to \a majorVer and - \a minorVer. - - \sa method() path() majorVersion() minorVersion() -*/ -void TQHttpRequestHeader::setRequest( const TQString& method, const TQString& path, int majorVer, int minorVer ) -{ - setValid( TRUE ); - m = method; - p = path; - majVer = majorVer; - minVer = minorVer; -} - -/*! - Returns the method of the HTTP request header. - - \sa path() majorVersion() minorVersion() setRequest() -*/ -TQString TQHttpRequestHeader::method() const -{ - return m; -} - -/*! - Returns the request-URI of the HTTP request header. - - \sa method() majorVersion() minorVersion() setRequest() -*/ -TQString TQHttpRequestHeader::path() const -{ - return p; -} - -/*! - Returns the major protocol-version of the HTTP request header. - - \sa minorVersion() method() path() setRequest() -*/ -int TQHttpRequestHeader::majorVersion() const -{ - return majVer; -} - -/*! - Returns the minor protocol-version of the HTTP request header. - - \sa majorVersion() method() path() setRequest() -*/ -int TQHttpRequestHeader::minorVersion() const -{ - return minVer; -} - -/*! \reimp -*/ -bool TQHttpRequestHeader::parseLine( const TQString& line, int number ) -{ - if ( number != 0 ) - return TQHttpHeader::parseLine( line, number ); - - TQStringList lst = TQStringList::split( " ", line.simplifyWhiteSpace() ); - if ( lst.count() > 0 ) { - m = lst[0]; - if ( lst.count() > 1 ) { - p = lst[1]; - if ( lst.count() > 2 ) { - TQString v = lst[2]; - if ( v.length() >= 8 && v.left( 5 ) == "HTTP/" && - v[5].isDigit() && v[6] == '.' && v[7].isDigit() ) { - majVer = v[5].latin1() - '0'; - minVer = v[7].latin1() - '0'; - return TRUE; - } - } - } - } - - return FALSE; -} - -/*! \reimp -*/ -TQString TQHttpRequestHeader::toString() const -{ - TQString first( "%1 %2"); - TQString last(" HTTP/%3.%4\r\n%5\r\n" ); - return first.arg( m ).arg( p ) + - last.arg( majVer ).arg( minVer ).arg( TQHttpHeader::toString()); -} - - -/**************************************************** - * - * TQHttp - * - ****************************************************/ -/*! - \class TQHttp tqhttp.h - \brief The TQHttp class provides an implementation of the HTTP protocol. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class provides two different interfaces: one is the - TQNetworkProtocol interface that allows you to use HTTP through the - TQUrlOperator abstraction. The other is a direct interface to HTTP - that allows you to have more control over the requests and that - allows you to access the response header fields. - - Don't mix the two interfaces, since the behavior is not - well-defined. - - If you want to use TQHttp with the TQNetworkProtocol interface, you - do not use it directly, but rather through a TQUrlOperator, for - example: - - \code - TQUrlOperator op( "http://www.trolltech.com" ); - op.get( "index.html" ); - \endcode - - This code will only work if the TQHttp class is registered; to - register the class, you must call qInitNetworkProtocols() before - using a TQUrlOperator with HTTP. - - The TQNetworkProtocol interface for HTTP only supports the - operations operationGet() and operationPut(), i.e. - TQUrlOperator::get() and TQUrlOperator::put(), if you use it with a - TQUrlOperator. - - The rest of this descrption describes the direct interface to - HTTP. - - The class works asynchronously, so there are no blocking - functions. If an operation cannot be executed immediately, the - function will still return straight away and the operation will be - scheduled for later execution. The results of scheduled operations - are reported via Q_SIGNALS. This approach depends on the event loop - being in operation. - - The operations that can be scheduled (they are called "requests" - in the rest of the documentation) are the following: setHost(), - get(), post(), head() and request(). - - All of these requests return a unique identifier that allows you - to keep track of the request that is currently executed. When the - execution of a request starts, the requestStarted() signal with - the identifier is emitted and when the request is finished, the - requestFinished() signal is emitted with the identifier and a bool - that indicates if the request finished with an error. - - To make an HTTP request you must set up suitable HTTP headers. The - following example demonstrates, how to request the main HTML page - from the Trolltech home page (i.e. the URL - http://www.trolltech.com/index.html): - - \code - TQHttpRequestHeader header( "GET", "/index.html" ); - header.setValue( "Host", "www.trolltech.com" ); - http->setHost( "www.trolltech.com" ); - http->request( header ); - \endcode - - For the common HTTP requests \c GET, \c POST and \c HEAD, TQHttp - provides the convenience functions get(), post() and head(). They - already use a reasonable header and if you don't have to set - special header fields, they are easier to use. The above example - can also be written as: - - \code - http->setHost( "www.trolltech.com" ); // id == 1 - http->get( "/index.html" ); // id == 2 - \endcode - - For this example the following sequence of Q_SIGNALS is emitted - (with small variations, depending on network traffic, etc.): - - \code - requestStarted( 1 ) - requestFinished( 1, FALSE ) - - requestStarted( 2 ) - stateChanged( Connecting ) - stateChanged( Sending ) - dataSendProgress( 77, 77 ) - stateChanged( Reading ) - responseHeaderReceived( responseheader ) - dataReadProgress( 5388, 0 ) - readyRead( responseheader ) - dataReadProgress( 18300, 0 ) - readyRead( responseheader ) - stateChanged( Connected ) - requestFinished( 2, FALSE ) - - done( FALSE ) - - stateChanged( Closing ) - stateChanged( Unconnected ) - \endcode - - The dataSendProgress() and dataReadProgress() Q_SIGNALS in the above - example are useful if you want to show a \link TQProgressBar - progressbar\endlink to inform the user about the progress of the - download. The second argument is the total size of data. In - certain cases it is not possible to know the total amount in - advance, in which case the second argument is 0. (If you connect - to a TQProgressBar a total of 0 results in a busy indicator.) - - When the response header is read, it is reported with the - responseHeaderReceived() signal. - - The readyRead() signal tells you that there is data ready to be - read. The amount of data can then be queried with the - bytesAvailable() function and it can be read with the readBlock() - or readAll() functions. - - If an error occurs during the execution of one of the commands in - a sequence of commands, all the pending commands (i.e. scheduled, - but not yet executed commands) are cleared and no Q_SIGNALS are - emitted for them. - - For example, if you have the following sequence of reqeusts - - \code - http->setHost( "www.foo.bar" ); // id == 1 - http->get( "/index.html" ); // id == 2 - http->post( "register.html", data ); // id == 3 - \endcode - - and the get() request fails because the host lookup fails, then - the post() request is never executed and the Q_SIGNALS would look - like this: - - \code - requestStarted( 1 ) - requestFinished( 1, FALSE ) - - requestStarted( 2 ) - stateChanged( HostLookup ) - requestFinished( 2, TRUE ) - - done( TRUE ) - - stateChanged( Unconnected ) - \endcode - - You can then get details about the error with the error() and - errorString() functions. Note that only unexpected behaviour, like - network failure is considered as an error. If the server response - contains an error status, like a 404 response, this is reported as - a normal response case. So you should always check the \link - TQHttpResponseHeader::statusCode() status code \endlink of the - response header. - - The functions currentId() and currentRequest() provide more - information about the currently executing request. - - The functions hasPendingRequests() and clearPendingRequests() - allow you to query and clear the list of pending requests. - - \sa \link network.html TQt Network Documentation \endlink TQNetworkProtocol, TQUrlOperator TQFtp -*/ - -/*! - Constructs a TQHttp object. -*/ -TQHttp::TQHttp() -{ - init(); -} - -/*! - Constructs a TQHttp object. The parameters \a parent and \a name - are passed on to the TQObject constructor. -*/ -TQHttp::TQHttp( TQObject* parent, const char* name ) -{ - if ( parent ) - parent->insertChild( this ); - setName( name ); - init(); -} - -/*! - Constructs a TQHttp object. Subsequent requests are done by - connecting to the server \a hostname on port \a port. The - parameters \a parent and \a name are passed on to the TQObject - constructor. - - \sa setHost() -*/ -TQHttp::TQHttp( const TQString &hostname, TQ_UINT16 port, TQObject* parent, const char* name ) -{ - if ( parent ) - parent->insertChild( this ); - setName( name ); - init(); - - d->hostname = hostname; - d->port = port; -} - -void TQHttp::init() -{ - bytesRead = 0; - d = new TQHttpPrivate; - d->errorString = tr( "Unknown error" ); - - connect( &d->socket, TQT_SIGNAL( connected() ), - this, TQT_SLOT( slotConnected() ) ); - connect( &d->socket, TQT_SIGNAL( connectionClosed() ), - this, TQT_SLOT( slotClosed() ) ); - connect( &d->socket, TQT_SIGNAL( delayedCloseFinished() ), - this, TQT_SLOT( slotClosed() ) ); - connect( &d->socket, TQT_SIGNAL( readyRead() ), - this, TQT_SLOT( slotReadyRead() ) ); - connect( &d->socket, TQT_SIGNAL( error(int) ), - this, TQT_SLOT( slotError(int) ) ); - connect( &d->socket, TQT_SIGNAL( bytesWritten(int) ), - this, TQT_SLOT( slotBytesWritten(int) ) ); - - d->idleTimer = startTimer( 0 ); -} - -/*! - Destroys the TQHttp object. If there is an open connection, it is - closed. -*/ -TQHttp::~TQHttp() -{ - abort(); - delete d; -} - -/*! - \enum TQHttp::State - - This enum is used to specify the state the client is in: - - \value Unconnected There is no connection to the host. - \value HostLookup A host name lookup is in progress. - \value Connecting An attempt to connect to the host is in progress. - \value Sending The client is sending its request to the server. - \value Reading The client's request has been sent and the client - is reading the server's response. - \value Connected The connection to the host is open, but the client is - neither sending a request, nor waiting for a response. - \value Closing The connection is closing down, but is not yet - closed. (The state will be \c Unconnected when the connection is - closed.) - - \sa stateChanged() state() -*/ - -/*! \enum TQHttp::Error - - This enum identifies the error that occurred. - - \value NoError No error occurred. - \value HostNotFound The host name lookup failed. - \value ConnectionRefused The server refused the connection. - \value UnexpectedClose The server closed the connection unexpectedly. - \value InvalidResponseHeader The server sent an invalid response header. - \value WrongContentLength The client could not read the content correctly - because an error with respect to the content length occurred. - \value Aborted The request was aborted with abort(). - \value UnknownError An error other than those specified above - occurred. - - \sa error() -*/ - -/*! - \fn void TQHttp::stateChanged( int state ) - - This signal is emitted when the state of the TQHttp object changes. - The argument \a state is the new state of the connection; it is - one of the \l State values. - - This usually happens when a request is started, but it can also - happen when the server closes the connection or when a call to - closeConnection() succeeded. - - \sa get() post() head() request() closeConnection() state() State -*/ - -/*! - \fn void TQHttp::responseHeaderReceived( const TQHttpResponseHeader& resp ) - - This signal is emitted when the HTTP header of a server response - is available. The header is passed in \a resp. - - \sa get() post() head() request() readyRead() -*/ - -/*! - \fn void TQHttp::readyRead( const TQHttpResponseHeader& resp ) - - This signal is emitted when there is new response data to read. - - If you specified a tqdevice in the request where the data should be - written to, then this signal is \e not emitted; instead the data - is written directly to the tqdevice. - - The response header is passed in \a resp. - - You can read the data with the readAll() or readBlock() functions - - This signal is useful if you want to process the data in chunks as - soon as it becomes available. If you are only interested in the - complete data, just connect to the requestFinished() signal and - read the data then instead. - - \sa get() post() request() readAll() readBlock() bytesAvailable() -*/ - -/*! - \fn void TQHttp::dataSendProgress( int done, int total ) - - This signal is emitted when this object sends data to a HTTP - server to inform it about the progress of the upload. - - \a done is the amount of data that has already arrived and \a - total is the total amount of data. It is possible that the total - amount of data that should be transferred cannot be determined, in - which case \a total is 0.(If you connect to a TQProgressBar, the - progress bar shows a busy indicator if the total is 0). - - \warning \a done and \a total are not necessarily the size in - bytes, since for large files these values might need to be - "scaled" to avoid overflow. - - \sa dataReadProgress() post() request() TQProgressBar::setProgress() -*/ - -/*! - \fn void TQHttp::dataReadProgress( int done, int total ) - - This signal is emitted when this object reads data from a HTTP - server to indicate the current progress of the download. - - \a done is the amount of data that has already arrived and \a - total is the total amount of data. It is possible that the total - amount of data that should be transferred cannot be determined, in - which case \a total is 0.(If you connect to a TQProgressBar, the - progress bar shows a busy indicator if the total is 0). - - \warning \a done and \a total are not necessarily the size in - bytes, since for large files these values might need to be - "scaled" to avoid overflow. - - \sa dataSendProgress() get() post() request() TQProgressBar::setProgress() -*/ - -/*! - \fn void TQHttp::requestStarted( int id ) - - This signal is emitted when processing the request identified by - \a id starts. - - \sa requestFinished() done() -*/ - -/*! - \fn void TQHttp::requestFinished( int id, bool error ) - - This signal is emitted when processing the request identified by - \a id has finished. \a error is TRUE if an error occurred during - the processing; otherwise \a error is FALSE. - - \sa requestStarted() done() error() errorString() -*/ - -/*! - \fn void TQHttp::done( bool error ) - - This signal is emitted when the last pending request has finished; - (it is emitted after the last request's requestFinished() signal). - \a error is TRUE if an error occurred during the processing; - otherwise \a error is FALSE. - - \sa requestFinished() error() errorString() -*/ - -/*! - Aborts the current request and deletes all scheduled requests. - - For the current request, the requestFinished() signal with the \c - error argument \c TRUE is emitted. For all other requests that are - affected by the abort(), no Q_SIGNALS are emitted. - - Since this slot also deletes the scheduled requests, there are no - requests left and the done() signal is emitted (with the \c error - argument \c TRUE). - - \sa clearPendingRequests() -*/ -void TQHttp::abort() -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r == 0 ) - return; - - finishedWithError( tr("Request aborted"), Aborted ); - clearPendingRequests(); - d->socket.clearPendingData(); - close(); -} - -/*! - Returns the number of bytes that can be read from the response - content at the moment. - - \sa get() post() request() readyRead() readBlock() readAll() -*/ -TQ_ULONG TQHttp::bytesAvailable() const -{ -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp::bytesAvailable(): %d bytes", (int)d->rba.size() ); -#endif - return d->rba.size(); -} - -/*! - Reads \a maxlen bytes from the response content into \a data and - returns the number of bytes read. Returns -1 if an error occurred. - - \sa get() post() request() readyRead() bytesAvailable() readAll() -*/ -TQ_LONG TQHttp::readBlock( char *data, TQ_ULONG maxlen ) -{ - if ( data == 0 && maxlen != 0 ) { -#if defined(TQT_CHECK_NULL) - qWarning( "TQHttp::readBlock: Null pointer error" ); -#endif - return -1; - } - if ( maxlen >= d->rba.size() ) - maxlen = d->rba.size(); - d->rba.consumeBytes( maxlen, data ); - - d->bytesDone += maxlen; -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp::readBlock(): read %d bytes (%d bytes done)", (int)maxlen, d->bytesDone ); -#endif - return maxlen; -} - -/*! - Reads all the bytes from the response content and returns them. - - \sa get() post() request() readyRead() bytesAvailable() readBlock() -*/ -TQByteArray TQHttp::readAll() -{ - TQ_ULONG avail = bytesAvailable(); - TQByteArray tmp( avail ); - TQ_LONG read = readBlock( tmp.data(), avail ); - tmp.resize( read ); - return tmp; -} - -/*! - Returns the identifier of the HTTP request being executed or 0 if - there is no request being executed (i.e. they've all finished). - - \sa currentRequest() -*/ -int TQHttp::currentId() const -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r == 0 ) - return 0; - return r->id; -} - -/*! - Returns the request header of the HTTP request being executed. If - the request is one issued by setHost() or closeConnection(), it - returns an invalid request header, i.e. - TQHttpRequestHeader::isValid() returns FALSE. - - \sa currentId() -*/ -TQHttpRequestHeader TQHttp::currentRequest() const -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r != 0 && r->hasRequestHeader() ) - return r->requestHeader(); - return TQHttpRequestHeader(); -} - -/*! - Returns the TQIODevice pointer that is used as the data source of the HTTP - request being executed. If there is no current request or if the request - does not use an IO tqdevice as the data source, this function returns 0. - - This function can be used to delete the TQIODevice in the slot connected to - the requestFinished() signal. - - \sa currentDestinationDevice() post() request() -*/ -TQIODevice* TQHttp::currentSourceDevice() const -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( !r ) - return 0; - return r->sourceDevice(); -} - -/*! - Returns the TQIODevice pointer that is used as to store the data of the HTTP - request being executed. If there is no current request or if the request - does not store the data to an IO tqdevice, this function returns 0. - - This function can be used to delete the TQIODevice in the slot connected to - the requestFinished() signal. - - \sa currentDestinationDevice() get() post() request() -*/ -TQIODevice* TQHttp::currentDestinationDevice() const -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( !r ) - return 0; - return r->destinationDevice(); -} - -/*! - Returns TRUE if there are any requests scheduled that have not yet - been executed; otherwise returns FALSE. - - The request that is being executed is \e not considered as a - scheduled request. - - \sa clearPendingRequests() currentId() currentRequest() -*/ -bool TQHttp::hasPendingRequests() const -{ - return d->pending.count() > 1; -} - -/*! - Deletes all pending requests from the list of scheduled requests. - This does not affect the request that is being executed. If - you want to stop this this as well, use abort(). - - \sa hasPendingRequests() abort() -*/ -void TQHttp::clearPendingRequests() -{ - TQHttpRequest *r = 0; - if ( d->pending.count() > 0 ) - r = d->pending.take( 0 ); - d->pending.clear(); - if ( r ) - d->pending.append( r ); -} - -/*! - Sets the HTTP server that is used for requests to \a hostname on - port \a port. - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - \sa get() post() head() request() requestStarted() requestFinished() done() -*/ -int TQHttp::setHost(const TQString &hostname, TQ_UINT16 port ) -{ - return addRequest( new TQHttpSetHostRequest( hostname, port ) ); -} - -/*! - Sends a get request for \a path to the server set by setHost() or - as specified in the constructor. - - \a path must be an absolute path like \c /index.html or an - absolute URI like \c http://www.trolltech.com/index.html. - - If the IO tqdevice \a to is 0 the readyRead() signal is emitted - every time new content data is available to read. - - If the IO tqdevice \a to is not 0, the content data of the response - is written directly to the tqdevice. Make sure that the \a to - pointer is valid for the duration of the operation (it is safe to - delete it when the requestFinished() signal is emitted). - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - \sa setHost() post() head() request() requestStarted() requestFinished() done() -*/ -int TQHttp::get( const TQString& path, TQIODevice* to ) -{ - TQHttpRequestHeader header( "GET", path ); - header.setValue( "Connection", "Keep-Alive" ); - return addRequest( new TQHttpPGHRequest( header, (TQIODevice*)0, to ) ); -} - -/*! - Sends a post request for \a path to the server set by setHost() or - as specified in the constructor. - - \a path must be an absolute path like \c /index.html or an - absolute URI like \c http://www.trolltech.com/index.html. - - The incoming data comes via the \a data IO tqdevice. - - If the IO tqdevice \a to is 0 the readyRead() signal is emitted - every time new content data is available to read. - - If the IO tqdevice \a to is not 0, the content data of the response - is written directly to the tqdevice. Make sure that the \a to - pointer is valid for the duration of the operation (it is safe to - delete it when the requestFinished() signal is emitted). - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - \sa setHost() get() head() request() requestStarted() requestFinished() done() -*/ -int TQHttp::post( const TQString& path, TQIODevice* data, TQIODevice* to ) -{ - TQHttpRequestHeader header( "POST", path ); - header.setValue( "Connection", "Keep-Alive" ); - return addRequest( new TQHttpPGHRequest( header, data, to ) ); -} - -/*! - \overload - - \a data is used as the content data of the HTTP request. -*/ -int TQHttp::post( const TQString& path, const TQByteArray& data, TQIODevice* to ) -{ - TQHttpRequestHeader header( "POST", path ); - header.setValue( "Connection", "Keep-Alive" ); - return addRequest( new TQHttpPGHRequest( header, new TQByteArray(data), to ) ); -} - -/*! - Sends a header request for \a path to the server set by setHost() - or as specified in the constructor. - - \a path must be an absolute path like \c /index.html or an - absolute URI like \c http://www.trolltech.com/index.html. - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - \sa setHost() get() post() request() requestStarted() requestFinished() done() -*/ -int TQHttp::head( const TQString& path ) -{ - TQHttpRequestHeader header( "HEAD", path ); - header.setValue( "Connection", "Keep-Alive" ); - return addRequest( new TQHttpPGHRequest( header, (TQIODevice*)0, 0 ) ); -} - -/*! - Sends a request to the server set by setHost() or as specified in - the constructor. Uses the \a header as the HTTP request header. - You are responsible for setting up a header that is appropriate - for your request. - - The incoming data comes via the \a data IO tqdevice. - - If the IO tqdevice \a to is 0 the readyRead() signal is emitted - every time new content data is available to read. - - If the IO tqdevice \a to is not 0, the content data of the response - is written directly to the tqdevice. Make sure that the \a to - pointer is valid for the duration of the operation (it is safe to - delete it when the requestFinished() signal is emitted). - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - \sa setHost() get() post() head() requestStarted() requestFinished() done() -*/ -int TQHttp::request( const TQHttpRequestHeader &header, TQIODevice *data, TQIODevice *to ) -{ - return addRequest( new TQHttpNormalRequest( header, data, to ) ); -} - -/*! - \overload - - \a data is used as the content data of the HTTP request. -*/ -int TQHttp::request( const TQHttpRequestHeader &header, const TQByteArray &data, TQIODevice *to ) -{ - return addRequest( new TQHttpNormalRequest( header, new TQByteArray(data), to ) ); -} - -/*! - Closes the connection; this is useful if you have a keep-alive - connection and want to close it. - - For the requests issued with get(), post() and head(), TQHttp sets - the connection to be keep-alive. You can also do this using the - header you pass to the request() function. TQHttp only closes the - connection to the HTTP server if the response header requires it - to do so. - - The function does not block and returns immediately. The request - is scheduled, and its execution is performed asynchronously. The - function returns a unique identifier which is passed by - requestStarted() and requestFinished(). - - When the request is started the requestStarted() signal is - emitted. When it is finished the requestFinished() signal is - emitted. - - If you want to close the connection immediately, you have to use - abort() instead. - - \sa stateChanged() abort() requestStarted() requestFinished() done() -*/ -int TQHttp::closeConnection() -{ - return addRequest( new TQHttpCloseRequest() ); -} - -int TQHttp::addRequest( TQHttpRequest *req ) -{ - d->pending.append( req ); - - if ( d->pending.count() == 1 ) - // don't emit the requestStarted() signal before the id is returned - TQTimer::singleShot( 0, this, TQT_SLOT(startNextRequest()) ); - - return req->id; -} - -void TQHttp::startNextRequest() -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r == 0 ) - return; - - d->error = NoError; - d->errorString = tr( "Unknown error" ); - - if ( bytesAvailable() ) - readAll(); // clear the data - emit requestStarted( r->id ); - r->start( this ); -} - -void TQHttp::sendRequest() -{ - if ( d->hostname.isNull() ) { - finishedWithError( tr("No server set to connect to"), UnknownError ); - return; - } - - killIdleTimer(); - - // Do we need to setup a new connection or can we reuse an - // existing one ? - if ( d->socket.peerName() != d->hostname || d->socket.peerPort() != d->port - || d->socket.state() != TQSocket::Connection ) { - setState( TQHttp::Connecting ); - d->socket.connectToHost( d->hostname, d->port ); - } else { - slotConnected(); - } - -} - -void TQHttp::finishedWithSuccess() -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r == 0 ) - return; - - emit requestFinished( r->id, FALSE ); - d->pending.removeFirst(); - if ( d->pending.isEmpty() ) { - emit done( FALSE ); - } else { - startNextRequest(); - } -} - -void TQHttp::finishedWithError( const TQString& detail, int errorCode ) -{ - TQHttpRequest *r = d->pending.getFirst(); - if ( r == 0 ) - return; - - d->error = (Error)errorCode; - d->errorString = detail; - emit requestFinished( r->id, TRUE ); - - d->pending.clear(); - emit done( TRUE ); -} - -void TQHttp::slotClosed() -{ - if ( d->state == Closing ) - return; - - if ( d->state == Reading ) { - if ( d->response.hasKey( "content-length" ) ) { - // We got Content-Length, so did we get all bytes? - if ( d->bytesDone+bytesAvailable() != d->response.contentLength() ) { - finishedWithError( tr("Wrong content length"), WrongContentLength ); - } - } - } else if ( d->state == Connecting || d->state == Sending ) { - finishedWithError( tr("Server closed connection unexpectedly"), UnexpectedClose ); - } - - d->postDevice = 0; - setState( Closing ); - d->idleTimer = startTimer( 0 ); -} - -void TQHttp::slotConnected() -{ - if ( d->state != Sending ) { - d->bytesDone = 0; - setState( Sending ); - } - - TQString str = d->header.toString(); - d->bytesTotal = str.length(); - d->socket.writeBlock( str.latin1(), d->bytesTotal ); -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp: write request header:\n---{\n%s}---", str.latin1() ); -#endif - - if ( d->postDevice ) { - d->bytesTotal += d->postDevice->size(); - } else { - d->bytesTotal += d->buffer.size(); - d->socket.writeBlock( d->buffer.data(), d->buffer.size() ); - d->buffer = TQByteArray(); // save memory - } -} - -void TQHttp::slotError( int err ) -{ - d->postDevice = 0; - - if ( d->state == Connecting || d->state == Reading || d->state == Sending ) { - switch ( err ) { - case TQSocket::ErrConnectionRefused: - finishedWithError( tr("Connection refused"), ConnectionRefused ); - break; - case TQSocket::ErrHostNotFound: - finishedWithError( tr("Host %1 not found").arg(d->socket.peerName()), HostNotFound ); - break; - default: - finishedWithError( tr("HTTP request failed"), UnknownError ); - break; - } - } - - close(); -} - -void TQHttp::slotBytesWritten( int written ) -{ - d->bytesDone += written; - emit dataSendProgress( d->bytesDone, d->bytesTotal ); - - if ( !d->postDevice ) - return; - - if ( d->socket.bytesToWrite() == 0 ) { - int max = TQMIN( 4096, d->postDevice->size() - d->postDevice->at() ); - TQByteArray arr( max ); - - int n = d->postDevice->readBlock( arr.data(), max ); - if ( n != max ) { - qWarning("Could not read enough bytes from the tqdevice"); - close(); - return; - } - if ( d->postDevice->atEnd() ) { - d->postDevice = 0; - } - - d->socket.writeBlock( arr.data(), max ); - } -} - -void TQHttp::slotReadyRead() -{ - if ( d->state != Reading ) { - setState( Reading ); - d->buffer = TQByteArray(); - d->readHeader = TRUE; - d->headerStr = ""; - d->bytesDone = 0; - d->chunkedSize = -1; - } - - while ( d->readHeader ) { - bool end = FALSE; - TQString tmp; - while ( !end && d->socket.canReadLine() ) { - tmp = d->socket.readLine(); - if ( tmp == "\r\n" || tmp == "\n" ) - end = TRUE; - else - d->headerStr += tmp; - } - - if ( !end ) - return; - -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp: read response header:\n---{\n%s}---", d->headerStr.latin1() ); -#endif - d->response = TQHttpResponseHeader( d->headerStr ); - d->headerStr = ""; -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp: read response header:\n---{\n%s}---", d->response.toString().latin1() ); -#endif - // Check header - if ( !d->response.isValid() ) { - finishedWithError( tr("Invalid HTTP response header"), InvalidResponseHeader ); - close(); - return; - } - - // The 100-continue header is ignored, because when using the - // POST method, we send both the request header and data in - // one chunk. - if (d->response.statusCode() != 100) { - d->readHeader = FALSE; - if ( d->response.hasKey( "transfer-encoding" ) && - d->response.value( "transfer-encoding" ).lower().contains( "chunked" ) ) - d->chunkedSize = 0; - - emit responseHeaderReceived( d->response ); - } - } - - if ( !d->readHeader ) { - bool everythingRead = FALSE; - - if ( currentRequest().method() == "HEAD" ) { - everythingRead = TRUE; - } else { - TQ_ULONG n = d->socket.bytesAvailable(); - TQByteArray *arr = 0; - if ( d->chunkedSize != -1 ) { - // transfer-encoding is chunked - for ( ;; ) { - // get chunk size - if ( d->chunkedSize == 0 ) { - if ( !d->socket.canReadLine() ) - break; - TQString sizeString = d->socket.readLine(); - int tPos = sizeString.find( ';' ); - if ( tPos != -1 ) - sizeString.truncate( tPos ); - bool ok; - d->chunkedSize = sizeString.toInt( &ok, 16 ); - if ( !ok ) { - finishedWithError( tr("Invalid HTTP chunked body"), WrongContentLength ); - close(); - delete arr; - return; - } - if ( d->chunkedSize == 0 ) // last-chunk - d->chunkedSize = -2; - } - - // read trailer - while ( d->chunkedSize == -2 && d->socket.canReadLine() ) { - TQString read = d->socket.readLine(); - if ( read == "\r\n" || read == "\n" ) - d->chunkedSize = -1; - } - if ( d->chunkedSize == -1 ) { - everythingRead = TRUE; - break; - } - - // make sure that you can read the terminating CRLF, - // otherwise wait until next time... - n = d->socket.bytesAvailable(); - if ( n == 0 ) - break; - if ( (TQ_LONG)n == d->chunkedSize || (TQ_LONG)n == d->chunkedSize+1 ) { - n = d->chunkedSize - 1; - if ( n == 0 ) - break; - } - - // read data - uint toRead = TQMIN( (TQ_LONG)n, (d->chunkedSize < 0 ? (TQ_LONG)n : d->chunkedSize) ); - if ( !arr ) - arr = new TQByteArray( 0 ); - uint oldArrSize = arr->size(); - arr->resize( oldArrSize + toRead ); - TQ_LONG read = d->socket.readBlock( arr->data()+oldArrSize, toRead ); - arr->resize( oldArrSize + read ); - - d->chunkedSize -= read; - - if ( d->chunkedSize == 0 && n - read >= 2 ) { - // read terminating CRLF - char tmp[2]; - d->socket.readBlock( tmp, 2 ); - if ( tmp[0] != '\r' || tmp[1] != '\n' ) { - finishedWithError( tr("Invalid HTTP chunked body"), WrongContentLength ); - close(); - return; - } - } - } - } else if ( d->response.hasContentLength() ) { - n = TQMIN( d->response.contentLength() - d->bytesDone, n ); - if ( n > 0 ) { - arr = new TQByteArray( n ); - TQ_LONG read = d->socket.readBlock( arr->data(), n ); - arr->resize( read ); - } - if ( d->bytesDone + bytesAvailable() + n == d->response.contentLength() ) - everythingRead = TRUE; - } else if ( n > 0 ) { - // workaround for VC++ bug - TQByteArray temp = TQT_TQBYTEARRAY_OBJECT(d->socket.readAll()); - arr = new TQByteArray( temp ); - } - - if ( arr ) { - n = arr->size(); - if ( d->toDevice ) { - d->toDevice->writeBlock( arr->data(), n ); - delete arr; - d->bytesDone += n; -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp::slotReadyRead(): read %ld bytes (%d bytes done)", n, d->bytesDone ); -#endif - if ( d->response.hasContentLength() ) - emit dataReadProgress( d->bytesDone, d->response.contentLength() ); - else - emit dataReadProgress( d->bytesDone, 0 ); - } else { - d->rba.append( arr ); -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp::slotReadyRead(): read %ld bytes (%ld bytes done)", n, d->bytesDone + bytesAvailable() ); -#endif - if ( d->response.hasContentLength() ) - emit dataReadProgress( d->bytesDone + bytesAvailable(), d->response.contentLength() ); - else - emit dataReadProgress( d->bytesDone + bytesAvailable(), 0 ); - emit readyRead( d->response ); - } - } - } - - if ( everythingRead ) { - // Handle "Connection: close" - if ( d->response.value("connection").lower() == "close" ) { - close(); - } else { - setState( Connected ); - // Start a timer, so that we emit the keep alive signal - // "after" this method returned. - d->idleTimer = startTimer( 0 ); - } - } - } -} - -/*! - Returns the current state of the object. When the state changes, - the stateChanged() signal is emitted. - - \sa State stateChanged() -*/ -TQHttp::State TQHttp::state() const -{ - return d->state; -} - -/*! - Returns the last error that occurred. This is useful to find out - what happened when receiving a requestFinished() or a done() - signal with the \c error argument \c TRUE. - - If you start a new request, the error status is reset to \c NoError. -*/ -TQHttp::Error TQHttp::error() const -{ - return d->error; -} - -/*! - Returns a human-readable description of the last error that - occurred. This is useful to present a error message to the user - when receiving a requestFinished() or a done() signal with the \c - error argument \c TRUE. -*/ -TQString TQHttp::errorString() const -{ - return d->errorString; -} - -/*! \reimp -*/ -void TQHttp::timerEvent( TQTimerEvent *e ) -{ - if ( e->timerId() == d->idleTimer ) { - killTimer( d->idleTimer ); - d->idleTimer = 0; - - if ( d->state == Connected ) { - finishedWithSuccess(); - } else if ( d->state != Unconnected ) { - setState( Unconnected ); - finishedWithSuccess(); - } - } else { - TQObject::timerEvent( e ); - } -} - -void TQHttp::killIdleTimer() -{ - killTimer( d->idleTimer ); - d->idleTimer = 0; -} - -void TQHttp::setState( int s ) -{ -#if defined(TQHTTP_DEBUG) - qDebug( "TQHttp state changed %d -> %d", d->state, s ); -#endif - d->state = (State)s; - emit stateChanged( s ); -} - -void TQHttp::close() -{ - // If no connection is open -> ignore - if ( d->state == Closing || d->state == Unconnected ) - return; - - d->postDevice = 0; - setState( Closing ); - - // Already closed ? - if ( !d->socket.isOpen() ) { - d->idleTimer = startTimer( 0 ); - } else { - // Close now. - d->socket.close(); - - // Did close succeed immediately ? - if ( d->socket.state() == TQSocket::Idle ) { - // Prepare to emit the requestFinished() signal. - d->idleTimer = startTimer( 0 ); - } - } -} - -/********************************************************************** - * - * TQHttp implementation of the TQNetworkProtocol interface - * - *********************************************************************/ -/*! \reimp -*/ -int TQHttp::supportedOperations() const -{ - return OpGet | OpPut; -} - -/*! \reimp -*/ -void TQHttp::operationGet( TQNetworkOperation *op ) -{ - connect( this, TQT_SIGNAL(readyRead(const TQHttpResponseHeader&)), - this, TQT_SLOT(clientReply(const TQHttpResponseHeader&)) ); - connect( this, TQT_SIGNAL(done(bool)), - this, TQT_SLOT(clientDone(bool)) ); - connect( this, TQT_SIGNAL(stateChanged(int)), - this, TQT_SLOT(clientStateChanged(int)) ); - - bytesRead = 0; - op->setState( StInProgress ); - TQUrl u( operationInProgress()->arg( 0 ) ); - TQHttpRequestHeader header( "GET", u.encodedPathAndQuery(), 1, 0 ); - header.setValue( "Host", u.host() ); - setHost( u.host(), u.port() != -1 ? u.port() : 80 ); - request( header ); -} - -/*! \reimp -*/ -void TQHttp::operationPut( TQNetworkOperation *op ) -{ - connect( this, TQT_SIGNAL(readyRead(const TQHttpResponseHeader&)), - this, TQT_SLOT(clientReply(const TQHttpResponseHeader&)) ); - connect( this, TQT_SIGNAL(done(bool)), - this, TQT_SLOT(clientDone(bool)) ); - connect( this, TQT_SIGNAL(stateChanged(int)), - this, TQT_SLOT(clientStateChanged(int)) ); - - bytesRead = 0; - op->setState( StInProgress ); - TQUrl u( operationInProgress()->arg( 0 ) ); - TQHttpRequestHeader header( "POST", u.encodedPathAndQuery(), 1, 0 ); - header.setValue( "Host", u.host() ); - setHost( u.host(), u.port() != -1 ? u.port() : 80 ); - request( header, op->rawArg(1) ); -} - -void TQHttp::clientReply( const TQHttpResponseHeader &rep ) -{ - TQNetworkOperation *op = operationInProgress(); - if ( op ) { - if ( rep.statusCode() >= 400 && rep.statusCode() < 600 ) { - op->setState( StFailed ); - op->setProtocolDetail( - TQString("%1 %2").arg(rep.statusCode()).arg(rep.reasonPhrase()) - ); - switch ( rep.statusCode() ) { - case 401: - case 403: - case 405: - op->setErrorCode( ErrPermissionDenied ); - break; - case 404: - op->setErrorCode(ErrFileNotExisting ); - break; - default: - if ( op->operation() == OpGet ) - op->setErrorCode( ErrGet ); - else - op->setErrorCode( ErrPut ); - break; - } - } - // ### In cases of an error, should we still emit the data() Q_SIGNALS? - if ( op->operation() == OpGet && bytesAvailable() > 0 ) { - TQByteArray ba = readAll(); - emit data( ba, op ); - bytesRead += ba.size(); - if ( rep.hasContentLength() ) { - emit dataTransferProgress( bytesRead, rep.contentLength(), op ); - } - } - } -} - -void TQHttp::clientDone( bool err ) -{ - disconnect( this, TQT_SIGNAL(readyRead(const TQHttpResponseHeader&)), - this, TQT_SLOT(clientReply(const TQHttpResponseHeader&)) ); - disconnect( this, TQT_SIGNAL(done(bool)), - this, TQT_SLOT(clientDone(bool)) ); - disconnect( this, TQT_SIGNAL(stateChanged(int)), - this, TQT_SLOT(clientStateChanged(int)) ); - - if ( err ) { - TQNetworkOperation *op = operationInProgress(); - if ( op ) { - op->setState( TQNetworkProtocol::StFailed ); - op->setProtocolDetail( errorString() ); - switch ( error() ) { - case ConnectionRefused: - op->setErrorCode( ErrHostNotFound ); - break; - case HostNotFound: - op->setErrorCode( ErrHostNotFound ); - break; - default: - if ( op->operation() == OpGet ) - op->setErrorCode( ErrGet ); - else - op->setErrorCode( ErrPut ); - break; - } - emit finished( op ); - } - } else { - TQNetworkOperation *op = operationInProgress(); - if ( op ) { - if ( op->state() != StFailed ) { - op->setState( TQNetworkProtocol::StDone ); - op->setErrorCode( TQNetworkProtocol::NoError ); - } - emit finished( op ); - } - } - -} - -void TQHttp::clientStateChanged( int state ) -{ - if ( url() ) { - switch ( (State)state ) { - case Connecting: - emit connectionStateChanged( ConHostFound, tr( "Host %1 found" ).arg( url()->host() ) ); - break; - case Sending: - emit connectionStateChanged( ConConnected, tr( "Connected to host %1" ).arg( url()->host() ) ); - break; - case Unconnected: - emit connectionStateChanged( ConClosed, tr( "Connection to %1 closed" ).arg( url()->host() ) ); - break; - default: - break; - } - } else { - switch ( (State)state ) { - case Connecting: - emit connectionStateChanged( ConHostFound, tr( "Host found" ) ); - break; - case Sending: - emit connectionStateChanged( ConConnected, tr( "Connected to host" ) ); - break; - case Unconnected: - emit connectionStateChanged( ConClosed, tr( "Connection closed" ) ); - break; - default: - break; - } - } -} - -#endif diff --git a/tqtinterface/qt4/src/network/tqhttp.h b/tqtinterface/qt4/src/network/tqhttp.h deleted file mode 100644 index fb20d4f..0000000 --- a/tqtinterface/qt4/src/network/tqhttp.h +++ /dev/null @@ -1,278 +0,0 @@ -/**************************************************************************** -** -** Definition of TQHttp and related classes. -** -** Created : 970521 -** -** Copyright (C) 1997-2008 Trolltech ASA. All rights reserved. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQHTTP_H -#define TQHTTP_H - -#ifndef TQT_H -#include "tqobject.h" -#include "tqnetworkprotocol.h" -#include "tqstringlist.h" -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_HTTP -#define TQM_TEMPLATE_EXTERN_HTTP -#else -#define TQM_EXPORT_HTTP TQ_EXPORT -#define TQM_TEMPLATE_EXTERN_HTTP TQ_TEMPLATE_EXTERN -#endif - -#ifndef TQT_NO_NETWORKPROTOCOL_HTTP - -class TQSocket; -class TQTimerEvent; -class TQTextStream; -class TQIODevice; - -class TQHttpPrivate; -class TQHttpRequest; - -class TQM_EXPORT_HTTP TQHttpHeader -{ -public: - TQHttpHeader(); - TQHttpHeader( const TQHttpHeader& header ); - TQHttpHeader( const TQString& str ); - virtual ~TQHttpHeader(); - - TQHttpHeader& operator=( const TQHttpHeader& h ); - - TQString value( const TQString& key ) const; - void setValue( const TQString& key, const TQString& value ); - void removeValue( const TQString& key ); - - TQStringList keys() const; - bool hasKey( const TQString& key ) const; - - bool hasContentLength() const; - uint contentLength() const; - void setContentLength( int len ); - - bool hasContentType() const; - TQString contentType() const; - void setContentType( const TQString& type ); - - virtual TQString toString() const; - bool isValid() const; - - virtual int majorVersion() const = 0; - virtual int minorVersion() const = 0; - -protected: - virtual bool parseLine( const TQString& line, int number ); - bool parse( const TQString& str ); - void setValid( bool ); - -private: - TQMap<TQString,TQString> values; - bool valid; -}; - -class TQM_EXPORT_HTTP TQHttpResponseHeader : public TQHttpHeader -{ -private: - TQHttpResponseHeader( int code, const TQString& text = TQString::null, int majorVer = 1, int minorVer = 1 ); - TQHttpResponseHeader( const TQString& str ); - - void setqStatusLine( int code, const TQString& text = TQString::null, int majorVer = 1, int minorVer = 1 ); - -public: - TQHttpResponseHeader(); - TQHttpResponseHeader( const TQHttpResponseHeader& header ); - - int statusCode() const; - TQString reasonPhrase() const; - - int majorVersion() const; - int minorVersion() const; - - TQString toString() const; - -protected: - bool parseLine( const TQString& line, int number ); - -private: - int statCode; - TQString reasonPhr; - int majVer; - int minVer; - - friend class TQHttp; -}; - -class TQM_EXPORT_HTTP TQHttpRequestHeader : public TQHttpHeader -{ -public: - TQHttpRequestHeader(); - TQHttpRequestHeader( const TQString& method, const TQString& path, int majorVer = 1, int minorVer = 1 ); - TQHttpRequestHeader( const TQHttpRequestHeader& header ); - TQHttpRequestHeader( const TQString& str ); - - void setRequest( const TQString& method, const TQString& path, int majorVer = 1, int minorVer = 1 ); - - TQString method() const; - TQString path() const; - - int majorVersion() const; - int minorVersion() const; - - TQString toString() const; - -protected: - bool parseLine( const TQString& line, int number ); - -private: - TQString m; - TQString p; - int majVer; - int minVer; -}; - -class TQM_EXPORT_HTTP TQHttp : public TQNetworkProtocol -{ - Q_OBJECT - TQ_OBJECT - -public: - TQHttp(); - TQHttp( TQObject* parent, const char* name = 0 ); // ### TQt 4.0: make parent=0 and get rid of the TQHttp() constructor - TQHttp( const TQString &hostname, TQ_UINT16 port=80, TQObject* parent=0, const char* name = 0 ); - virtual ~TQHttp(); - - int supportedOperations() const; - - enum State { Unconnected, HostLookup, Connecting, Sending, Reading, Connected, Closing }; - enum Error { - NoError, - UnknownError, - HostNotFound, - ConnectionRefused, - UnexpectedClose, - InvalidResponseHeader, - WrongContentLength, - Aborted - }; - - int setHost(const TQString &hostname, TQ_UINT16 port=80 ); - - int get( const TQString& path, TQIODevice* to=0 ); - int post( const TQString& path, TQIODevice* data, TQIODevice* to=0 ); - int post( const TQString& path, const TQByteArray& data, TQIODevice* to=0 ); - int head( const TQString& path ); - int request( const TQHttpRequestHeader &header, TQIODevice *tqdevice=0, TQIODevice *to=0 ); - int request( const TQHttpRequestHeader &header, const TQByteArray &data, TQIODevice *to=0 ); - - int closeConnection(); - - TQ_ULONG bytesAvailable() const; - TQ_LONG readBlock( char *data, TQ_ULONG maxlen ); - TQByteArray readAll(); - - int currentId() const; - TQIODevice* currentSourceDevice() const; - TQIODevice* currentDestinationDevice() const; - TQHttpRequestHeader currentRequest() const; - bool hasPendingRequests() const; - void clearPendingRequests(); - - State state() const; - - Error error() const; - TQString errorString() const; - -public Q_SLOTS: - void abort(); - -Q_SIGNALS: - void stateChanged( int ); - void responseHeaderReceived( const TQHttpResponseHeader& resp ); - void readyRead( const TQHttpResponseHeader& resp ); - void dataSendProgress( int, int ); - void dataReadProgress( int, int ); - - void requestStarted( int ); - void requestFinished( int, bool ); - void done( bool ); - -protected: - void operationGet( TQNetworkOperation *op ); - void operationPut( TQNetworkOperation *op ); - - void timerEvent( TQTimerEvent * ); - -private Q_SLOTS: - void clientReply( const TQHttpResponseHeader &rep ); - void clientDone( bool ); - void clientStateChanged( int ); - - void startNextRequest(); - void slotReadyRead(); - void slotConnected(); - void slotError( int ); - void slotClosed(); - void slotBytesWritten( int ); - -private: - TQHttpPrivate *d; - void *unused; // ### TQt 4.0: remove this (in for binary compatibility) - int bytesRead; - - int addRequest( TQHttpRequest * ); - void sendRequest(); - void finishedWithSuccess(); - void finishedWithError( const TQString& detail, int errorCode ); - - void killIdleTimer(); - - void init(); - void setState( int ); - void close(); - - friend class TQHttpNormalRequest; - friend class TQHttpSetHostRequest; - friend class TQHttpCloseRequest; - friend class TQHttpPGHRequest; -}; - -#define TQ_DEFINED_TQHTTP -#include "tqwinexport.h" -#endif -#endif diff --git a/tqtinterface/qt4/src/network/tqnetwork.cpp b/tqtinterface/qt4/src/network/tqnetwork.cpp deleted file mode 100644 index 14cdda5..0000000 --- a/tqtinterface/qt4/src/network/tqnetwork.cpp +++ /dev/null @@ -1,71 +0,0 @@ -/**************************************************************************** -** -** Implementation of qInitNetworkProtocols function. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqnetwork.h" - -#ifndef TQT_NO_NETWORK - -#include "tqnetworkprotocol.h" - -// protocols -#include "tqftp.h" -#include "tqhttp.h" - -/*! \file tqnetwork.h */ -/*! - \relates TQUrlOperator - - This function registers the network protocols for FTP and HTTP. - You must call this function before you use TQUrlOperator for - these protocols. - - This function is declared in \l tqnetwork.h. -*/ -void qInitNetworkProtocols() -{ -#ifndef TQT_NO_NETWORKPROTOCOL_FTP - TQNetworkProtocol::registerNetworkProtocol( "ftp", new TQNetworkProtocolFactory< TQFtp > ); -#endif -#ifndef TQT_NO_NETWORKPROTOCOL_HTTP - TQNetworkProtocol::registerNetworkProtocol( "http", new TQNetworkProtocolFactory< TQHttp > ); -#endif -} - -#endif // TQT_NO_NETWORK diff --git a/tqtinterface/qt4/src/network/tqnetwork.h b/tqtinterface/qt4/src/network/tqnetwork.h deleted file mode 100644 index 131f40f..0000000 --- a/tqtinterface/qt4/src/network/tqnetwork.h +++ /dev/null @@ -1,60 +0,0 @@ -/**************************************************************************** -** -** Definition of qInitNetworkProtocols function. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQNETWORK_H -#define TQNETWORK_H - -#ifndef TQT_H -#include "tqglobal.h" -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_NETWORK -#else -#define TQM_EXPORT_NETWORK TQ_EXPORT -#endif - -#ifndef TQT_NO_NETWORK - -TQM_EXPORT_NETWORK void qInitNetworkProtocols(); - -#endif - -#endif diff --git a/tqtinterface/qt4/src/network/tqserversocket.cpp b/tqtinterface/qt4/src/network/tqserversocket.cpp deleted file mode 100644 index 02a15b0..0000000 --- a/tqtinterface/qt4/src/network/tqserversocket.cpp +++ /dev/null @@ -1,297 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQServerSocket class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqserversocket.h" - -#ifndef TQT_NO_NETWORK - -#include "tqsocketnotifier.h" - -class TQServerSocketPrivate { -public: - TQServerSocketPrivate(): s(0), n(0) {} - ~TQServerSocketPrivate() { delete n; delete s; } - TQSocketDevice *s; - TQSocketNotifier *n; -}; - - -/*! - \class TQServerSocket tqserversocket.h - \brief The TQServerSocket class provides a TCP-based server. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class is a convenience class for accepting incoming TCP - connections. You can specify the port or have TQServerSocket pick - one, and listen on just one address or on all the machine's - addresses. - - Using the API is very simple: subclass TQServerSocket, call the - constructor of your choice, and implement newConnection() to - handle new incoming connections. There is nothing more to do. - - (Note that due to lack of support in the underlying APIs, - TQServerSocket cannot accept or reject connections conditionally.) - - \sa TQSocket, TQSocketDevice, TQHostAddress, TQSocketNotifier -*/ - - -/*! - Creates a server socket object, that will serve the given \a port - on all the addresses of this host. If \a port is 0, TQServerSocket - will pick a suitable port in a system-dependent manner. Use \a - backlog to specify how many pending connections the server can - have. - - The \a parent and \a name arguments are passed on to the TQObject - constructor. - - \warning On Tru64 Unix systems a value of 0 for \a backlog means - that you don't accept any connections at all; you should specify a - value larger than 0. -*/ - -TQServerSocket::TQServerSocket( TQ_UINT16 port, int backlog, - TQObject *parent, const char *name ) - : TQObject( parent, name ) -{ - d = new TQServerSocketPrivate; - init( TQHostAddress(), port, backlog ); -} - - -/*! - Creates a server socket object, that will serve the given \a port - only on the given \a address. Use \a backlog to specify how many - pending connections the server can have. - - The \a parent and \a name arguments are passed on to the TQObject - constructor. - - \warning On Tru64 Unix systems a value of 0 for \a backlog means - that you don't accept any connections at all; you should specify a - value larger than 0. -*/ - -TQServerSocket::TQServerSocket( const TQHostAddress & address, TQ_UINT16 port, - int backlog, - TQObject *parent, const char *name ) - : TQObject( parent, name ) -{ - d = new TQServerSocketPrivate; - init( address, port, backlog ); -} - - -/*! - Construct an empty server socket. - - This constructor, in combination with setSocket(), allows us to - use the TQServerSocket class as a wrapper for other socket types - (e.g. Unix Domain Sockets under Unix). - - The \a parent and \a name arguments are passed on to the TQObject - constructor. - - \sa setSocket() -*/ - -TQServerSocket::TQServerSocket( TQObject *parent, const char *name ) - : TQObject( parent, name ) -{ - d = new TQServerSocketPrivate; -} - - -/*! - Returns TRUE if the construction succeeded; otherwise returns FALSE. -*/ -bool TQServerSocket::ok() const -{ - return !!d->s; -} - -/* - The common bit of the constructors. - */ -void TQServerSocket::init( const TQHostAddress & address, TQ_UINT16 port, int backlog ) -{ - d->s = new TQSocketDevice( TQSocketDevice::Stream, address.isIPv4Address() - ? TQSocketDevice::IPv4 : TQSocketDevice::IPv6, 0 ); -#if !defined(TQ_OS_WIN32) - // Under Unix, we want to be able to use the port, even if a socket on the - // same address-port is in TIME_WAIT. Under Windows this is possible anyway - // -- furthermore, the meaning of reusable is different: it means that you - // can use the same address-port for multiple listening sockets. - d->s->setAddressReusable( TRUE ); -#endif - if ( d->s->bind( address, port ) - && d->s->listen( backlog ) ) - { - d->n = new TQSocketNotifier( d->s->socket(), TQSocketNotifier::Read, - this, "accepting new connections" ); - connect( d->n, TQT_SIGNAL(activated(int)), - this, TQT_SLOT(incomingConnection(int)) ); - } else { - qWarning( "TQServerSocket: failed to bind or listen to the socket" ); - delete d->s; - d->s = 0; - } -} - - -/*! - Destroys the socket. - - This causes any backlogged connections (connections that have - reached the host, but not yet been completely set up by calling - TQSocketDevice::accept()) to be severed. - - Existing connections continue to exist; this only affects the - acceptance of new connections. -*/ -TQServerSocket::~TQServerSocket() -{ - delete d; -} - - -/*! - \fn void TQServerSocket::newConnection( int socket ) - - This pure virtual function is responsible for setting up a new - incoming connection. \a socket is the fd (file descriptor) for the - newly accepted connection. -*/ - - -void TQServerSocket::incomingConnection( int ) -{ - int fd = d->s->accept(); - if ( fd >= 0 ) - newConnection( fd ); -} - - -/*! - Returns the port number on which this server socket listens. This - is always non-zero; if you specify 0 in the constructor, - TQServerSocket will pick a non-zero port itself. ok() must be TRUE - before calling this function. - - \sa address() TQSocketDevice::port() -*/ -TQ_UINT16 TQServerSocket::port() const -{ - if ( !d || !d->s ) - return 0; - return d->s->port(); -} - - -/*! - Returns the operating system socket. -*/ -int TQServerSocket::socket() const -{ - if ( !d || !d->s ) - return -1; - - return d->s->socket(); -} - -/*! - Returns the address on which this object listens, or 0.0.0.0 if - this object listens on more than one address. ok() must be TRUE - before calling this function. - - \sa port() TQSocketDevice::address() -*/ -TQHostAddress TQServerSocket::address() const -{ - if ( !d || !d->s ) - return TQHostAddress(); - - return d->s->address(); -} - - -/*! - Returns a pointer to the internal socket tqdevice. The returned - pointer is 0 if there is no connection or pending connection. - - There is normally no need to manipulate the socket tqdevice directly - since this class does all the necessary setup for most client or - server socket applications. -*/ -TQSocketDevice *TQServerSocket::socketDevice() -{ - if ( !d ) - return 0; - - return d->s; -} - - -/*! - Sets the socket to use \a socket. bind() and listen() should - already have been called for \a socket. - - This allows us to use the TQServerSocket class as a wrapper for - other socket types (e.g. Unix Domain Sockets). -*/ -void TQServerSocket::setSocket( int socket ) -{ - delete d; - d = new TQServerSocketPrivate; - d->s = new TQSocketDevice( socket, TQSocketDevice::Stream ); - d->n = new TQSocketNotifier( d->s->socket(), TQSocketNotifier::Read, - this, "accepting new connections" ); - connect( d->n, TQT_SIGNAL(activated(int)), - this, TQT_SLOT(incomingConnection(int)) ); -} - -#endif //TQT_NO_NETWORK diff --git a/tqtinterface/qt4/src/network/tqserversocket.h b/tqtinterface/qt4/src/network/tqserversocket.h deleted file mode 100644 index d1f4a83..0000000 --- a/tqtinterface/qt4/src/network/tqserversocket.h +++ /dev/null @@ -1,95 +0,0 @@ -/**************************************************************************** -** -** Definition of TQServerSocketClass. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQSERVERSOCKET_H -#define TQSERVERSOCKET_H - -#ifndef TQT_H -#include "tqobject.h" -#include "tqhostaddress.h" -#include "tqsocketdevice.h" // ### remove or keep for users' convenience? -#endif // TQT_H -#ifndef TQT_NO_NETWORK - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_NETWORK -#else -#define TQM_EXPORT_NETWORK TQ_EXPORT -#endif - -class TQServerSocketPrivate; - - -class TQM_EXPORT_NETWORK TQServerSocket : public TQObject -{ - Q_OBJECT - TQ_OBJECT -public: - TQServerSocket( TQ_UINT16 port, int backlog = 1, - TQObject *parent=0, const char *name=0 ); - TQServerSocket( const TQHostAddress & address, TQ_UINT16 port, int backlog = 1, - TQObject *parent=0, const char *name=0 ); - TQServerSocket( TQObject *parent=0, const char *name=0 ); - virtual ~TQServerSocket(); - - bool ok() const; - - TQ_UINT16 port() const ; - - int socket() const ; - virtual void setSocket( int socket ); - - TQHostAddress address() const ; - - virtual void newConnection( int socket ) = 0; - -protected: - TQSocketDevice *socketDevice(); - -private Q_SLOTS: - void incomingConnection( int socket ); - -private: - TQServerSocketPrivate *d; - void init( const TQHostAddress & address, TQ_UINT16 port, int backlog ); -}; - -#endif // TQT_NO_NETWORK -#endif // TQSERVERSOCKET_H diff --git a/tqtinterface/qt4/src/network/tqsocket.cpp b/tqtinterface/qt4/src/network/tqsocket.cpp deleted file mode 100644 index 8e4d40a..0000000 --- a/tqtinterface/qt4/src/network/tqsocket.cpp +++ /dev/null @@ -1,1668 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQSocket class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqsocket.h" -#ifndef TQT_NO_NETWORK -#include "tqptrlist.h" -#include "tqtimer.h" -#include "tqsocketdevice.h" -#include "tqdns.h" -#include "private/tqinternal_p.h" - -#include <string.h> -#ifndef NO_ERRNO_H -#include <errno.h> -#endif - -//#define TQSOCKET_DEBUG - -/* - Perhaps this private functionality needs to be refactored. - - Comment from Robert D Gatlin (Intel): - - It would be nice to have the functionality inherent in TQSocket available - as a separate class as a standard part of the TQt library, something along - the line of: - - class TQByteBuffer : public TQIODevice { ... } - - The same class could/would be used within TQSocket for the Read/Write - buffers. - - The above class could be used in the following way(s): - - buffer.open( IO_WriteOnly | IO_Append ); - buffer.writeBlock( a ); // a = TQByteArray - buffer.close(); - - TQByteArray b; - b.resize( buffer.size() ); - buffer.open( IO_ReadOnly ); - buffer.readBlock( b.data(), b.size() ); - buffer.close(); - - But would also be useable with TQDataStream (via TQIODevice) with: - - buffer.open( IO_WriteOnly | IO_Append ); - TQDataStream is( &buffer ); - is << 100; - buffer.close(); - - buffer.open( IO_ReadOnly ); - TQDataStream os( &buffer ); - TQ_UINT32 x; - os >> x; - buffer.close(); - - The real usefulness is with any situations where data (TQByteArray) arrives - incrementally (as in TQSocket and filter case above). - - I tried using TQBuffer, but TQBuffer does not trim bytes from the front of - the buffer in cases like: - - TQBuffer buf; - buf.open( IO_ReadOnly ); - TQDataStream ds( &buf ); - TQ_INT32 x; - ds >> x; - buf.close(); - - In the above case, buf.size() will be identical before and after the - operation with TQDataStream. Based on the implementation of TQBuffer, it - does not appear well suited for this kind of operation. -*/ - -// Private class for TQSocket - -class TQSocketPrivate { -public: - TQSocketPrivate(); - ~TQSocketPrivate(); - void closeSocket(); - void close(); - void connectionClosed(); - void setSocketDevice( TQSocket *q, TQSocketDevice *tqdevice ); - - TQSocket::State state; // connection state - TQString host; // host name - TQ_UINT16 port; // host port - TQSocketDevice *socket; // connection socket - TQSocketNotifier *rsn, *wsn; // socket notifiers - TQMembuf rba; // read buffer - TQ_ULONG readBufferSize; // limit for the read buffer size - TQPtrList<TQByteArray> wba; // list of write bufs - TQHostAddress addr; // connection address - TQValueList<TQHostAddress> addresses; // alternatives looked up - TQIODevice::Offset wsize; // write total buf size - TQIODevice::Offset windex; // write index -#ifndef TQT_NO_DNS - TQDns *dns4; - TQDns *dns6; -#endif - static TQPtrList<TQSocket> sn_read_alreadyCalled; // used to avoid unwanted recursion - TQValueList<TQHostAddress> l4; - TQValueList<TQHostAddress> l6; -}; - -TQPtrList<TQSocket> TQSocketPrivate::sn_read_alreadyCalled; - -TQSocketPrivate::TQSocketPrivate() - : state(TQSocket::Idle), host(TQString::tqfromLatin1("")), port(0), - socket(0), rsn(0), wsn(0), readBufferSize(0), wsize(0), windex(0) -{ -#ifndef TQT_NO_DNS - dns4 = 0; - dns6 = 0; -#endif - wba.setAutoDelete( TRUE ); -} - -TQSocketPrivate::~TQSocketPrivate() -{ - close(); - delete socket; -#ifndef TQT_NO_DNS - delete dns4; - delete dns6; -#endif -} - -void TQSocketPrivate::closeSocket() -{ - // Order is important here - the socket notifiers must go away - // before the socket does, otherwise libc or the kernel will - // become unhappy. - delete rsn; - rsn = 0; - delete wsn; - wsn = 0; - if ( socket ) - socket->close(); -} - -void TQSocketPrivate::close() -{ - closeSocket(); - wsize = 0; - rba.clear(); wba.clear(); - windex = 0; -} - -void TQSocketPrivate::connectionClosed() -{ - // We keep the open state in case there's unread incoming data - state = TQSocket::Idle; - closeSocket(); - wba.clear(); - windex = wsize = 0; -} - -void TQSocketPrivate::setSocketDevice( TQSocket *q, TQSocketDevice *tqdevice ) -{ - delete socket; - delete rsn; - delete wsn; - - if ( tqdevice ) { - socket = tqdevice; - } else { - socket = new TQSocketDevice( TQSocketDevice::Stream, - ( addr.isIPv4Address() ? - TQSocketDevice::IPv4 : - TQSocketDevice::IPv6 ), 0 ); - socket->setBlocking( FALSE ); - socket->setAddressReusable( TRUE ); - } - - rsn = new TQSocketNotifier( socket->socket(), - TQSocketNotifier::Read, TQT_TQOBJECT(q), "read" ); - wsn = new TQSocketNotifier( socket->socket(), - TQSocketNotifier::Write, TQT_TQOBJECT(q), "write" ); - - TQObject::connect( rsn, TQT_SIGNAL(activated(int)), q, TQT_SLOT(sn_read()) ); - rsn->setEnabled( FALSE ); - TQObject::connect( wsn, TQT_SIGNAL(activated(int)), q, TQT_SLOT(sn_write()) ); - wsn->setEnabled( FALSE ); -} - -/*! - \class TQSocket tqsocket.h - \brief The TQSocket class provides a buffered TCP connection. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - It provides a totally non-blocking TQIODevice, and modifies and - extends the API of TQIODevice with socket-specific code. - - Note that a TQApplication must have been constructed before this - class can be used. - - The functions you're likely to call most are connectToHost(), - bytesAvailable(), canReadLine() and the ones it inherits from - TQIODevice. - - connectToHost() is the most-used function. As its name implies, - it opens a connection to a named host. - - Most network protocols are either packet-oriented or - line-oriented. canReadLine() indicates whether a connection - contains an entire unread line or not, and bytesAvailable() - returns the number of bytes available for reading. - - The Q_SIGNALS error(), connected(), readyRead() and - connectionClosed() inform you of the progress of the connection. - There are also some less commonly used Q_SIGNALS. hostFound() is - emitted when connectToHost() has finished its DNS lookup and is - starting its TCP connection. delayedCloseFinished() is emitted - when close() succeeds. bytesWritten() is emitted when TQSocket - moves data from its "to be written" queue into the TCP - implementation. - - There are several access functions for the socket: state() returns - whether the object is idle, is doing a DNS lookup, is connecting, - has an operational connection, etc. address() and port() return - the IP address and port used for the connection. The peerAddress() - and peerPort() functions return the IP address and port used by - the peer, and peerName() returns the name of the peer (normally - the name that was passed to connectToHost()). socketDevice() - returns a pointer to the TQSocketDevice used for this socket. - - TQSocket inherits TQIODevice, and reimplements some functions. In - general, you can treat it as a TQIODevice for writing, and mostly - also for reading. The match isn't perfect, since the TQIODevice - API is designed for tqdevices that are controlled by the same - machine, and an asynchronous peer-to-peer network connection isn't - quite like that. For example, there is nothing that matches - TQIODevice::size() exactly. The documentation for open(), close(), - flush(), size(), at(), atEnd(), readBlock(), writeBlock(), - getch(), putch(), ungetch() and readLine() describes the - differences in detail. - - \warning TQSocket is not suitable for use in threads. If you need - to uses sockets in threads use the lower-level TQSocketDevice class. - - \warning Because TQt doesn't use the native socketstream - implementation on Mac OS X, TQSocket has an implicit transfer - latency of 100ms. You can achieve lower latency on Mac OS X by - using TQSocketDevice instead. - - \sa TQSocketDevice, TQHostAddress, TQSocketNotifier -*/ - -#ifdef USE_QT4 - -/*! - Reads \a maxlen bytes from the socket into \a data and returns the - number of bytes read. Returns -1 if an error occurred. -*/ - -qint64 TQSocket::readData( char *data, qint64 maxlen ) -{ - if ( data == 0 && maxlen != 0 ) { -#if defined(QT_CHECK_NULL) - qWarning( "TQSocket::readBlock: Null pointer error" ); -#endif - return -1; - } - if ( !isOpen() ) { -#if defined(QT_CHECK_STATE) - qWarning( "TQSocket::readBlock: Socket is not open" ); -#endif - return -1; - } - if ( maxlen >= d->rba.size() ) - maxlen = d->rba.size(); -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): readBlock %d bytes", name(), (int)maxlen ); -#endif - d->rba.consumeBytes( maxlen, data ); - // After we read data from our internal buffer, if we use the - // setReadBufferSize() to limit our buffer, we might now be able to - // read more data in our buffer. So enable the read socket notifier, - // but do this only if we are not in a slot connected to the - // readyRead() signal since this might cause a bad recursive behavior. - // We can test for this condition by looking at the - // sn_read_alreadyCalled flag. - if ( d->rsn && TQSocketPrivate::sn_read_alreadyCalled.findRef(this) == -1 ) - d->rsn->setEnabled( true ); - return maxlen; -} - - -/*! - Writes \a len bytes to the socket from \a data and returns the - number of bytes written. Returns -1 if an error occurred. -*/ - -qint64 TQSocket::writeData( const char *data, qint64 len ) -{ -#if defined(QT_CHECK_NULL) - if ( data == 0 && len != 0 ) { - qWarning( "TQSocket::writeBlock: Null pointer error" ); - } -#endif -#if defined(QT_CHECK_STATE) - if ( !isOpen() ) { - qWarning( "TQSocket::writeBlock: Socket is not open" ); - return -1; - } -#endif -#if defined(QT_CHECK_STATE) - if ( d->state == Closing ) { - qWarning( "TQSocket::writeBlock: Cannot write, socket is closing" ); - } -#endif - if ( len == 0 || d->state == Closing || d->state == Idle ) - return 0; - TQByteArray *a = d->wba.last(); - - // next bit is sensitive. if we're writing really small chunks, - // try to buffer up since system calls are expensive, and nagle's - // algorithm is even more expensive. but if anything even - // remotely large is being written, try to issue a write at once. - - bool writeNow = ( d->wsize + len >= 1400 || len > 512 ); - - if ( a && a->size() + len < 128 ) { - // small buffer, resize - int i = a->size(); - a->resize( i+len ); - memcpy( a->data()+i, data, len ); - } else { - // append new buffer - a = new TQByteArray( len ); - memcpy( a->data(), data, len ); - d->wba.append( a ); - } - d->wsize += len; - if ( writeNow ) - flush(); - else if ( d->wsn ) - d->wsn->setEnabled( true ); -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): writeBlock %d bytes", name(), (int)len ); -#endif - return len; -} - -#endif // USE_QT4 - - -/*! - Creates a TQSocket object in \c TQSocket::Idle state. - - The \a parent and \a name arguments are passed on to the TQObject - constructor. - - Note that a TQApplication must have been constructed before sockets - can be used. -*/ - -TQSocket::TQSocket( TQObject *parent, const char *name ) -#ifdef USE_QT4 - : TQIODevice() -#else // USE_QT4 - : TQObject( parent, name ) -#endif // USE_QT4 -{ -#ifdef USE_QT4 - setParent(parent); - setObjectName(name); -#endif // USE_QT4 - d = new TQSocketPrivate; - setSocketDevice( 0 ); - setFlags( IO_Direct ); - resetqStatus(); -} - - -/*! - Destroys the socket. Closes the connection if necessary. - - \sa close() -*/ - -TQSocket::~TQSocket() -{ -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): Destroy", name() ); -#endif - if ( state() != Idle ) - close(); - TQ_ASSERT( d != 0 ); - delete d; -} - - -/*! - Returns a pointer to the internal socket tqdevice. - - There is normally no need to manipulate the socket tqdevice directly - since this class does the necessary setup for most applications. -*/ - -TQSocketDevice *TQSocket::socketDevice() -{ - return d->socket; -} - -/*! - Sets the internal socket tqdevice to \a tqdevice. Passing a \a tqdevice - of 0 will cause the internal socket tqdevice to be used. Any - existing connection will be disconnected before using the new \a - tqdevice. - - The new tqdevice should not be connected before being associated - with a TQSocket; after setting the socket call connectToHost() to - make the connection. - - This function is useful if you need to subclass TQSocketDevice and - want to use the TQSocket API, for example, to implement Unix domain - sockets. -*/ - -void TQSocket::setSocketDevice( TQSocketDevice *tqdevice ) -{ - if ( state() != Idle ) - close(); - d->setSocketDevice( this, tqdevice ); -} - -/*! - \enum TQSocket::State - - This enum defines the connection states: - - \value Idle if there is no connection - \value HostLookup during a DNS lookup - \value Connecting during TCP connection establishment - \value Connected when there is an operational connection - \value Closing if the socket is closing down, but is not yet closed. -*/ - -/*! - Returns the current state of the socket connection. - - \sa TQSocket::State -*/ - -TQSocket::State TQSocket::state() const -{ - return d->state; -} - - -#ifndef TQT_NO_DNS - -/*! - Attempts to make a connection to \a host on the specified \a port - and return immediately. - - Any connection or pending connection is closed immediately, and - TQSocket goes into the \c HostLookup state. When the lookup - succeeds, it emits hostFound(), starts a TCP connection and goes - into the \c Connecting state. Finally, when the connection - succeeds, it emits connected() and goes into the \c Connected - state. If there is an error at any point, it emits error(). - - \a host may be an IP address in string form, or it may be a DNS - name. TQSocket will do a normal DNS lookup if required. Note that - \a port is in native byte order, unlike some other libraries. - - \sa state() -*/ - -void TQSocket::connectToHost( const TQString &host, TQ_UINT16 port ) -{ -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::connectToHost: host %s, port %d", - name(), host.ascii(), port ); -#endif - setSocketIntern( -1 ); - d->state = HostLookup; - d->host = host; - d->port = port; - d->dns4 = new TQDns( host, TQDns::A ); - d->dns6 = new TQDns( host, TQDns::Aaaa ); - - // try if the address is already available (for faster connecting...) - tryConnecting(); - if ( d->state == HostLookup ) { - connect( d->dns4, TQT_SIGNAL(resultsReady()), - this, TQT_SLOT(tryConnecting()) ); - connect( d->dns6, TQT_SIGNAL(resultsReady()), - this, TQT_SLOT(tryConnecting()) ); - } -} - -#endif - - -/*! - This private Q_SLOTS continues the connection process where - connectToHost() leaves off. -*/ - -void TQSocket::tryConnecting() -{ -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::tryConnecting()", name() ); -#endif - // ### this ifdef isn't correct - addresses() also does /etc/hosts and - // numeric-address-as-string handling. -#ifndef TQT_NO_DNS - - if ( d->dns4 ) { - d->l4 = d->dns4->addresses(); - if ( !d->l4.isEmpty() || !d->dns4->isWorking() ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::tryConnecting: host %s, port %d: " - "%d IPv4 addresses", - name(), d->host.ascii(), d->port, d->l4.count() ); -#endif - delete d->dns4; - d->dns4 = 0; - } - } - - if ( d->dns6 ) { - d->l6 = d->dns6->addresses(); - if ( !d->l6.isEmpty() || !d->dns6->isWorking() ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::tryConnecting: host %s, port %d: " - "%d IPv6 addresses", - name(), d->host.ascii(), d->port, d->l6.count() ); -#endif - delete d->dns6; - d->dns6 = 0; - } - } - - if ( d->state == HostLookup ) { - if ( d->l4.isEmpty() && d->l6.isEmpty() && - !d->dns4 && !d->dns6 ) { - // no results and we're not still looking: give up - d->state = Idle; - emit error( ErrHostNotFound ); - return; - } - if ( d->l4.isEmpty() && d->l6.isEmpty() ) { - // no results (yet): try again later - return; - } - - // we've found something. press on with that. if we later find - // more, fine. - emit hostFound(); - d->state = Connecting; - } - - if ( d->state == Connecting ) { - d->addresses += d->l4; - d->addresses += d->l6; - d->l4.clear(); - d->l6.clear(); - - // try one address at a time, falling back to the next one if - // there is a connection failure. (should also support a timeout, - // or do multiple TCP-level connects at a time, with staggered - // starts to avoid bandwidth waste and cause fewer - // "connect-and-abort" errors. but that later.) - bool stuck = TRUE; - while( stuck ) { - stuck = FALSE; - if ( d->socket && - d->socket->connect( d->addr, d->port ) == FALSE ) { - if ( d->socket->error() == TQSocketDevice::NoError ) { - if ( d->wsn ) - d->wsn->setEnabled( TRUE ); - return; // not serious, try again later - } - -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::tryConnecting: " - "Gave up on IP address %s", - name(), d->socket->peerAddress().toString().ascii() ); -#endif - delete d->wsn; - d->wsn = 0; - delete d->rsn; - d->rsn = 0; - delete d->socket; - d->socket = 0; - - if(d->addresses.isEmpty()) { - emit error( ErrConnectionRefused ); - return; - } - } - // if the host has more addresses, try another some. - if ( d->socket == 0 && !d->addresses.isEmpty() ) { - d->addr = *d->addresses.begin(); - d->addresses.remove( d->addresses.begin() ); - d->setSocketDevice( this, 0 ); - stuck = TRUE; -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s)::tryConnecting: Trying IP address %s", - name(), d->addr.toString().ascii() ); -#endif - } - }; - - // The socket write notifier will fire when the connection succeeds - if ( d->wsn ) - d->wsn->setEnabled( TRUE ); - } -#endif -} - -/*! - \enum TQSocket::Error - - This enum specifies the possible errors: - \value ErrConnectionRefused if the connection was refused - \value ErrHostNotFound if the host was not found - \value ErrSocketRead if a read from the socket failed -*/ - -/*! - \fn void TQSocket::error( int ) - - This signal is emitted after an error occurred. The parameter is - the \l Error value. -*/ - -/*! - \fn void TQSocket::hostFound() - - This signal is emitted after connectToHost() has been called and - the host lookup has succeeded. - - \sa connected() -*/ - - -/*! - \fn void TQSocket::connected() - - This signal is emitted after connectToHost() has been called and a - connection has been successfully established. - - \sa connectToHost(), connectionClosed() -*/ - - -/*! - \fn void TQSocket::connectionClosed() - - This signal is emitted when the other end has closed the - connection. The read buffers may contain buffered input data which - you can read after the connection was closed. - - \sa connectToHost(), close() -*/ - - -/*! - \fn void TQSocket::delayedCloseFinished() - - This signal is emitted when a delayed close is finished. - - If you call close() and there is buffered output data to be - written, TQSocket goes into the \c TQSocket::Closing state and - returns immediately. It will then keep writing to the socket until - all the data has been written. Then, the delayedCloseFinished() - signal is emitted. - - \sa close() -*/ - - -/*! - \fn void TQSocket::readyRead() - - This signal is emitted every time there is new incoming data. - - Bear in mind that new incoming data is only reported once; if you do not - read all the data, this class buffers the data and you can read it later, - but no signal is emitted unless new data arrives. A good practice is to - read all data in the slot connected to this signal unless you are sure that - you need to receive more data to be able to process it. - - \sa readBlock(), readLine(), bytesAvailable() -*/ - - -/*! - \fn void TQSocket::bytesWritten( int nbytes ) - - This signal is emitted when data has been written to the network. - The \a nbytes parameter specifies how many bytes were written. - - The bytesToWrite() function is often used in the same context; it - indicates how many buffered bytes there are left to write. - - \sa writeBlock(), bytesToWrite() -*/ - - -/*! - Opens the socket using the specified TQIODevice file mode \a m. - This function is called automatically when needed and you should - not call it yourself. - - \sa close() -*/ - -bool TQSocket::open( int m ) -{ - if ( isOpen() ) { -#if defined(TQT_CHECK_STATE) - qWarning( "TQSocket::open: Already open" ); -#endif - return FALSE; - } - TQIODevice::setMode( m & IO_ReadWrite ); - setState( IO_Open ); - return TRUE; -} - - -/*! - Closes the socket. - - The read buffer is cleared. - - If the output buffer is empty, the state is set to \c - TQSocket::Idle and the connection is terminated immediately. If the - output buffer still contains data to be written, TQSocket goes into - the \c TQSocket::Closing state and the rest of the data will be - written. When all of the outgoing data have been written, the - state is set to \c TQSocket::Idle and the connection is terminated. - At this point, the delayedCloseFinished() signal is emitted. - - If you don't want that the data of the output buffer is written, call - clearPendingData() before you call close(). - - \sa state(), bytesToWrite() clearPendingData() -*/ - -void TQSocket::close() -{ - if ( !isOpen() || d->state == Idle ) // already closed - return; - if ( d->state == Closing ) - return; - if ( !d->rsn || !d->wsn ) - return; -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): close socket", name() ); -#endif - if ( d->socket && d->wsize ) { // there's data to be written - d->state = Closing; - if ( d->rsn ) - d->rsn->setEnabled( FALSE ); - if ( d->wsn ) - d->wsn->setEnabled( TRUE ); - d->rba.clear(); // clear incoming data - return; - } - setFlags( IO_Sequential ); - resetqStatus(); - setState( 0 ); - d->close(); - d->state = Idle; -} - - -/*! - This function consumes \a nbytes bytes of data from the write - buffer. -*/ - -bool TQSocket::consumeWriteBuf( TQ_ULONG nbytes ) -{ - if ( nbytes <= 0 || nbytes > d->wsize ) - return FALSE; -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): skipWriteBuf %d bytes", name(), (int)nbytes ); -#endif - d->wsize -= nbytes; - for ( ;; ) { - TQByteArray *a = d->wba.first(); - if ( d->windex + nbytes >= a->size() ) { - nbytes -= a->size() - d->windex; - d->wba.remove(); - d->windex = 0; - if ( nbytes == 0 ) - break; - } else { - d->windex += nbytes; - break; - } - } - return TRUE; -} - - - -/*! - Implementation of the abstract virtual TQIODevice::flush() function. -*/ - -void TQSocket::flush() -{ - if ( !d->socket ) - return; - bool osBufferFull = FALSE; - int consumed = 0; - while ( !osBufferFull && d->state >= Connecting && d->wsize > 0 ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): flush: Write data to the socket", name() ); -#endif - TQByteArray *a = d->wba.first(); - int nwritten; - int i = 0; - if ( (int)a->size() - d->windex < 1460 ) { - // Concatenate many smaller blocks. the first may be - // partial, but each subsequent block is copied entirely - // or not at all. the sizes here are picked so that we - // generally won't trigger nagle's algorithm in the tcp - // implementation: we concatenate if we'd otherwise send - // less than PMTU bytes (we assume PMTU is 1460 bytes), - // and concatenate up to the largest payload TCP/IP can - // carry. with these precautions, nagle's algorithm - // should apply only when really appropriate. - TQByteArray out( 65536 ); - int j = d->windex; - int s = a->size() - j; - while ( a && i+s < (int)out.size() ) { - memcpy( out.data()+i, a->data()+j, s ); - j = 0; - i += s; - a = d->wba.next(); - s = a ? a->size() : 0; - } - nwritten = d->socket->writeBlock( out.data(), i ); - if ( d->wsn ) - d->wsn->setEnabled( FALSE ); // the TQSocketNotifier documentation says so - } else { - // Big block, write it immediately - i = a->size() - d->windex; - nwritten = d->socket->writeBlock( a->data() + d->windex, i ); - if ( d->wsn ) - d->wsn->setEnabled( FALSE ); // the TQSocketNotifier documentation says so - } - if ( nwritten > 0 ) { - if ( consumeWriteBuf( nwritten ) ) - consumed += nwritten; - } - if ( nwritten < i ) - osBufferFull = TRUE; - } - if ( consumed > 0 ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): flush: wrote %d bytes, %d left", - name(), consumed, (int)d->wsize ); -#endif - emit bytesWritten( consumed ); - } - if ( d->state == Closing && d->wsize == 0 ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): flush: Delayed close done. Terminating.", - name() ); -#endif - setFlags( IO_Sequential ); - resetqStatus(); - setState( 0 ); - d->close(); - d->state = Idle; - emit delayedCloseFinished(); - return; - } - if ( !d->socket->isOpen() ) { - d->connectionClosed(); - emit connectionClosed(); - return; - } - if ( d->wsn ) - d->wsn->setEnabled( d->wsize > 0 ); // write if there's data -} - - -/*! - Returns the number of incoming bytes that can be read right now - (like bytesAvailable()). -*/ - -#ifdef USE_QT4 -qint64 TQSocket::size() const -#else // USE_QT4 -TQIODevice::Offset TQSocket::size() const -#endif // USE_QT4 -{ - return (Offset)bytesAvailable(); -} - - -/*! - Returns the current read index. Since TQSocket is a sequential - tqdevice, the current read index is always zero. -*/ - -#ifdef USE_QT4 -qint64 TQSocket::at() const -#else // USE_QT4 -TQIODevice::Offset TQSocket::at() const -#endif // USE_QT4 -{ - return 0; -} - - -/*! - \overload - - Moves the read index forward to \a index and returns TRUE if the - operation was successful; otherwise returns FALSE. Moving the - index forward means skipping incoming data. -*/ - -bool TQSocket::at( Offset index ) -{ - if ( index > d->rba.size() ) - return FALSE; - d->rba.consumeBytes( (TQ_ULONG)index, 0 ); // throw away data 0..index-1 - // After we read data from our internal buffer, if we use the - // setReadBufferSize() to limit our buffer, we might now be able to - // read more data in our buffer. So enable the read socket notifier, - // but do this only if we are not in a slot connected to the - // readyRead() signal since this might cause a bad recursive behavior. - // We can test for this condition by looking at the - // sn_read_alreadyCalled flag. - if ( d->rsn && TQSocketPrivate::sn_read_alreadyCalled.findRef(this) == -1 ) - d->rsn->setEnabled( TRUE ); - return TRUE; -} - - -/*! - Returns TRUE if there is no more data to read; otherwise returns FALSE. -*/ - -bool TQSocket::atEnd() const -{ - if ( d->socket == 0 ) - return TRUE; - TQSocket * that = (TQSocket *)this; - if ( that->d->socket->bytesAvailable() ) // a little slow, perhaps... - that->sn_read(); - return that->d->rba.size() == 0; -} - - -/*! - Returns the number of incoming bytes that can be read, i.e. the - size of the input buffer. Equivalent to size(). - - This function can trigger the readyRead() signal, if more data has - arrived on the socket. - - \sa bytesToWrite() -*/ - -#ifdef USE_QT4 -qint64 TQSocket::bytesAvailable() const -#else // USE_QT4 -TQ_ULONG TQSocket::bytesAvailable() const -#endif // USE_QT4 -{ - if ( d->socket == 0 ) - return 0; - TQSocket * that = (TQSocket *)this; - if ( that->d->socket->bytesAvailable() ) // a little slow, perhaps... - (void)that->sn_read(); - return that->d->rba.size(); -} - - -/*! - Wait up to \a msecs milliseconds for more data to be available. - - If \a msecs is -1 the call will block indefinitely. - - Returns the number of bytes available. - - If \a timeout is non-null and no error occurred (i.e. it does not - return -1): this function sets \a *timeout to TRUE, if the reason - for returning was that the timeout was reached; otherwise it sets - \a *timeout to FALSE. This is useful to find out if the peer - closed the connection. - - \warning This is a blocking call and should be avoided in event - driven applications. - - \sa bytesAvailable() -*/ - -TQ_ULONG TQSocket::waitForMore( int msecs, bool *timeout ) const -{ - if ( d->socket == 0 ) - return 0; - TQSocket * that = (TQSocket *)this; - if ( that->d->socket->waitForMore( msecs, timeout ) > 0 ) - (void)that->sn_read( TRUE ); - return that->d->rba.size(); -} - -/*! \overload -*/ - -TQ_ULONG TQSocket::waitForMore( int msecs ) const -{ - return waitForMore( msecs, 0 ); -} - -/*! - Returns the number of bytes that are waiting to be written, i.e. - the size of the output buffer. - - \sa bytesAvailable() clearPendingData() -*/ - -#ifdef USE_QT4 -qint64 TQSocket::bytesToWrite() const -#else // USE_QT4 -TQ_ULONG TQSocket::bytesToWrite() const -#endif // USE_QT4 -{ - return d->wsize; -} - -/*! - Deletes the data that is waiting to be written. This is useful if you want - to close the socket without waiting for all the data to be written. - - \sa bytesToWrite() close() delayedCloseFinished() -*/ - -void TQSocket::clearPendingData() -{ - d->wba.clear(); - d->windex = d->wsize = 0; -} - -/*! - Reads \a maxlen bytes from the socket into \a data and returns the - number of bytes read. Returns -1 if an error occurred. -*/ - -TQ_LONG TQSocket::readBlock( char *data, TQ_ULONG maxlen ) -{ - if ( data == 0 && maxlen != 0 ) { -#if defined(TQT_CHECK_NULL) - qWarning( "TQSocket::readBlock: Null pointer error" ); -#endif - return -1; - } - if ( !isOpen() ) { -#if defined(TQT_CHECK_STATE) - qWarning( "TQSocket::readBlock: Socket is not open" ); -#endif - return -1; - } - if ( maxlen >= d->rba.size() ) - maxlen = d->rba.size(); -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): readBlock %d bytes", name(), (int)maxlen ); -#endif - d->rba.consumeBytes( maxlen, data ); - // After we read data from our internal buffer, if we use the - // setReadBufferSize() to limit our buffer, we might now be able to - // read more data in our buffer. So enable the read socket notifier, - // but do this only if we are not in a slot connected to the - // readyRead() signal since this might cause a bad recursive behavior. - // We can test for this condition by looking at the - // sn_read_alreadyCalled flag. - if ( d->rsn && TQSocketPrivate::sn_read_alreadyCalled.findRef(this) == -1 ) - d->rsn->setEnabled( TRUE ); - return maxlen; -} - - -/*! - Writes \a len bytes to the socket from \a data and returns the - number of bytes written. Returns -1 if an error occurred. -*/ - -TQ_LONG TQSocket::writeBlock( const char *data, TQ_ULONG len ) -{ -#if defined(TQT_CHECK_NULL) - if ( data == 0 && len != 0 ) { - qWarning( "TQSocket::writeBlock: Null pointer error" ); - } -#endif -#if defined(TQT_CHECK_STATE) - if ( !isOpen() ) { - qWarning( "TQSocket::writeBlock: Socket is not open" ); - return -1; - } -#endif -#if defined(TQT_CHECK_STATE) - if ( d->state == Closing ) { - qWarning( "TQSocket::writeBlock: Cannot write, socket is closing" ); - } -#endif - if ( len == 0 || d->state == Closing || d->state == Idle ) - return 0; - TQByteArray *a = d->wba.last(); - - // next bit is sensitive. if we're writing really small chunks, - // try to buffer up since system calls are expensive, and nagle's - // algorithm is even more expensive. but if anything even - // remotely large is being written, try to issue a write at once. - - bool writeNow = ( d->wsize + len >= 1400 || len > 512 ); - - if ( a && a->size() + len < 128 ) { - // small buffer, resize - int i = a->size(); - a->resize( i+len ); - memcpy( a->data()+i, data, len ); - } else { - // append new buffer - a = new TQByteArray( len ); - memcpy( a->data(), data, len ); - d->wba.append( a ); - } - d->wsize += len; - if ( writeNow ) - flush(); - else if ( d->wsn ) - d->wsn->setEnabled( TRUE ); -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): writeBlock %d bytes", name(), (int)len ); -#endif - return len; -} - - -/*! - Reads a single byte/character from the internal read buffer. - Returns the byte/character read, or -1 if there is nothing to be - read. - - \sa bytesAvailable(), putch() -*/ - -int TQSocket::getch() -{ - if ( isOpen() && d->rba.size() > 0 ) { - uchar c; - d->rba.consumeBytes( 1, (char*)&c ); - // After we read data from our internal buffer, if we use the - // setReadBufferSize() to limit our buffer, we might now be able to - // read more data in our buffer. So enable the read socket notifier, - // but do this only if we are not in a slot connected to the - // readyRead() signal since this might cause a bad recursive behavior. - // We can test for this condition by looking at the - // sn_read_alreadyCalled flag. - if ( d->rsn && TQSocketPrivate::sn_read_alreadyCalled.findRef(this) == -1 ) - d->rsn->setEnabled( TRUE ); - return c; - } - return -1; -} - - -/*! - Writes the character \a ch to the output buffer. - - Returns \a ch, or -1 if an error occurred. - - \sa getch() -*/ - -int TQSocket::putch( int ch ) -{ - char buf[2]; - buf[0] = ch; - return writeBlock(buf, 1) == 1 ? ch : -1; -} - - -/*! - This implementation of the virtual function TQIODevice::ungetch() - prepends the character \a ch to the read buffer so that the next - read returns this character as the first character of the output. -*/ - -int TQSocket::ungetch( int ch ) -{ -#if defined(TQT_CHECK_STATE) - if ( !isOpen() ) { - qWarning( "TQSocket::ungetch: Socket not open" ); - return -1; - } -#endif - return d->rba.ungetch( ch ); -} - - -/*! - Returns TRUE if it's possible to read an entire line of text from - this socket at this time; otherwise returns FALSE. - - Note that if the peer closes the connection unexpectedly, this - function returns FALSE. This means that loops such as this won't - work: - - \code - while( !socket->canReadLine() ) // WRONG - ; - \endcode - - \sa readLine() -*/ - -bool TQSocket::canReadLine() const -{ - if ( ((TQSocket*)this)->d->rba.scanNewline( 0 ) ) - return TRUE; - return ( bytesAvailable() > 0 && - ((TQSocket*)this)->d->rba.scanNewline( 0 ) ); -} - -/*! - \reimp - \internal - So that it's not hidden by our other readLine(). -*/ -TQ_LONG TQSocket::readLine( char *data, TQ_ULONG maxlen ) -{ - return TQIODevice::readLine(data,maxlen); -} - -/*! - Returns a line of text including a terminating newline character - (\n). Returns "" if canReadLine() returns FALSE. - - \sa canReadLine() -*/ - -TQString TQSocket::readLine() -{ - TQByteArray a(256); - bool nl = d->rba.scanNewline( &a ); - TQString s; - if ( nl ) { - at( a.size() ); // skips the data read - s = TQString( a ); - } - return s; -} - -/*! - \internal - Internal slot for handling socket read notifications. - - This function has can usually only be entered once (i.e. no - recursive calls). If the argument \a force is TRUE, the function - is executed, but no readyRead() Q_SIGNALS are emitted. This - behaviour is useful for the waitForMore() function, so that it is - possible to call waitForMore() in a slot connected to the - readyRead() signal. -*/ - -void TQSocket::sn_read( bool force ) -{ - TQ_LONG maxToRead = 0; - if ( d->readBufferSize > 0 ) { - maxToRead = d->readBufferSize - d->rba.size(); - if ( maxToRead <= 0 ) { - if ( d->rsn ) - d->rsn->setEnabled( FALSE ); - return; - } - } - - // Use TQSocketPrivate::sn_read_alreadyCalled to avoid recursive calls of - // sn_read() (and as a result avoid emitting the readyRead() signal in a - // slot for readyRead(), if you use bytesAvailable()). - if ( !force && TQSocketPrivate::sn_read_alreadyCalled.findRef(this) != -1 ) - return; - TQSocketPrivate::sn_read_alreadyCalled.append( this ); - - char buf[4096]; - TQ_LONG nbytes = d->socket->bytesAvailable(); - TQ_LONG nread; - TQByteArray *a = 0; - - if ( state() == Connecting ) { - if ( nbytes > 0 ) { - tryConnection(); - } else { - // nothing to do, nothing to care about - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } - } - if ( state() == Idle ) { - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } - - if ( nbytes <= 0 ) { // connection closed? - // On Windows this may happen when the connection is still open. - // This happens when the system is heavily loaded and we have - // read all the data on the socket before a new WSAAsyncSelect - // event is processed. A new read operation would then block. - // This code is also useful when TQSocket is used without an - // event loop. - nread = d->socket->readBlock( buf, maxToRead ? TQMIN((TQ_LONG)sizeof(buf),maxToRead) : sizeof(buf) ); - if ( nread == 0 ) { // really closed - if ( !d->socket->isOpen() ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): sn_read: Connection closed", name() ); -#endif - d->connectionClosed(); - emit connectionClosed(); - } - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } else { - if ( nread < 0 ) { - if ( d->socket->error() == TQSocketDevice::NoError ) { - // all is fine - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } -#if defined(TQSOCKET_DEBUG) - qWarning( "TQSocket::sn_read (%s): Close error", name() ); -#endif - if ( d->rsn ) - d->rsn->setEnabled( FALSE ); - emit error( ErrSocketRead ); - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } - a = new TQByteArray( nread ); - memcpy( a->data(), buf, nread ); - } - - } else { // data to be read -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): sn_read: %ld incoming bytes", name(), nbytes ); -#endif - if ( nbytes > (int)sizeof(buf) ) { - // big - a = new TQByteArray( nbytes ); - nread = d->socket->readBlock( a->data(), maxToRead ? TQMIN(nbytes,maxToRead) : nbytes ); - } else { - a = 0; - nread = d->socket->readBlock( buf, maxToRead ? TQMIN((TQ_LONG)sizeof(buf),maxToRead) : sizeof(buf) ); - if ( nread > 0 ) { - // ##### could setRawData - a = new TQByteArray( nread ); - memcpy( a->data(), buf, nread ); - } - } - if ( nread == 0 ) { -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): sn_read: Connection closed", name() ); -#endif - // ### we should rather ask the socket tqdevice if it is closed - d->connectionClosed(); - emit connectionClosed(); - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - delete a; - return; - } else if ( nread < 0 ) { - delete a; - - if ( d->socket->error() == TQSocketDevice::NoError ) { - // all is fine - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } -#if defined(TQT_CHECK_RANGE) - qWarning( "TQSocket::sn_read: Read error" ); -#endif - if ( d->rsn ) - d->rsn->setEnabled( FALSE ); - emit error( ErrSocketRead ); - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); - return; - } - if ( nread != (int)a->size() ) { // unexpected -#if defined(CHECK_RANGE) && !defined(TQ_OS_WIN32) - qWarning( "TQSocket::sn_read: Unexpected short read" ); -#endif - a->resize( nread ); - } - } - d->rba.append( a ); - if ( !force ) { - if ( d->rsn ) - d->rsn->setEnabled( FALSE ); - emit readyRead(); - if ( d->rsn ) - d->rsn->setEnabled( TRUE ); - } - - TQSocketPrivate::sn_read_alreadyCalled.removeRef( this ); -} - - -/*! - \internal - Internal slot for handling socket write notifications. -*/ - -void TQSocket::sn_write() -{ - if ( d->state == Connecting ) // connection established? - tryConnection(); - flush(); -} - -void TQSocket::emitErrorConnectionRefused() -{ - emit error( ErrConnectionRefused ); -} - -void TQSocket::tryConnection() -{ - if ( d->socket->connect( d->addr, d->port ) ) { - d->state = Connected; -#if defined(TQSOCKET_DEBUG) - qDebug( "TQSocket (%s): sn_write: Got connection to %s", - name(), peerName().ascii() ); -#endif - if ( d->rsn ) - d->rsn->setEnabled( TRUE ); - emit connected(); - } else { - d->state = Idle; - TQTimer::singleShot( 0, this, TQT_SLOT(emitErrorConnectionRefused()) ); - return; - } -} - - -/*! - Returns the socket number, or -1 if there is no socket at the moment. -*/ - -int TQSocket::socket() const -{ - if ( d->socket == 0 ) - return -1; - return d->socket->socket(); -} - -/*! - Sets the socket to use \a socket and the state() to \c Connected. - The socket must already be connected. - - This allows us to use the TQSocket class as a wrapper for other - socket types (e.g. Unix Domain Sockets). -*/ - -void TQSocket::setSocket( int socket ) -{ - setSocketIntern( socket ); - d->state = Connection; - d->rsn->setEnabled( TRUE ); -} - - -/*! - Sets the socket to \a socket. This is used by both setSocket() and - connectToHost() and can also be used on unconnected sockets. -*/ - -void TQSocket::setSocketIntern( int socket ) -{ - if ( state() != Idle ) { - clearPendingData(); - close(); - } - TQ_ULONG oldBufferSize = d ? d->readBufferSize : 0; - delete d; - - d = new TQSocketPrivate; - if (oldBufferSize) - d->readBufferSize = oldBufferSize; - if ( socket >= 0 ) { - TQSocketDevice *sd = new TQSocketDevice( socket, TQSocketDevice::Stream ); - sd->setBlocking( FALSE ); - sd->setAddressReusable( TRUE ); - d->setSocketDevice( this, sd ); - } - d->state = Idle; - - // Initialize the IO tqdevice flags - setFlags( IO_Direct ); - resetqStatus(); - open( IO_ReadWrite ); - - // hm... this is not very nice. - d->host = TQString::null; - d->port = 0; -#ifndef TQT_NO_DNS - delete d->dns4; - d->dns4 = 0; - delete d->dns6; - d->dns6 = 0; -#endif -} - - -/*! - Returns the host port number of this socket, in native byte order. -*/ - -TQ_UINT16 TQSocket::port() const -{ - if ( d->socket == 0 ) - return 0; - return d->socket->port(); -} - - -/*! - Returns the peer's host port number, normally as specified to the - connectToHost() function. If none has been set, this function - returns 0. - - Note that TQt always uses native byte order, i.e. 67 is 67 in TQt; - there is no need to call htons(). -*/ - -TQ_UINT16 TQSocket::peerPort() const -{ - if ( d->socket == 0 ) - return 0; - return d->socket->peerPort(); -} - - -/*! - Returns the host address of this socket. (This is normally the - main IP address of the host, but can be e.g. 127.0.0.1 for - connections to localhost.) -*/ - -TQHostAddress TQSocket::address() const -{ - if ( d->socket == 0 ) { - TQHostAddress tmp; - return tmp; - } - return d->socket->address(); -} - - -/*! - Returns the address of the connected peer if the socket is in - Connected state; otherwise an empty TQHostAddress is returned. -*/ - -TQHostAddress TQSocket::peerAddress() const -{ - if ( d->socket == 0 ) { - TQHostAddress tmp; - return tmp; - } - return d->socket->peerAddress(); -} - - -/*! - Returns the host name as specified to the connectToHost() - function. An empty string is returned if none has been set. -*/ - -TQString TQSocket::peerName() const -{ - return d->host; -} - -/*! - Sets the size of the TQSocket's internal read buffer to \a bufSize. - - Usually TQSocket reads all data that is available from the operating - system's socket. If the buffer size is limited to a certain size, this - means that the TQSocket class doesn't buffer more than this size of data. - - If the size of the read buffer is 0, the read buffer is unlimited and all - incoming data is buffered. This is the default. - - If you read the data in the readyRead() signal, you shouldn't use this - option since it might slow down your program unnecessary. This option is - useful if you only need to read the data at certain points in time, like in - a realtime streaming application. - - \sa readBufferSize() -*/ - -void TQSocket::setReadBufferSize( TQ_ULONG bufSize ) -{ - d->readBufferSize = bufSize; -} - -/*! - Returns the size of the read buffer. - - \sa setReadBufferSize() -*/ - -TQ_ULONG TQSocket::readBufferSize() const -{ - return d->readBufferSize; -} - -#endif //TQT_NO_NETWORK diff --git a/tqtinterface/qt4/src/network/tqsocket.h b/tqtinterface/qt4/src/network/tqsocket.h deleted file mode 100644 index 7b9453e..0000000 --- a/tqtinterface/qt4/src/network/tqsocket.h +++ /dev/null @@ -1,174 +0,0 @@ -/**************************************************************************** -** -** Definition of TQSocket class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQSOCKET_H -#define TQSOCKET_H - -#ifndef TQT_H -#include "tqobject.h" -#include "tqiodevice.h" -#include "tqhostaddress.h" // int->TQHostAddress conversion -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_NETWORK -#else -#define TQM_EXPORT_NETWORK TQ_EXPORT -#endif - -#ifndef TQT_NO_NETWORK -class TQSocketPrivate; -class TQSocketDevice; - -// #ifdef USE_QT4 -class TQM_EXPORT_NETWORK TQSocket : public TQIODevice -// #else // USE_QT4 -// class TQM_EXPORT_NETWORK TQSocket : public TQObject, public TQIODevice -// #endif // USE_QT4 -{ - Q_OBJECT - TQ_OBJECT -public: - enum Error { - ErrConnectionRefused, - ErrHostNotFound, - ErrSocketRead - }; - - TQSocket( TQObject *parent=0, const char *name=0 ); - virtual ~TQSocket(); - - enum State { Idle, HostLookup, Connecting, - Connected, Closing, - Connection=Connected }; - State state() const; - - int socket() const; - virtual void setSocket( int ); - - TQSocketDevice *socketDevice(); - virtual void setSocketDevice( TQSocketDevice * ); - -#ifndef TQT_NO_DNS - virtual void connectToHost( const TQString &host, TQ_UINT16 port ); -#endif - TQString peerName() const; - - // Implementation of TQIODevice abstract virtual functions - bool open( int mode ); - void close(); - void flush(); -#ifdef USE_QT4 - qint64 size() const; - qint64 at() const; -#else // USE_QT4 - Offset size() const; - Offset at() const; -#endif // USE_QT4 - bool at( Offset ); - bool atEnd() const; - -#ifdef USE_QT4 - qint64 bytesAvailable() const; // ### TQIODevice::Offset instead? - qint64 bytesToWrite() const; -#else // USE_QT4 - TQ_ULONG bytesAvailable() const; // ### TQIODevice::Offset instead? - TQ_ULONG bytesToWrite() const; -#endif // USE_QT4 - TQ_ULONG waitForMore( int msecs, bool *timeout ) const; - TQ_ULONG waitForMore( int msecs ) const; // ### TQt 4.0: merge the two overloads - void clearPendingData(); - - TQ_LONG readBlock( char *data, TQ_ULONG maxlen ); - TQ_LONG writeBlock( const char *data, TQ_ULONG len ); - TQ_LONG readLine( char *data, TQ_ULONG maxlen ); - - int getch(); - int putch( int ); - int ungetch(int); - - bool canReadLine() const; - virtual TQString readLine(); - - TQ_UINT16 port() const; - TQ_UINT16 peerPort() const; - TQHostAddress address() const; - TQHostAddress peerAddress() const; - - void setReadBufferSize( TQ_ULONG ); - TQ_ULONG readBufferSize() const; - -Q_SIGNALS: - void hostFound(); - void connected(); - void connectionClosed(); - void delayedCloseFinished(); - void readyRead(); - void bytesWritten( int nbytes ); - void error( int ); - -protected Q_SLOTS: - virtual void sn_read( bool force=FALSE ); - virtual void sn_write(); - -private Q_SLOTS: - void tryConnecting(); - void emitErrorConnectionRefused(); - -private: - TQSocketPrivate *d; - - bool consumeWriteBuf( TQ_ULONG nbytes ); - void tryConnection(); - void setSocketIntern( int socket ); - -private: // Disabled copy constructor and operator= -#if defined(TQ_DISABLE_COPY) - TQSocket( const TQSocket & ); - TQSocket &operator=( const TQSocket & ); -#endif - -protected: - qint64 readData(char *data, qint64 maxlen); - qint64 writeData(const char *data, qint64 len); -}; - -#endif //TQT_NO_NETWORK -#endif // TQSOCKET_H diff --git a/tqtinterface/qt4/src/network/tqsocketdevice.cpp b/tqtinterface/qt4/src/network/tqsocketdevice.cpp deleted file mode 100644 index e2f7287..0000000 --- a/tqtinterface/qt4/src/network/tqsocketdevice.cpp +++ /dev/null @@ -1,584 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQSocketDevice class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqsocketdevice.h" -#ifndef TQT_NO_NETWORK - -#include "tqwindowdefs.h" -#include <string.h> - - -//#define TQSOCKETDEVICE_DEBUG - - -class TQSocketDevicePrivate -{ -public: - TQSocketDevicePrivate( TQSocketDevice::Protocol p ) - : protocol(p) - { } - - TQSocketDevice::Protocol protocol; -}; - - -/*! - \class TQSocketDevice tqsocketdevice.h - \brief The TQSocketDevice class provides a platform-independent low-level socket API. -\if defined(commercial) - It is part of the <a href="commercialeditions.html">TQt Enterprise Edition</a>. -\endif - - \ingroup io - \module network - - This class provides a low level API for working with sockets. Users of - this class are assumed to have networking experience. For most users the - TQSocket class provides a much easier and high level alternative, but - certain things (like UDP) can't be done with TQSocket and if you need a - platform-independent API for those, TQSocketDevice is the right choice. - - The essential purpose of the class is to provide a TQIODevice that - works on sockets, wrapped in a platform-independent API. - - When calling connect() or bind(), TQSocketDevice detects the - protocol family (IPv4, IPv6) automatically. Passing the protocol - family to TQSocketDevice's constructor or to setSocket() forces - creation of a socket tqdevice of a specific protocol. If not set, the - protocol will be detected at the first call to connect() or bind(). - - \sa TQSocket, TQSocketNotifier, TQHostAddress -*/ - - -/*! - \enum TQSocketDevice::Protocol - - This enum type describes the protocol family of the socket. Possible values - are: - - \value IPv4 The socket is an IPv4 socket. - \value IPv6 The socket is an IPv6 socket. - \value Unknown The protocol family of the socket is not known. This can - happen if you use TQSocketDevice with an already existing socket; it - tries to determine the protocol family, but this can fail if the - protocol family is not known to TQSocketDevice. - - \sa protocol() setSocket() -*/ - -/*! - \enum TQSocketDevice::Error - - This enum type describes the error states of TQSocketDevice. - - \value NoError No error has occurred. - - \value AlreadyBound The tqdevice is already bound, according to bind(). - - \value Inaccessible The operating system or firewall prohibited - the action. - - \value NoResources The operating system ran out of a resource. - - \value InternalError An internal error occurred in TQSocketDevice. - - \value Impossible An attempt was made to do something which makes - no sense. For example: - \code - ::close( sd->socket() ); - sd->writeBlock( someData, 42 ); - \endcode - The libc ::close() closes the socket, but TQSocketDevice is not aware - of this. So when you call writeBlock(), the impossible happens. - - \value NoFiles The operating system will not let TQSocketDevice open - another file. - - \value ConnectionRefused A connection attempt was rejected by the - peer. - - \value NetworkFailure There is a network failure. - - \value UnknownError The operating system did something - unexpected. -*/ - -/*! - \enum TQSocketDevice::Type - - This enum type describes the type of the socket: - \value Stream a stream socket (TCP, usually) - \value Datagram a datagram socket (UDP, usually) -*/ - - -/*! - Creates a TQSocketDevice object for the existing socket \a socket. - - The \a type argument must match the actual socket type; use \c - TQSocketDevice::Stream for a reliable, connection-oriented TCP - socket, or \c TQSocketDevice::Datagram for an unreliable, - connectionless UDP socket. -*/ -TQSocketDevice::TQSocketDevice( int socket, Type type ) - : fd( socket ), t( type ), p( 0 ), pp( 0 ), e( NoError ), - d(new TQSocketDevicePrivate(Unknown)) -{ -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice: Created TQSocketDevice %p (socket %x, type %d)", - this, socket, type ); -#endif - init(); - setSocket( socket, type ); -} - -/*! - Creates a TQSocketDevice object for a stream or datagram socket. - - The \a type argument must be either \c TQSocketDevice::Stream for a - reliable, connection-oriented TCP socket, or \c - TQSocketDevice::Datagram for an unreliable UDP socket. - - The socket is created as an IPv4 socket. - - \sa blocking() protocol() -*/ -TQSocketDevice::TQSocketDevice( Type type ) - : fd( -1 ), t( type ), p( 0 ), pp( 0 ), e( NoError ), - d(new TQSocketDevicePrivate(IPv4)) -{ -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice: Created TQSocketDevice object %p, type %d", - this, type ); -#endif - init(); - setSocket( createNewSocket(), type ); -} - -/*! - Creates a TQSocketDevice object for a stream or datagram socket. - - The \a type argument must be either \c TQSocketDevice::Stream for a - reliable, connection-oriented TCP socket, or \c - TQSocketDevice::Datagram for an unreliable UDP socket. - - The \a protocol indicates whether the socket should be of type IPv4 - or IPv6. Passing \c Unknown is not meaningful in this context and you - should avoid using (it creates an IPv4 socket, but your code is not easily - readable). - - The argument \a dummy is necessary for compatibility with some - compilers. - - \sa blocking() protocol() -*/ -TQSocketDevice::TQSocketDevice( Type type, Protocol protocol, int ) - : fd( -1 ), t( type ), p( 0 ), pp( 0 ), e( NoError ), - d(new TQSocketDevicePrivate(protocol)) -{ -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice: Created TQSocketDevice object %p, type %d", - this, type ); -#endif - init(); - setSocket( createNewSocket(), type ); -} - -/*! - Destroys the socket tqdevice and closes the socket if it is open. -*/ -TQSocketDevice::~TQSocketDevice() -{ - close(); - delete d; - d = 0; -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice: Destroyed TQSocketDevice %p", this ); -#endif -} - - -/*! - Returns TRUE if this is a valid socket; otherwise returns FALSE. - - \sa socket() -*/ -bool TQSocketDevice::isValid() const -{ - return fd != -1; -} - - -/*! - \fn Type TQSocketDevice::type() const - - Returns the socket type which is either \c TQSocketDevice::Stream - or \c TQSocketDevice::Datagram. - - \sa socket() -*/ -TQSocketDevice::Type TQSocketDevice::type() const -{ - return t; -} - -/*! - Returns the socket's protocol family, which is one of \c Unknown, \c IPv4, - or \c IPv6. - - TQSocketDevice either creates a socket with a well known protocol family or - it uses an already existing socket. In the first case, this function - returns the protocol family it was constructed with. In the second case, it - tries to determine the protocol family of the socket; if this fails, it - returns \c Unknown. - - \sa Protocol setSocket() -*/ -TQSocketDevice::Protocol TQSocketDevice::protocol() const -{ - if ( d->protocol == Unknown ) - d->protocol = getProtocol(); - return d->protocol; -} - -/*! - Returns the socket number, or -1 if it is an invalid socket. - - \sa isValid(), type() -*/ -int TQSocketDevice::socket() const -{ - return fd; -} - - -/*! - Sets the socket tqdevice to operate on the existing socket \a - socket. - - The \a type argument must match the actual socket type; use \c - TQSocketDevice::Stream for a reliable, connection-oriented TCP - socket, or \c TQSocketDevice::Datagram for an unreliable, - connectionless UDP socket. - - Any existing socket is closed. - - \sa isValid(), close() -*/ -void TQSocketDevice::setSocket( int socket, Type type ) -{ - if ( fd != -1 ) // close any open socket - close(); -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice::setSocket: socket %x, type %d", socket, type ); -#endif - t = type; - fd = socket; - d->protocol = Unknown; - e = NoError; - setFlags( IO_Sequential ); - resetqStatus(); - open( IO_ReadWrite ); - fetchConnectionParameters(); -} - - -/*! - \reimp - - Opens the socket using the specified TQIODevice file \a mode. This - function is called from the TQSocketDevice constructors and from - the setSocket() function. You should not call it yourself. - - \sa close(). -*/ -bool TQSocketDevice::open( int mode ) -{ - if ( isOpen() || !isValid() ) - return FALSE; -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice::open: mode %x", mode ); -#endif - setMode( mode & IO_ReadWrite ); - setState( IO_Open ); - return TRUE; -} - - -/*! - \reimp - - The current TQSocketDevice implementation does not buffer at all, - so this is a no-op. -*/ -void TQSocketDevice::flush() -{ -} - - -/*! - \reimp - - The size is meaningless for a socket, therefore this function returns 0. -*/ -#ifdef USE_QT4 -qint64 TQSocketDevice::size() const -#else // USE_QT4 -TQIODevice::Offset TQSocketDevice::size() const -#endif // USE_QT4 -{ - return 0; -} - - -/*! - \reimp - - The read/write index is meaningless for a socket, therefore this - function returns 0. -*/ -#ifdef USE_QT4 -qint64 TQSocketDevice::at() const -#else // USE_QT4 -TQIODevice::Offset TQSocketDevice::at() const -#endif // USE_QT4 -{ - return 0; -} - - -/*! - \reimp - - The read/write index is meaningless for a socket, therefore this - function does nothing and returns TRUE. -*/ -bool TQSocketDevice::at( Offset ) -{ - return TRUE; -} - - -/*! - \reimp - - Returns TRUE if no data is currently available at the socket; - otherwise returns FALSE. -*/ -bool TQSocketDevice::atEnd() const -{ - return bytesAvailable() <= 0; -} - - -/*! - \reimp - - \warning getch() is implemented as a one-byte readBlock(), so it - may be very slow if you call it more than a few times. - - \sa putch() readBlock() -*/ -int TQSocketDevice::getch() -{ - char buf[2]; - return readBlock(buf,1) == 1 ? buf[0] : -1; -} - - -/*! - \reimp - - \warning putch() is implemented as a one-byte writeBlock(), so it - may be very slow if you call it more than a few times. - - \sa getch() -*/ -int TQSocketDevice::putch( int ch ) -{ - char buf[2]; - buf[0] = ch; - return writeBlock(buf, 1) == 1 ? ch : -1; -} - - -/*! - \reimp - - This implementation of ungetch returns -1 (error). A socket is a - sequential tqdevice and does not allow any ungetch operation. -*/ -int TQSocketDevice::ungetch( int ) -{ - return -1; -} - - -/*! - Returns TRUE if the address of this socket can be used by other - sockets at the same time, and FALSE if this socket claims - exclusive ownership. - - \sa setAddressReusable() -*/ -bool TQSocketDevice::addressReusable() const -{ - return option( ReuseAddress ); -} - - -/*! - Sets the address of this socket to be usable by other sockets too - if \a enable is TRUE, and to be used exclusively by this socket if - \a enable is FALSE. - - When a socket is reusable, other sockets can use the same port - number (and IP address), which is generally useful. Of course - other sockets cannot use the same - (address,port,peer-address,peer-port) 4-tuple as this socket, so - there is no risk of confusing the two TCP connections. - - \sa addressReusable() -*/ -void TQSocketDevice::setAddressReusable( bool enable ) -{ - setOption( ReuseAddress, enable ); -} - - -/*! - Returns the size of the operating system receive buffer. - - \sa setReceiveBufferSize() -*/ -int TQSocketDevice::receiveBufferSize() const -{ - return option( ReceiveBuffer ); -} - - -/*! - Sets the size of the operating system receive buffer to \a size. - - The operating system receive buffer size effectively limits two - things: how much data can be in transit at any one moment, and how - much data can be received in one iteration of the main event loop. - - The default is operating system-dependent. A socket that receives - large amounts of data is probably best with a buffer size of - 49152. -*/ -void TQSocketDevice::setReceiveBufferSize( uint size ) -{ - setOption( ReceiveBuffer, size ); -} - - -/*! - Returns the size of the operating system send buffer. - - \sa setSendBufferSize() -*/ -int TQSocketDevice::sendBufferSize() const -{ - return option( SendBuffer ); -} - - -/*! - Sets the size of the operating system send buffer to \a size. - - The operating system send buffer size effectively limits how much - data can be in transit at any one moment. - - The default is operating system-dependent. A socket that sends - large amounts of data is probably best with a buffer size of - 49152. -*/ -void TQSocketDevice::setSendBufferSize( uint size ) -{ - setOption( SendBuffer, size ); -} - - -/*! - Returns the port number of this socket tqdevice. This may be 0 for a - while, but is set to something sensible as soon as a sensible - value is available. - - Note that TQt always uses native byte order, i.e. 67 is 67 in TQt; - there is no need to call htons(). -*/ -TQ_UINT16 TQSocketDevice::port() const -{ - return p; -} - - -/*! - Returns the address of this socket tqdevice. This may be 0.0.0.0 for - a while, but is set to something sensible as soon as a sensible - value is available. -*/ -TQHostAddress TQSocketDevice::address() const -{ - return a; -} - - -/*! - Returns the first error seen. -*/ -TQSocketDevice::Error TQSocketDevice::error() const -{ - return e; -} - - -/*! - Allows subclasses to set the error state to \a err. -*/ -void TQSocketDevice::setError( Error err ) -{ - e = err; -} -#endif //TQT_NO_NETWORK - diff --git a/tqtinterface/qt4/src/network/tqsocketdevice.h b/tqtinterface/qt4/src/network/tqsocketdevice.h deleted file mode 100644 index 050ad5b..0000000 --- a/tqtinterface/qt4/src/network/tqsocketdevice.h +++ /dev/null @@ -1,185 +0,0 @@ -/**************************************************************************** -** -** Definition of TQSocketDevice class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#ifndef TQSOCKETDEVICE_H -#define TQSOCKETDEVICE_H - -#ifndef TQT_H -#include "tqiodevice.h" -#include "tqhostaddress.h" // int->TQHostAddress conversion -#endif // TQT_H - -#if !defined( TQT_MODULE_NETWORK ) || defined( TQT_LICENSE_PROFESSIONAL ) || defined( TQT_INTERNAL_NETWORK ) -#define TQM_EXPORT_NETWORK -#else -#define TQM_EXPORT_NETWORK TQ_EXPORT -#endif - -#ifndef TQT_NO_NETWORK -class TQSocketDevicePrivate; - -#define QT_SOCKLEN_T socklen_t - -class TQM_EXPORT_NETWORK TQSocketDevice: public TQIODevice -{ -public: - enum Type { Stream, Datagram }; - enum Protocol { IPv4, IPv6, Unknown }; - - TQSocketDevice( Type type = Stream ); - TQSocketDevice( Type type, Protocol protocol, int dummy ); - TQSocketDevice( int socket, Type type ); - virtual ~TQSocketDevice(); - - bool isValid() const; - Type type() const; - Protocol protocol() const; - - int socket() const; - virtual void setSocket( int socket, Type type ); - - bool open( int mode ); - void close(); - void flush(); - - // Implementation of TQIODevice abstract virtual functions -#ifdef USE_QT4 - qint64 size() const; - qint64 at() const; -#else // USE_QT4 - Offset size() const; - Offset at() const; -#endif // USE_QT4 - bool at( Offset ); - bool atEnd() const; - - bool blocking() const; - virtual void setBlocking( bool ); - - bool addressReusable() const; - virtual void setAddressReusable( bool ); - - int receiveBufferSize() const; - virtual void setReceiveBufferSize( uint ); - int sendBufferSize() const; - virtual void setSendBufferSize( uint ); - - virtual bool connect( const TQHostAddress &, TQ_UINT16 ); - - virtual bool bind( const TQHostAddress &, TQ_UINT16 ); - virtual bool listen( int backlog ); - virtual int accept(); - -#ifdef USE_QT4 - qint64 bytesAvailable() const; -#else // USE_QT4 - TQ_LONG bytesAvailable() const; -#endif // USE_QT4 - TQ_LONG waitForMore( int msecs, bool *timeout=0 ) const; - TQ_LONG readBlock( char *data, TQ_ULONG maxlen ); - TQ_LONG writeBlock( const char *data, TQ_ULONG len ); - virtual TQ_LONG writeBlock( const char *data, TQ_ULONG len, - const TQHostAddress & host, TQ_UINT16 port ); - - int getch(); - int putch( int ); - int ungetch(int); - - TQ_UINT16 port() const; - TQ_UINT16 peerPort() const; - TQHostAddress address() const; - TQHostAddress peerAddress() const; - - enum Error { - NoError, - AlreadyBound, - Inaccessible, - NoResources, - InternalError, - Bug = InternalError, // ### remove in 4.0? - Impossible, - NoFiles, - ConnectionRefused, - NetworkFailure, - UnknownError - }; - Error error() const; - -protected: - void setError( Error err ); - -private: - int fd; - Type t; - TQ_UINT16 p; - TQHostAddress a; - TQ_UINT16 pp; - TQHostAddress pa; - TQSocketDevice::Error e; - TQSocketDevicePrivate * d; - - enum Option { Broadcast, ReceiveBuffer, ReuseAddress, SendBuffer }; - - int option( Option ) const; - virtual void setOption( Option, int ); - - void fetchConnectionParameters(); -#if defined(TQ_OS_WIN32) - void fetchPeerConnectionParameters(); -#endif - - static void init(); - int createNewSocket(); - Protocol getProtocol() const; - -private: // Disabled copy constructor and operator= -#if defined(TQ_DISABLE_COPY) - TQSocketDevice( const TQSocketDevice & ); - TQSocketDevice &operator=( const TQSocketDevice & ); -#endif - -protected: -// void setError( Error err ); - qint64 readData(char *data, qint64 maxlen); - qint64 writeData(const char *data, qint64 len); -}; - -#endif // TQT_NO_NETWORK -#endif // TQSOCKETDEVICE_H diff --git a/tqtinterface/qt4/src/network/tqsocketdevice_unix.cpp b/tqtinterface/qt4/src/network/tqsocketdevice_unix.cpp deleted file mode 100644 index 46c3932..0000000 --- a/tqtinterface/qt4/src/network/tqsocketdevice_unix.cpp +++ /dev/null @@ -1,1269 +0,0 @@ -/**************************************************************************** -** -** Implementation of TQSocketDevice class. -** -** Created : 970521 -** -** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. -** -** This file is part of the network module of the TQt GUI Toolkit. -** -** This file may be used under the terms of the GNU General -** Public License versions 2.0 or 3.0 as published by the Free -** Software Foundation and appearing in the files LICENSE.GPL2 -** and LICENSE.GPL3 included in the packaging of this file. -** Alternatively you may (at your option) use any later version -** of the GNU General Public License if such license has been -** publicly approved by Trolltech ASA (or its successors, if any) -** and the KDE Free TQt Foundation. -** -** Please review the following information to ensure GNU General -** Public Licensing requirements will be met: -** http://trolltech.com/products/qt/licenses/licensing/opensource/. -** If you are unsure which license is appropriate for your use, please -** review the following information: -** http://trolltech.com/products/qt/licenses/licensing/licensingoverview -** or contact the sales department at [email protected]. -** -** This file may be used under the terms of the Q Public License as -** defined by Trolltech ASA and appearing in the file LICENSE.TQPL -** included in the packaging of this file. Licensees holding valid TQt -** Commercial licenses may use this file in accordance with the TQt -** Commercial License Agreement provided with the Software. -** -** This file is provided "AS IS" with NO WARRANTY OF ANY KIND, -** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted -** herein. -** -**********************************************************************/ - -#include "tqplatformdefs.h" - -// Almost always the same. If not, specify in qplatformdefs.h. -#if !defined(TQT_SOCKOPTLEN_T) -# define TQT_SOCKOPTLEN_T TQT_SOCKLEN_T -#endif - -// Tru64 redefines accept -> _accept with _XOPEN_SOURCE_EXTENDED -static inline int qt_socket_accept(int s, struct sockaddr *addr, TQT_SOCKLEN_T *addrlen) -{ return ::accept(s, addr, addrlen); } -#if defined(accept) -# undef accept -#endif - -// Solaris redefines bind -> __xnet_bind with _XOPEN_SOURCE_EXTENDED -static inline int qt_socket_bind(int s, struct sockaddr *addr, TQT_SOCKLEN_T addrlen) -{ return ::bind(s, addr, addrlen); } -#if defined(bind) -# undef bind -#endif - -// Solaris redefines connect -> __xnet_connect with _XOPEN_SOURCE_EXTENDED -static inline int qt_socket_connect(int s, struct sockaddr *addr, TQT_SOCKLEN_T addrlen) -{ return ::connect(s, addr, addrlen); } -#if defined(connect) -# undef connect -#endif - -// UnixWare 7 redefines listen -> _listen -static inline int qt_socket_listen(int s, int backlog) -{ return ::listen(s, backlog); } -#if defined(listen) -# undef listen -#endif - -// UnixWare 7 redefines socket -> _socket -static inline int qt_socket_socket(int domain, int type, int protocol) -{ return ::socket(domain, type, protocol); } -#if defined(socket) -# undef socket -#endif - -#include "tqsocketdevice.h" - -#ifndef TQT_NO_NETWORK - -#include "tqwindowdefs.h" - -#include <errno.h> -#include <sys/types.h> - - -static inline void qt_socket_getportaddr( struct sockaddr *sa, - TQ_UINT16 *port, TQHostAddress *addr ) -{ -#if !defined(TQT_NO_IPV6) - if ( sa->sa_family == AF_INET6 ) { - struct sockaddr_in6 *sa6 = ( struct sockaddr_in6 * )sa; - TQ_IPV6ADDR tmp; - memcpy( &tmp, &sa6->sin6_addr.s6_addr, sizeof(tmp) ); - TQHostAddress a( tmp ); - *addr = a; - *port = ntohs( sa6->sin6_port ); - return; - } -#endif - struct sockaddr_in *sa4 = (struct sockaddr_in *)sa; - TQHostAddress a( ntohl( sa4->sin_addr.s_addr ) ); - *port = ntohs( sa4->sin_port ); - *addr = TQHostAddress( ntohl( sa4->sin_addr.s_addr ) ); - return; -} - -#ifdef USE_QT4 - -qint64 TQSocketDevice::readData( char *data, qint64 maxlen ) -{ -#if defined(QT_CHECK_NULL) - if ( data == 0 && maxlen != 0 ) { - qWarning( "TQSocketDevice::readBlock: Null pointer error" ); - } -#endif -#if defined(QT_CHECK_STATE) - if ( !isValid() ) { - qWarning( "TQSocketDevice::readBlock: Invalid socket" ); - return -1; - } - if ( !isOpen() ) { - qWarning( "TQSocketDevice::readBlock: Device is not open" ); - return -1; - } - if ( !isReadable() ) { - qWarning( "TQSocketDevice::readBlock: Read operation not permitted" ); - return -1; - } -#endif - bool done = false; - int r = 0; - while ( done == false ) { - if ( t == Datagram ) { -#if !defined(QT_NO_IPV6) - struct sockaddr_storage aa; -#else - struct sockaddr_in aa; -#endif - memset( &aa, 0, sizeof(aa) ); - QT_SOCKLEN_T sz; - sz = sizeof( aa ); - r = ::recvfrom( fd, data, maxlen, 0, - (struct sockaddr *)&aa, &sz ); - - qt_socket_getportaddr( (struct sockaddr *)&aa, &pp, &pa); - - } else { - r = ::read( fd, data, maxlen ); - } - done = true; - if ( r == 0 && t == Stream && maxlen > 0 ) { - // connection closed - close(); - } else if ( r >= 0 || errno == EAGAIN || errno == EWOULDBLOCK ) { - // nothing - } else if ( errno == EINTR ) { - done = false; - } else if ( e == NoError ) { - switch( errno ) { - case EIO: - case EISDIR: - case EBADF: - case EINVAL: - case EFAULT: - case ENOTCONN: - case ENOTSOCK: - e = Impossible; - break; -#if defined(ENONET) - case ENONET: -#endif - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - e = NetworkFailure; - break; - case EPIPE: - case ECONNRESET: - // connection closed - close(); - r = 0; - break; - default: - e = UnknownError; - break; - } - } - } - return r; -} - - -qint64 TQSocketDevice::writeData( const char *data, qint64 len ) -{ - if ( data == 0 && len != 0 ) { -#if defined(QT_CHECK_NULL) || defined(QSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Null pointer error" ); -#endif - return -1; - } - if ( !isValid() ) { -#if defined(QT_CHECK_STATE) || defined(QSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Invalid socket" ); -#endif - return -1; - } - if ( !isOpen() ) { -#if defined(QT_CHECK_STATE) || defined(QSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Device is not open" ); -#endif - return -1; - } - if ( !isWritable() ) { -#if defined(QT_CHECK_STATE) || defined(QSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Write operation not permitted" ); -#endif - return -1; - } - bool done = false; - int r = 0; - bool timeout; - while ( !done ) { - r = ::write( fd, data, len ); - done = true; - if ( r < 0 && e == NoError && - errno != EAGAIN && errno != EWOULDBLOCK ) { - switch( errno ) { - case EINTR: // signal - call read() or whatever again - done = false; - break; - case EPIPE: - case ECONNRESET: - // connection closed - close(); - r = 0; - break; - case ENOSPC: - case EIO: - case EISDIR: - case EBADF: - case EINVAL: - case EFAULT: - case ENOTCONN: - case ENOTSOCK: - e = Impossible; - break; -#if defined(ENONET) - case ENONET: -#endif - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - e = NetworkFailure; - break; - default: - e = UnknownError; - break; - } - } else if ( waitForMore( 0, &timeout ) == 0 ) { - if ( !timeout ) { - // connection closed - close(); - } - } - } - return r; -} - -#endif // USE_QT4 - - -//#define TQSOCKETDEVICE_DEBUG - -// internal -void TQSocketDevice::init() -{ -} - - -TQSocketDevice::Protocol TQSocketDevice::getProtocol() const -{ - if ( isValid() ) { -#if !defined (TQT_NO_IPV6) - struct sockaddr_storage sa; -#else - struct sockaddr sa; -#endif - memset( &sa, 0, sizeof(sa) ); - TQT_SOCKLEN_T sz = sizeof( sa ); -#if !defined (TQT_NO_IPV6) - struct sockaddr *sap = reinterpret_cast<struct sockaddr *>(&sa); - if ( !::getsockname(fd, sap, &sz) ) { - switch ( sap->sa_family ) { - case AF_INET: - return IPv4; - case AF_INET6: - return IPv6; - default: - return Unknown; - } - } -#else - if ( !::getsockname(fd, &sa, &sz) ) { - switch ( sa.sa_family ) { - case AF_INET: - return IPv4; - default: - return Unknown; - } - } -#endif - } - return Unknown; -} - -/*! - Creates a new socket identifier. Returns -1 if there is a failure - to create the new identifier; error() explains why. - - \sa setSocket() -*/ - -int TQSocketDevice::createNewSocket() -{ -#if !defined(TQT_NO_IPV6) - int s = qt_socket_socket( protocol() == IPv6 ? AF_INET6 : AF_INET, - t == Datagram ? SOCK_DGRAM : SOCK_STREAM, 0 ); -#else - int s = qt_socket_socket( AF_INET, t==Datagram?SOCK_DGRAM:SOCK_STREAM, 0 ); -#endif - if ( s < 0 ) { - switch( errno ) { - case EPROTONOSUPPORT: - e = InternalError; // 0 is supposed to work for both types - break; - case ENFILE: - e = NoFiles; // special case for this - break; - case EACCES: - e = Inaccessible; - break; - case ENOBUFS: - case ENOMEM: - e = NoResources; - break; - case EINVAL: - e = Impossible; - break; - default: - e = UnknownError; - break; - } - } else { - // ensure that the socket is closed on exec..() after being dup()'ed by - // fork() in TQProcess. - ::fcntl(s, F_SETFD, FD_CLOEXEC); - return s; - } - return -1; -} - -/*! - \reimp - - Closes the socket and sets the socket identifier to -1 (invalid). - - (This function ignores errors; if there are any then a file - descriptor leakage might result. As far as we know, the only error - that can arise is EBADF, and that would of course not cause - leakage. There may be OS-specfic errors that we haven't come - across, however.) - - \sa open() -*/ -void TQSocketDevice::close() -{ - if ( fd == -1 || !isOpen() ) // already closed - return; - setFlags( IO_Sequential ); - resetqStatus(); - setState( 0 ); - ::close( fd ); -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice::close: Closed socket %x", fd ); -#endif - fd = -1; - fetchConnectionParameters(); -} - - -/*! - Returns TRUE if the socket is valid and in blocking mode; - otherwise returns FALSE. - - Note that this function does not set error(). - - \warning On Windows, this function always returns TRUE since the - ioctlsocket() function is broken. - - \sa setBlocking(), isValid() -*/ -bool TQSocketDevice::blocking() const -{ - if ( !isValid() ) - return TRUE; - int s = fcntl(fd, F_GETFL, 0); - return !(s >= 0 && ((s & O_NDELAY) != 0)); -} - - -/*! - Makes the socket blocking if \a enable is TRUE or nonblocking if - \a enable is FALSE. - - Sockets are blocking by default, but we recommend using - nonblocking socket operations, especially for GUI programs that - need to be responsive. - - \warning On Windows, this function should be used with care since - whenever you use a TQSocketNotifier on Windows, the socket is - immediately made nonblocking. - - \sa blocking(), isValid() -*/ -void TQSocketDevice::setBlocking( bool enable ) -{ -#if defined(TQSOCKETDEVICE_DEBUG) - qDebug( "TQSocketDevice::setBlocking( %d )", enable ); -#endif - if ( !isValid() ) - return; - int tmp = ::fcntl(fd, F_GETFL, 0); - if ( tmp >= 0 ) - tmp = ::fcntl( fd, F_SETFL, enable ? (tmp&~O_NDELAY) : (tmp|O_NDELAY) ); - if ( tmp >= 0 ) - return; - if ( e ) - return; - switch( errno ) { - case EACCES: - case EBADF: - e = Impossible; - break; - case EFAULT: - case EAGAIN: -#if EAGAIN != EWOULDBLOCK - case EWOULDBLOCK: -#endif - case EDEADLK: - case EINTR: - case EINVAL: - case EMFILE: - case ENOLCK: - case EPERM: - default: - e = UnknownError; - } -} - - -/*! - Returns the value of the socket option \a opt. -*/ -int TQSocketDevice::option( Option opt ) const -{ - if ( !isValid() ) - return -1; - int n = -1; - int v = -1; - switch ( opt ) { - case Broadcast: - n = SO_BROADCAST; - break; - case ReceiveBuffer: - n = SO_RCVBUF; - break; - case ReuseAddress: - n = SO_REUSEADDR; - break; - case SendBuffer: - n = SO_SNDBUF; - break; - } - if ( n != -1 ) { - TQT_SOCKOPTLEN_T len; - len = sizeof(v); - int r = ::getsockopt( fd, SOL_SOCKET, n, (char*)&v, &len ); - if ( r >= 0 ) - return v; - if ( !e ) { - TQSocketDevice *that = (TQSocketDevice*)this; // mutable function - switch( errno ) { - case EBADF: - case ENOTSOCK: - that->e = Impossible; - break; - case EFAULT: - that->e = InternalError; - break; - default: - that->e = UnknownError; - break; - } - } - return -1; - } - return v; -} - - -/*! - Sets the socket option \a opt to \a v. -*/ -void TQSocketDevice::setOption( Option opt, int v ) -{ - if ( !isValid() ) - return; - int n = -1; // for really, really bad compilers - switch ( opt ) { - case Broadcast: - n = SO_BROADCAST; - break; - case ReceiveBuffer: - n = SO_RCVBUF; - break; - case ReuseAddress: - n = SO_REUSEADDR; - break; - case SendBuffer: - n = SO_SNDBUF; - break; - default: - return; - } - if ( ::setsockopt( fd, SOL_SOCKET, n, (char*)&v, sizeof(v)) < 0 && - e == NoError ) { - switch( errno ) { - case EBADF: - case ENOTSOCK: - e = Impossible; - break; - case EFAULT: - e = InternalError; - break; - default: - e = UnknownError; - break; - } - } -} - - -/*! - Connects to the IP address and port specified by \a addr and \a - port. Returns TRUE if it establishes a connection; otherwise returns FALSE. - If it returns FALSE, error() explains why. - - Note that error() commonly returns NoError for non-blocking - sockets; this just means that you can call connect() again in a - little while and it'll probably succeed. -*/ -bool TQSocketDevice::connect( const TQHostAddress &addr, TQ_UINT16 port ) -{ - if ( !isValid() ) - return FALSE; - - pa = addr; - pp = port; - - struct sockaddr_in a4; - struct sockaddr *aa; - TQT_SOCKLEN_T aalen; - -#if !defined(TQT_NO_IPV6) - struct sockaddr_in6 a6; - - if ( addr.isIPv6Address() ) { - memset( &a6, 0, sizeof(a6) ); - a6.sin6_family = AF_INET6; - a6.sin6_port = htons( port ); - TQ_IPV6ADDR ip6 = addr.toIPv6Address(); - memcpy( &a6.sin6_addr.s6_addr, &ip6, sizeof(ip6) ); - - aalen = sizeof( a6 ); - aa = (struct sockaddr *)&a6; - } else -#endif - if ( addr.isIPv4Address() ) { - memset( &a4, 0, sizeof(a4) ); - a4.sin_family = AF_INET; - a4.sin_port = htons( port ); - a4.sin_addr.s_addr = htonl( addr.toIPv4Address() ); - - aalen = sizeof(a4); - aa = (struct sockaddr *)&a4; - } else { - e = Impossible; - return FALSE; - } - - int r = qt_socket_connect( fd, aa, aalen ); - if ( r == 0 ) { - fetchConnectionParameters(); - return TRUE; - } - if ( errno == EISCONN || errno == EALREADY || errno == EINPROGRESS ) { - fetchConnectionParameters(); - return TRUE; - } - if ( e != NoError || errno == EAGAIN || errno == EWOULDBLOCK ) { - return FALSE; - } - switch( errno ) { - case EBADF: - case ENOTSOCK: - e = Impossible; - break; - case EFAULT: - case EAFNOSUPPORT: - e = InternalError; - break; - case ECONNREFUSED: - e = ConnectionRefused; - break; - case ETIMEDOUT: - case ENETUNREACH: - e = NetworkFailure; - break; - case EADDRINUSE: - e = NoResources; - break; - case EACCES: - case EPERM: - e = Inaccessible; - break; - default: - e = UnknownError; - break; - } - return FALSE; -} - - -/*! - Assigns a name to an unnamed socket. The name is the host address - \a address and the port number \a port. If the operation succeeds, - bind() returns TRUE; otherwise it returns FALSE without changing - what port() and address() return. - - bind() is used by servers for setting up incoming connections. - Call bind() before listen(). -*/ -bool TQSocketDevice::bind( const TQHostAddress &address, TQ_UINT16 port ) -{ - if ( !isValid() ) - return FALSE; - int r; - struct sockaddr_in a4; -#if !defined(TQT_NO_IPV6) - struct sockaddr_in6 a6; - - if ( address.isIPv6Address() ) { - memset( &a6, 0, sizeof(a6) ); - a6.sin6_family = AF_INET6; - a6.sin6_port = htons( port ); - TQ_IPV6ADDR tmp = address.toIPv6Address(); - memcpy( &a6.sin6_addr.s6_addr, &tmp, sizeof(tmp) ); - - r = qt_socket_bind( fd, (struct sockaddr *)&a6, sizeof(a6) ); - } else -#endif - if ( address.isIPv4Address() ) { - memset( &a4, 0, sizeof(a4) ); - a4.sin_family = AF_INET; - a4.sin_port = htons( port ); - a4.sin_addr.s_addr = htonl( address.toIPv4Address() ); - - r = qt_socket_bind( fd, (struct sockaddr*)&a4, sizeof(a4) ); - } else { - e = Impossible; - return FALSE; - } - - if ( r < 0 ) { - switch( errno ) { - case EINVAL: - e = AlreadyBound; - break; - case EACCES: - e = Inaccessible; - break; - case ENOMEM: - e = NoResources; - break; - case EFAULT: // a was illegal - case ENAMETOOLONG: // sz was wrong - e = InternalError; - break; - case EBADF: // AF_UNIX only - case ENOTSOCK: // AF_UNIX only - case EROFS: // AF_UNIX only - case ENOENT: // AF_UNIX only - case ENOTDIR: // AF_UNIX only - case ELOOP: // AF_UNIX only - e = Impossible; - break; - default: - e = UnknownError; - break; - } - return FALSE; - } - fetchConnectionParameters(); - return TRUE; -} - - -/*! - Specifies how many pending connections a server socket can have. - Returns TRUE if the operation was successful; otherwise returns - FALSE. A \a backlog value of 50 is quite common. - - The listen() call only applies to sockets where type() is \c - Stream, i.e. not to \c Datagram sockets. listen() must not be - called before bind() or after accept(). - - \sa bind(), accept() -*/ -bool TQSocketDevice::listen( int backlog ) -{ - if ( !isValid() ) - return FALSE; - if ( qt_socket_listen( fd, backlog ) >= 0 ) - return TRUE; - if ( !e ) - e = Impossible; - return FALSE; -} - - -/*! - Extracts the first connection from the queue of pending - connections for this socket and returns a new socket identifier. - Returns -1 if the operation failed. - - \sa bind(), listen() -*/ -int TQSocketDevice::accept() -{ - if ( !isValid() ) - return -1; - -#if !defined (TQT_NO_IPV6) - struct sockaddr_storage aa; -#else - struct sockaddr aa; -#endif - TQT_SOCKLEN_T l = sizeof( aa ); - bool done; - int s; - do { - s = qt_socket_accept( fd, (struct sockaddr*)&aa, &l ); - // we'll blithely throw away the stuff accept() wrote to aa - done = TRUE; - if ( s < 0 && e == NoError ) { - switch( errno ) { - case EINTR: - done = FALSE; - break; -#if defined(EPROTO) - case EPROTO: -#endif -#if defined(ENONET) - case ENONET: -#endif - case ENOPROTOOPT: - case EHOSTDOWN: - case EOPNOTSUPP: - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - // in all these cases, an error happened during connection - // setup. we're not interested in what happened, so we - // just treat it like the client-closed-quickly case. - case EPERM: - // firewalling wouldn't let us accept. we treat it like - // the client-closed-quickly case. - case EAGAIN: -#if EAGAIN != EWOULDBLOCK - case EWOULDBLOCK: -#endif - // the client closed the connection before we got around - // to accept()ing it. - break; - case EBADF: - case ENOTSOCK: - e = Impossible; - break; - case EFAULT: - e = InternalError; - break; - case ENOMEM: - case ENOBUFS: - e = NoResources; - break; - default: - e = UnknownError; - break; - } - } - } while (!done); - // ensure that the socket is closed on exec..() after being dup()'ed by - // fork() in TQProcess. - ::fcntl(s, F_SETFD, FD_CLOEXEC); - return s; -} - - -/*! - Returns the number of bytes available for reading, or -1 if an - error occurred. - - \warning On Microsoft Windows, we use the ioctlsocket() function - to determine the number of bytes queued on the socket. According - to Microsoft (KB Q125486), ioctlsocket() sometimes returns an - incorrect number. The only safe way to determine the amount of - data on the socket is to read it using readBlock(). TQSocket has - workarounds to deal with this problem. -*/ -#ifdef USE_QT4 -qint64 TQSocketDevice::bytesAvailable() const -#else // USE_QT4 -TQ_LONG TQSocketDevice::bytesAvailable() const -#endif // USE_QT4 -{ - if ( !isValid() ) - return -1; - - /* - Apparently, there is not consistency among different operating - systems on how to use FIONREAD. - - FreeBSD, Linux and Solaris all expect the 3rd argument to - ioctl() to be an int, which is normally 32-bit even on 64-bit - machines. - - IRIX, on the other hand, expects a size_t, which is 64-bit on - 64-bit machines. - - So, the solution is to use size_t initialized to zero to make - sure all bits are set to zero, preventing underflow with the - FreeBSD/Linux/Solaris ioctls. - */ - size_t nbytes = 0; - // gives shorter than true amounts on Unix domain sockets. - if ( ::ioctl(fd, FIONREAD, (char*)&nbytes) < 0 ) - return -1; - return (TQ_LONG) *((int *) &nbytes); -} - - -/*! - Wait up to \a msecs milliseconds for more data to be available. If - \a msecs is -1 the call will block indefinitely. - - Returns the number of bytes available for reading, or -1 if an - error occurred. - - If \a timeout is non-null and no error occurred (i.e. it does not - return -1): this function sets \a *timeout to TRUE, if the reason - for returning was that the timeout was reached; otherwise it sets - \a *timeout to FALSE. This is useful to find out if the peer - closed the connection. - - \warning This is a blocking call and should be avoided in event - driven applications. - - \sa bytesAvailable() -*/ -TQ_LONG TQSocketDevice::waitForMore( int msecs, bool *timeout ) const -{ - if ( !isValid() ) - return -1; - if ( fd >= FD_SETSIZE ) - return -1; - - fd_set fds; - struct timeval tv; - - FD_ZERO( &fds ); - FD_SET( fd, &fds ); - - tv.tv_sec = msecs / 1000; - tv.tv_usec = (msecs % 1000) * 1000; - - int rv = select( fd+1, &fds, 0, 0, msecs < 0 ? 0 : &tv ); - - if ( rv < 0 ) - return -1; - - if ( timeout ) { - if ( rv == 0 ) - *timeout = TRUE; - else - *timeout = FALSE; - } - - return bytesAvailable(); -} - - -/*! - Reads \a maxlen bytes from the socket into \a data and returns the - number of bytes read. Returns -1 if an error occurred. Returning 0 - is not an error. For Stream sockets, 0 is returned when the remote - host closes the connection. For Datagram sockets, 0 is a valid - datagram size. -*/ -TQ_LONG TQSocketDevice::readBlock( char *data, TQ_ULONG maxlen ) -{ -#if defined(TQT_CHECK_NULL) - if ( data == 0 && maxlen != 0 ) { - qWarning( "TQSocketDevice::readBlock: Null pointer error" ); - } -#endif -#if defined(TQT_CHECK_STATE) - if ( !isValid() ) { - qWarning( "TQSocketDevice::readBlock: Invalid socket" ); - return -1; - } - if ( !isOpen() ) { - qWarning( "TQSocketDevice::readBlock: Device is not open" ); - return -1; - } - if ( !isReadable() ) { - qWarning( "TQSocketDevice::readBlock: Read operation not permitted" ); - return -1; - } -#endif - bool done = FALSE; - int r = 0; - while ( done == FALSE ) { - if ( t == Datagram ) { -#if !defined(TQT_NO_IPV6) - struct sockaddr_storage aa; -#else - struct sockaddr_in aa; -#endif - memset( &aa, 0, sizeof(aa) ); - TQT_SOCKLEN_T sz; - sz = sizeof( aa ); - r = ::recvfrom( fd, data, maxlen, 0, - (struct sockaddr *)&aa, &sz ); - - qt_socket_getportaddr( (struct sockaddr *)&aa, &pp, &pa); - - } else { - r = ::read( fd, data, maxlen ); - } - done = TRUE; - if ( r == 0 && t == Stream && maxlen > 0 ) { - // connection closed - close(); - } else if ( r >= 0 || errno == EAGAIN || errno == EWOULDBLOCK ) { - // nothing - } else if ( errno == EINTR ) { - done = FALSE; - } else if ( e == NoError ) { - switch( errno ) { - case EIO: - case EISDIR: - case EBADF: - case EINVAL: - case EFAULT: - case ENOTCONN: - case ENOTSOCK: - e = Impossible; - break; -#if defined(ENONET) - case ENONET: -#endif - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - e = NetworkFailure; - break; - case EPIPE: - case ECONNRESET: - // connection closed - close(); - r = 0; - break; - default: - e = UnknownError; - break; - } - } - } - return r; -} - - -/*! - Writes \a len bytes to the socket from \a data and returns the - number of bytes written. Returns -1 if an error occurred. - - This is used for \c TQSocketDevice::Stream sockets. -*/ -TQ_LONG TQSocketDevice::writeBlock( const char *data, TQ_ULONG len ) -{ - if ( data == 0 && len != 0 ) { -#if defined(TQT_CHECK_NULL) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Null pointer error" ); -#endif - return -1; - } - if ( !isValid() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Invalid socket" ); -#endif - return -1; - } - if ( !isOpen() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Device is not open" ); -#endif - return -1; - } - if ( !isWritable() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::writeBlock: Write operation not permitted" ); -#endif - return -1; - } - bool done = FALSE; - int r = 0; - bool timeout; - while ( !done ) { - r = ::write( fd, data, len ); - done = TRUE; - if ( r < 0 && e == NoError && - errno != EAGAIN && errno != EWOULDBLOCK ) { - switch( errno ) { - case EINTR: // signal - call read() or whatever again - done = FALSE; - break; - case EPIPE: - case ECONNRESET: - // connection closed - close(); - r = 0; - break; - case ENOSPC: - case EIO: - case EISDIR: - case EBADF: - case EINVAL: - case EFAULT: - case ENOTCONN: - case ENOTSOCK: - e = Impossible; - break; -#if defined(ENONET) - case ENONET: -#endif - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - e = NetworkFailure; - break; - default: - e = UnknownError; - break; - } - } else if ( waitForMore( 0, &timeout ) == 0 ) { - if ( !timeout ) { - // connection closed - close(); - } - } - } - return r; -} - - -/*! - \overload - - Writes \a len bytes to the socket from \a data and returns the - number of bytes written. Returns -1 if an error occurred. - - This is used for \c TQSocketDevice::Datagram sockets. You must - specify the \a host and \a port of the destination of the data. -*/ -TQ_LONG TQSocketDevice::writeBlock( const char * data, TQ_ULONG len, - const TQHostAddress & host, TQ_UINT16 port ) -{ - if ( t != Datagram ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::sendBlock: Not datagram" ); -#endif - return -1; // for now - later we can do t/tcp - } - - if ( data == 0 && len != 0 ) { -#if defined(TQT_CHECK_NULL) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::sendBlock: Null pointer error" ); -#endif - return -1; - } - if ( !isValid() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::sendBlock: Invalid socket" ); -#endif - return -1; - } - if ( !isOpen() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::sendBlock: Device is not open" ); -#endif - return -1; - } - if ( !isWritable() ) { -#if defined(TQT_CHECK_STATE) || defined(TQSOCKETDEVICE_DEBUG) - qWarning( "TQSocketDevice::sendBlock: Write operation not permitted" ); -#endif - return -1; - } - struct sockaddr_in a4; - struct sockaddr *aa; - TQT_SOCKLEN_T slen; -#if !defined(TQT_NO_IPV6) - struct sockaddr_in6 a6; - if ( host.isIPv6Address() ) { - memset( &a6, 0, sizeof(a6) ); - a6.sin6_family = AF_INET6; - a6.sin6_port = htons( port ); - - TQ_IPV6ADDR tmp = host.toIPv6Address(); - memcpy( &a6.sin6_addr.s6_addr, &tmp, sizeof(tmp) ); - slen = sizeof( a6 ); - aa = (struct sockaddr *)&a6; - } else -#endif - if ( host.isIPv4Address() ) { - memset( &a4, 0, sizeof(a4) ); - a4.sin_family = AF_INET; - a4.sin_port = htons( port ); - a4.sin_addr.s_addr = htonl( host.toIPv4Address() ); - slen = sizeof(a4); - aa = (struct sockaddr *)&a4; - } else { - e = Impossible; - return -1; - } - - // we'd use MSG_DONTWAIT + MSG_NOTQT_SIGNAL if Stevens were right. - // but apparently Stevens and most implementors disagree - bool done = FALSE; - int r = 0; - while ( !done ) { - r = ::sendto( fd, data, len, 0, aa, slen); - done = TRUE; - if ( r < 0 && e == NoError && - errno != EAGAIN && errno != EWOULDBLOCK ) { - switch( errno ) { - case EINTR: // signal - call read() or whatever again - done = FALSE; - break; - case ENOSPC: - case EPIPE: - case EIO: - case EISDIR: - case EBADF: - case EINVAL: - case EFAULT: - case ENOTCONN: - case ENOTSOCK: - e = Impossible; - break; -#if defined(ENONET) - case ENONET: -#endif - case EHOSTUNREACH: - case ENETDOWN: - case ENETUNREACH: - case ETIMEDOUT: - e = NetworkFailure; - break; - default: - e = UnknownError; - break; - } - } - } - return r; -} - - -/*! - Fetches information about both ends of the connection: whatever is - available. -*/ -void TQSocketDevice::fetchConnectionParameters() -{ - if ( !isValid() ) { - p = 0; - a = TQHostAddress(); - pp = 0; - pa = TQHostAddress(); - return; - } -#if !defined(TQT_NO_IPV6) - struct sockaddr_storage sa; -#else - struct sockaddr_in sa; -#endif - memset( &sa, 0, sizeof(sa) ); - TQT_SOCKLEN_T sz; - sz = sizeof( sa ); - if ( !::getsockname( fd, (struct sockaddr *)(&sa), &sz ) ) - qt_socket_getportaddr( (struct sockaddr *)&sa, &p, &a ); - - sz = sizeof( sa ); - if ( !::getpeername( fd, (struct sockaddr *)(&sa), &sz ) ) - qt_socket_getportaddr( (struct sockaddr *)&sa, &pp, &pa ); -} - - -/*! - Returns the port number of the port this socket tqdevice is - connected to. This may be 0 for a while, but is set to something - sensible as soon as a sensible value is available. - - Note that for Datagram sockets, this is the source port of the - last packet received, and that it is in native byte order. -*/ -TQ_UINT16 TQSocketDevice::peerPort() const -{ - return pp; -} - - -/*! - Returns the address of the port this socket tqdevice is connected - to. This may be 0.0.0.0 for a while, but is set to something - sensible as soon as a sensible value is available. - - Note that for Datagram sockets, this is the source port of the - last packet received. -*/ -TQHostAddress TQSocketDevice::peerAddress() const -{ - return pa; -} - -#endif //TQT_NO_NETWORK |