diff options
Diffstat (limited to 'kio/kioexec')
-rw-r--r-- | kio/kioexec/Makefile.am | 19 | ||||
-rw-r--r-- | kio/kioexec/README | 26 | ||||
-rw-r--r-- | kio/kioexec/main.cpp | 294 | ||||
-rw-r--r-- | kio/kioexec/main.h | 35 |
4 files changed, 374 insertions, 0 deletions
diff --git a/kio/kioexec/Makefile.am b/kio/kioexec/Makefile.am new file mode 100644 index 000000000..8de01676c --- /dev/null +++ b/kio/kioexec/Makefile.am @@ -0,0 +1,19 @@ +AM_CPPFLAGS = $(all_includes) + +bin_PROGRAMS = kioexec + +kioexec_SOURCES = main.cpp +kioexec_LDFLAGS = $(all_libraries) $(KDE_RPATH) +kioexec_LDADD = $(LIB_KIO) + +noinst_HEADERS = main.h +METASOURCES = AUTO + +messages: + $(XGETTEXT) $(kioexec_SOURCES) -o $(podir)/kioexec.pot + +install-exec-local: + @rm -f $(DESTDIR)$(bindir)/kfmexec + @$(LN_S) kioexec $(DESTDIR)$(bindir)/kfmexec + +include $(top_srcdir)/admin/Doxyfile.am diff --git a/kio/kioexec/README b/kio/kioexec/README new file mode 100644 index 000000000..9f844bb97 --- /dev/null +++ b/kio/kioexec/README @@ -0,0 +1,26 @@ +kfmexec is launched when the user wants to open a remote file with +an application that only supports local files. + +For this it does the following: +- downloads a remote file to a temp location +- starts a 'local' application with that temp file as argument +- wait fors application to be exited +- if the modification time of the file is different from the original one, +(because the file was modified) then it offers re-uploading the modified version. +This is how you offer network transparency to apps that don't have it. + +BUT: with KUniqueApplication, this breaks, because the app returns at once, +so we have no way to know when the user finished editing the file... + +Conclusion: if the application has network transparency built-in, it should +put "%u" in its desktop file - and kfmexec isn't used -. If it doesn't, either +it's a KApplication and kfmexec does its job, or it's a KUniqueApplication +and... kfmexec can't see modifications to the file. Be warned. + +From Waldo: "The program doesn't return _at once_. It returns +after "newInstance()" returns. So if you open the file there it will still work. +(Or rename it)" + +David Faure <faure@kde.org> +20-May-2000 + diff --git a/kio/kioexec/main.cpp b/kio/kioexec/main.cpp new file mode 100644 index 000000000..12a2d64bf --- /dev/null +++ b/kio/kioexec/main.cpp @@ -0,0 +1,294 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2000 David Faure <faure@kde.org> + Copyright (C) 2001 Waldo Bastian <bastian@kde.org> + + This program is free software; you can redistribute it and/or + modify it under the terms of the GNU General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; see the file COPYING. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include <config.h> +#include <unistd.h> +#include <stdlib.h> +#include <sys/stat.h> + +#include <qfile.h> + +#include <kapplication.h> +#include <kstandarddirs.h> +#include <kdebug.h> +#include <kmessagebox.h> +#include <kio/job.h> +#include <krun.h> +#include <kio/netaccess.h> +#include <kprocess.h> +#include <kservice.h> +#include <klocale.h> +#include <kcmdlineargs.h> +#include <kaboutdata.h> +#include <kstartupinfo.h> +#include <kshell.h> +#include <kde_file.h> + + +#include "main.h" + + +static const char description[] = + I18N_NOOP("KIO Exec - Opens remote files, watches modifications, asks for upload"); + +static KCmdLineOptions options[] = +{ + { "tempfiles", I18N_NOOP("Treat URLs as local files and delete them afterwards"), 0 }, + { "suggestedfilename <file name>", I18N_NOOP("Suggested file name for the downloaded file"), 0 }, + { "+command", I18N_NOOP("Command to execute"), 0 }, + { "+[URLs]", I18N_NOOP("URL(s) or local file(s) used for 'command'"), 0 }, + KCmdLineLastOption +}; + + +int jobCounter = 0; + +QPtrList<KIO::Job>* jobList = 0L; + +KIOExec::KIOExec() +{ + jobList = new QPtrList<KIO::Job>; + jobList->setAutoDelete( false ); // jobs autodelete themselves + + KCmdLineArgs *args = KCmdLineArgs::parsedArgs(); + if (args->count() < 1) + KCmdLineArgs::usage(i18n("'command' expected.\n")); + + tempfiles = args->isSet("tempfiles"); + if ( args->isSet( "suggestedfilename" ) ) + suggestedFileName = QString::fromLocal8Bit( args->getOption( "suggestedfilename" ) ); + expectedCounter = 0; + command = args->arg(0); + kdDebug() << "command=" << command << endl; + + for ( int i = 1; i < args->count(); i++ ) + { + KURL url = args->url(i); + // we need to map system:/ etc to make sure we get this right + url = KIO::NetAccess::mostLocalURL( url, 0 ); + + //kdDebug() << "url=" << url.url() << " filename=" << url.fileName() << endl; + // A local file, not an URL ? + // => It is not encoded and not shell escaped, too. + if ( url.isLocalFile() ) + { + fileInfo file; + file.path = url.path(); + file.url = url; + fileList.append(file); + } + // It is an URL + else + { + if ( !url.isValid() ) + KMessageBox::error( 0L, i18n( "The URL %1\nis malformed" ).arg( url.url() ) ); + else if ( tempfiles ) + KMessageBox::error( 0L, i18n( "Remote URL %1\nnot allowed with --tempfiles switch" ).arg( url.url() ) ); + else + // We must fetch the file + { + QString fileName = KIO::encodeFileName( url.fileName() ); + if ( !suggestedFileName.isEmpty() ) + fileName = suggestedFileName; + // Build the destination filename, in ~/.kde/cache-*/krun/ + // Unlike KDE-1.1, we put the filename at the end so that the extension is kept + // (Some programs rely on it) + QString tmp = KGlobal::dirs()->saveLocation( "cache", "krun/" ) + + QString("%1.%2.%3").arg(getpid()).arg(jobCounter++).arg(fileName); + fileInfo file; + file.path = tmp; + file.url = url; + fileList.append(file); + + expectedCounter++; + KURL dest; + dest.setPath( tmp ); + kdDebug() << "Copying " << url.prettyURL() << " to " << dest << endl; + KIO::Job *job = KIO::file_copy( url, dest ); + jobList->append( job ); + + connect( job, SIGNAL( result( KIO::Job * ) ), SLOT( slotResult( KIO::Job * ) ) ); + } + } + } + args->clear(); + + if ( tempfiles ) { + // #113991 + QTimer::singleShot( 0, this, SLOT( slotRunApp() ) ); + //slotRunApp(); // does not return + return; + } + + counter = 0; + if ( counter == expectedCounter ) + slotResult( 0L ); +} + +void KIOExec::slotResult( KIO::Job * job ) +{ + if (job && job->error()) + { + // That error dialog would be queued, i.e. not immediate... + //job->showErrorDialog(); + if ( (job->error() != KIO::ERR_USER_CANCELED) ) + KMessageBox::error( 0L, job->errorString() ); + + QString path = static_cast<KIO::FileCopyJob*>(job)->destURL().path(); + + QValueList<fileInfo>::Iterator it = fileList.begin(); + for(;it != fileList.end(); ++it) + { + if ((*it).path == path) + break; + } + + if ( it != fileList.end() ) + fileList.remove( it ); + else + kdDebug() << static_cast<KIO::FileCopyJob*>(job)->destURL().path() << " not found in list" << endl; + } + + counter++; + + if ( counter < expectedCounter ) + return; + + kdDebug() << "All files downloaded, will call slotRunApp shortly" << endl; + // We know we can run the app now - but let's finish the job properly first. + QTimer::singleShot( 0, this, SLOT( slotRunApp() ) ); + + jobList->clear(); +} + +void KIOExec::slotRunApp() +{ + if ( fileList.isEmpty() ) { + kdDebug() << k_funcinfo << "No files downloaded -> exiting" << endl; + exit(1); + } + + KService service("dummy", command, QString::null); + + KURL::List list; + // Store modification times + QValueList<fileInfo>::Iterator it = fileList.begin(); + for ( ; it != fileList.end() ; ++it ) + { + KDE_struct_stat buff; + (*it).time = KDE_stat( QFile::encodeName((*it).path), &buff ) ? 0 : buff.st_mtime; + KURL url; + url.setPath((*it).path); + list << url; + } + + QStringList params = KRun::processDesktopExec(service, list, false /*no shell*/); + + kdDebug() << "EXEC " << KShell::joinArgs( params ) << endl; + +#ifdef Q_WS_X11 + // propagate the startup indentification to the started process + KStartupInfoId id; + id.initId( kapp->startupId()); + id.setupStartupEnv(); +#endif + + KProcess proc; + proc << params; + proc.start( KProcess::Block ); + +#ifdef Q_WS_X11 + KStartupInfo::resetStartupEnv(); +#endif + + kdDebug() << "EXEC done" << endl; + + // Test whether one of the files changed + it = fileList.begin(); + for( ;it != fileList.end(); ++it ) + { + KDE_struct_stat buff; + QString src = (*it).path; + KURL dest = (*it).url; + if ( (KDE_stat( QFile::encodeName(src), &buff ) == 0) && + ((*it).time != buff.st_mtime) ) + { + if ( tempfiles ) + { + if ( KMessageBox::questionYesNo( 0L, + i18n( "The supposedly temporary file\n%1\nhas been modified.\nDo you still want to delete it?" ).arg(dest.prettyURL()), + i18n( "File Changed" ), KStdGuiItem::del(), i18n("Do Not Delete") ) != KMessageBox::Yes ) + continue; // don't delete the temp file + } + else if ( ! dest.isLocalFile() ) // no upload when it's already a local file + { + if ( KMessageBox::questionYesNo( 0L, + i18n( "The file\n%1\nhas been modified.\nDo you want to upload the changes?" ).arg(dest.prettyURL()), + i18n( "File Changed" ), i18n("Upload"), i18n("Do Not Upload") ) == KMessageBox::Yes ) + { + kdDebug() << QString("src='%1' dest='%2'").arg(src).arg(dest.url()).ascii() << endl; + // Do it the synchronous way. + if ( !KIO::NetAccess::upload( src, dest, 0 ) ) + { + KMessageBox::error( 0L, KIO::NetAccess::lastErrorString() ); + continue; // don't delete the temp file + } + } + } + } + + if ( !dest.isLocalFile() || tempfiles ) { + // Wait for a reasonable time so that even if the application forks on startup (like OOo or amarok) + // it will have time to start up and read the file before it gets deleted. #130709. + kdDebug() << "sleeping..." << endl; + sleep(180); // 3 mn + kdDebug() << "about to delete " << src << endl; + unlink( QFile::encodeName(src) ); + } + } + + //kapp->quit(); not efficient enough + exit(0); +} + +int main( int argc, char **argv ) +{ + KAboutData aboutData( "kioexec", I18N_NOOP("KIOExec"), + VERSION, description, KAboutData::License_GPL, + "(c) 1998-2000,2003 The KFM/Konqueror Developers"); + aboutData.addAuthor("David Faure",0, "faure@kde.org"); + aboutData.addAuthor("Stephan Kulow",0, "coolo@kde.org"); + aboutData.addAuthor("Bernhard Rosenkraenzer",0, "bero@arklinux.org"); + aboutData.addAuthor("Waldo Bastian",0, "bastian@kde.org"); + aboutData.addAuthor("Oswald Buddenhagen",0, "ossi@kde.org"); + + KCmdLineArgs::init( argc, argv, &aboutData ); + KCmdLineArgs::addCmdLineOptions( options ); + + KApplication app; + + KIOExec exec; + + kdDebug() << "Constructor returned..." << endl; + return app.exec(); +} + +#include "main.moc" diff --git a/kio/kioexec/main.h b/kio/kioexec/main.h new file mode 100644 index 000000000..8fb98130f --- /dev/null +++ b/kio/kioexec/main.h @@ -0,0 +1,35 @@ +#ifndef _main_h +#define _main_h + +#include <qobject.h> +#include <qstring.h> +#include <qstrlist.h> +#include <qtimer.h> + +namespace KIO { class Job; } + +class KIOExec : public QObject +{ + Q_OBJECT +public: + KIOExec(); + +public slots: + void slotResult( KIO::Job * ); + void slotRunApp(); + +protected: + bool tempfiles; + QString suggestedFileName; + int counter; + int expectedCounter; + QString command; + struct fileInfo { + QString path; + KURL url; + int time; + }; + QValueList<fileInfo> fileList; +}; + +#endif |