You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
qt3/src/kernel/qlocalfs.cpp

408 lines
11 KiB

/****************************************************************************
**
** Implementation of QLocalFs class
**
** Created : 950429
**
** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
**
** This file is part of the kernel module of the Qt 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 Qt 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 sales@trolltech.com.
**
** This file may be used under the terms of the Q Public License as
** defined by Trolltech ASA and appearing in the file LICENSE.QPL
** included in the packaging of this file. Licensees holding valid Qt
** Commercial licenses may use this file in accordance with the Qt
** 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 "qlocalfs.h"
#ifndef QT_NO_NETWORKPROTOCOL
#include "qfileinfo.h"
#include "qfile.h"
#include "qurlinfo.h"
#include "qapplication.h"
#include "qurloperator.h"
#include "qguardedptr.h"
//#define QLOCALFS_DEBUG
/*!
\class QLocalFs qlocalfs.h
\brief The QLocalFs class is an implementation of a
QNetworkProtocol that works on the local file system.
\if defined(commercial)
It is part of the <a href="commercialeditions.html">Qt Enterprise Edition</a>.
\endif
\module network
\ingroup io
This class is derived from QNetworkProtocol. QLocalFs is not
normally used directly, but rather through a QUrlOperator, for
example:
\code
QUrlOperator op( "file:///tmp" );
op.listChildren(); // Asks the server to provide a directory listing
\endcode
This code will only work if the QLocalFs class is registered; to
register the class, you must call qInitNetworkProtocols() before
using a QUrlOperator with QLocalFs.
If you really need to use QLocalFs directly, don't forget
to set its QUrlOperator with setUrl().
\sa \link network.html Qt Network Documentation \endlink QNetworkProtocol, QUrlOperator
*/
/*!
Constructor.
*/
QLocalFs::QLocalFs()
: QNetworkProtocol()
{
}
static int convertPermissions(QFileInfo *fi)
{
int p = 0;
if ( fi->permission( QFileInfo::ReadOwner ) )
p |= QUrlInfo::ReadOwner;
if ( fi->permission( QFileInfo::WriteOwner ) )
p |= QUrlInfo::WriteOwner;
if ( fi->permission( QFileInfo::ExeOwner ) )
p |= QUrlInfo::ExeOwner;
if ( fi->permission( QFileInfo::ReadGroup ) )
p |= QUrlInfo::ReadGroup;
if ( fi->permission( QFileInfo::WriteGroup ) )
p |= QUrlInfo::WriteGroup;
if ( fi->permission( QFileInfo::ExeGroup ) )
p |= QUrlInfo::ExeGroup;
if ( fi->permission( QFileInfo::ReadOther ) )
p |= QUrlInfo::ReadOther;
if ( fi->permission( QFileInfo::WriteOther ) )
p |= QUrlInfo::WriteOther;
if ( fi->permission( QFileInfo::ExeOther ) )
p |= QUrlInfo::ExeOther;
return p;
}
/*!
\reimp
*/
void QLocalFs::operationListChildren( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationListChildren" );
#endif
op->setState( StInProgress );
dir = QDir( url()->path() );
dir.setNameFilter( url()->nameFilter() );
dir.setMatchAllDirs( TRUE );
if ( !dir.isReadable() ) {
QString msg = tr( "Could not read directory\n%1" ).arg( url()->path() );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrListChildren );
emit finished( op );
return;
}
const QFileInfoList *filist = dir.entryInfoList( QDir::All | QDir::Hidden | QDir::System );
if ( !filist ) {
QString msg = tr( "Could not read directory\n%1" ).arg( url()->path() );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrListChildren );
emit finished( op );
return;
}
emit start( op );
QFileInfoListIterator it( *filist );
QFileInfo *fi;
QValueList<QUrlInfo> infos;
while ( ( fi = it.current() ) != 0 ) {
++it;
infos << QUrlInfo( fi->fileName(), convertPermissions(fi), fi->owner(), fi->group(),
fi->size(), fi->lastModified(), fi->lastRead(), fi->isDir(), fi->isFile(),
fi->isSymLink(), fi->isWritable(), fi->isReadable(), fi->isExecutable() );
}
emit newChildren( infos, op );
op->setState( StDone );
emit finished( op );
}
/*!
\reimp
*/
void QLocalFs::operationMkDir( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationMkDir" );
#endif
op->setState( StInProgress );
QString dirname = op->arg( 0 );
dir = QDir( url()->path() );
if ( dir.mkdir( dirname ) ) {
QFileInfo fi( dir, dirname );
QUrlInfo inf( fi.fileName(), convertPermissions(&fi), fi.owner(), fi.group(),
fi.size(), fi.lastModified(), fi.lastRead(), fi.isDir(), fi.isFile(),
fi.isSymLink(), fi.isWritable(), fi.isReadable(), fi.isExecutable() );
emit newChild( inf, op );
op->setState( StDone );
emit createdDirectory( inf, op );
emit finished( op );
} else {
QString msg = tr( "Could not create directory\n%1" ).arg( dirname );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrMkDir );
emit finished( op );
}
}
/*!
\reimp
*/
void QLocalFs::operationRemove( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationRemove" );
#endif
op->setState( StInProgress );
QString name = QUrl( op->arg( 0 ) ).path();
bool deleted = FALSE;
dir = QDir( url()->path() );
QFileInfo fi( dir, name );
if ( fi.isDir() ) {
if ( dir.rmdir( name ) )
deleted = TRUE;
}
if ( deleted || dir.remove( name ) ) {
op->setState( StDone );
emit removed( op );
emit finished( op );
} else {
QString msg = tr( "Could not remove file or directory\n%1" ).arg( name );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrRemove );
emit finished( op );
}
}
/*!
\reimp
*/
void QLocalFs::operationRename( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationRename" );
#endif
op->setState( StInProgress );
QString oldname = op->arg( 0 );
QString newname = op->arg( 1 );
dir = QDir( url()->path() );
if ( dir.rename( oldname, newname ) ) {
op->setState( StDone );
emit itemChanged( op );
emit finished( op );
} else {
QString msg = tr( "Could not rename\n%1\nto\n%2" ).arg( oldname ).arg( newname );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrRename );
emit finished( op );
}
}
/*!
\reimp
*/
void QLocalFs::operationGet( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationGet" );
#endif
op->setState( StInProgress );
QString from = QUrl( op->arg( 0 ) ).path();
QFile f( from );
if ( !f.open( IO_ReadOnly ) ) {
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: could not open %s", from.latin1() );
#endif
QString msg = tr( "Could not open\n%1" ).arg( from );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrGet );
emit finished( op );
return;
}
QByteArray s;
emit dataTransferProgress( 0, f.size(), op );
if ( f.size() != 0 ) {
int blockSize = calcBlockSize( f.size() );
if ( (int)f.size() < blockSize ) {
s.resize( f.size() );
f.readBlock( s.data(), f.size() );
emit data( s, op );
emit dataTransferProgress( f.size(), f.size(), op );
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: got all %d bytes at once", f.size() );
#endif
} else {
s.resize( blockSize );
int remaining = f.size();
QGuardedPtr<QObject> that = this;
while ( that && remaining > 0 ) {
if ( operationInProgress() != op )
return;
if ( remaining >= blockSize ) {
f.readBlock( s.data(), blockSize );
emit data( s, op );
emit dataTransferProgress( f.size() - remaining, f.size(), op );
remaining -= blockSize;
} else {
s.resize( remaining );
f.readBlock( s.data(), remaining );
emit data( s, op );
emit dataTransferProgress( f.size() - remaining, f.size(), op );
remaining -= remaining;
}
qApp->processEvents();
}
if (!that)
return;
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: got all %d bytes step by step", f.size() );
#endif
emit dataTransferProgress( f.size(), f.size(), op );
}
}
op->setState( StDone );
f.close();
emit finished( op );
}
/*!
\reimp
*/
void QLocalFs::operationPut( QNetworkOperation *op )
{
#ifdef QLOCALFS_DEBUG
qDebug( "QLocalFs: operationPut" );
#endif
op->setState( StInProgress );
QString to = QUrl( op->arg( 0 ) ).path();
QFile f( to );
if ( !f.open( IO_WriteOnly ) ) {
QString msg = tr( "Could not write\n%1" ).arg( to );
op->setState( StFailed );
op->setProtocolDetail( msg );
op->setErrorCode( (int)ErrPut );
emit finished( op );
return;
}
QByteArray ba( op->rawArg( 1 ) );
emit dataTransferProgress( 0, ba.size(), op );
int blockSize = calcBlockSize( ba.size() );
if ( (int)ba.size() < blockSize ) {
f.writeBlock( ba.data(), ba.size() );
emit dataTransferProgress( ba.size(), ba.size(), op );
} else {
int i = 0;
while ( i + blockSize < (int)ba.size() - 1 ) {
if ( operationInProgress() != op )
return;
f.writeBlock( &ba.data()[ i ], blockSize );
f.flush();
emit dataTransferProgress( i + blockSize, ba.size(), op );
i += blockSize;
QGuardedPtr<QObject> that = this;
qApp->processEvents();
if (!that)
return;
}
if ( i < (int)ba.size() - 1 )
f.writeBlock( &ba.data()[ i ], ba.size() - i );
emit dataTransferProgress( ba.size(), ba.size(), op );
}
op->setState( StDone );
f.close();
emit finished( op );
}
/*!
\reimp
*/
int QLocalFs::supportedOperations() const
{
return OpListChildren | OpMkDir | OpRemove | OpRename | OpGet | OpPut;
}
/*!
\internal
*/
int QLocalFs::calcBlockSize( int totalSize ) const
{
if ( totalSize == 0 )
return 1024;
int s = totalSize / 100;
// we want a block size between 1KB and 1MB
if ( s < 1024 )
s = 1024;
if ( s > 1048576 )
s = 1048576;
return s;
}
#endif // QT_NO_NETWORKPROTOCOL