mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 19:02:53 +00:00
256 lines
8 KiB
C++
256 lines
8 KiB
C++
/* This file is part of KDevelop
|
|
Copyright 2009 Andreas Pakulat <apaku@gmx.de>
|
|
Copyright 2009 Niko Sams <niko.sams@gmail.com>
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "scriptappjob.h"
|
|
#include "executescriptplugin.h"
|
|
|
|
#include <QFileInfo>
|
|
|
|
#include <kprocess.h>
|
|
#include <klocale.h>
|
|
#include <kmessagebox.h>
|
|
#include <kconfiggroup.h>
|
|
#include <kdebug.h>
|
|
#include <kparts/mainwindow.h>
|
|
|
|
#include <interfaces/ilaunchconfiguration.h>
|
|
#include <outputview/outputmodel.h>
|
|
#include <outputview/outputdelegate.h>
|
|
#include <util/processlinemaker.h>
|
|
#include <util/environmentgrouplist.h>
|
|
|
|
#include <kshell.h>
|
|
#include <interfaces/icore.h>
|
|
#include <interfaces/iuicontroller.h>
|
|
#include <interfaces/iplugincontroller.h>
|
|
#include <interfaces/idocumentcontroller.h>
|
|
#include <project/projectmodel.h>
|
|
|
|
#include "iexecutescriptplugin.h"
|
|
|
|
ScriptAppJob::ScriptAppJob(ExecuteScriptPlugin* parent, KDevelop::ILaunchConfiguration* cfg)
|
|
: KDevelop::OutputJob( parent ), proc(0)
|
|
{
|
|
kDebug() << "creating script app job";
|
|
setCapabilities(Killable);
|
|
|
|
IExecuteScriptPlugin* iface = KDevelop::ICore::self()->pluginController()->pluginForExtension("org.kdevelop.IExecuteScriptPlugin")->extension<IExecuteScriptPlugin>();
|
|
Q_ASSERT(iface);
|
|
|
|
KDevelop::EnvironmentGroupList l(KGlobal::config());
|
|
QString envgrp = iface->environmentGroup(cfg);
|
|
|
|
QString err;
|
|
QString interpreterString = iface->interpreter( cfg, err );
|
|
// check for errors happens in the executescript plugin already
|
|
KShell::Errors err_;
|
|
QStringList interpreter = KShell::splitArgs( interpreterString, KShell::TildeExpand | KShell::AbortOnMeta, &err_ );
|
|
if ( interpreter.isEmpty() ) {
|
|
// This should not happen, because of the checks done in the executescript plugin
|
|
kWarning() << "no interpreter specified";
|
|
return;
|
|
}
|
|
|
|
if( !err.isEmpty() )
|
|
{
|
|
setError( -1 );
|
|
setErrorText( err );
|
|
return;
|
|
}
|
|
|
|
KUrl script;
|
|
if( !iface->runCurrentFile( cfg ) )
|
|
{
|
|
script = iface->script( cfg, err );
|
|
} else {
|
|
KDevelop::IDocument* document = KDevelop::ICore::self()->documentController()->activeDocument();
|
|
if( !document )
|
|
{
|
|
setError( -1 );
|
|
setErrorText( i18n( "There is no active document to launch." ) );
|
|
return;
|
|
}
|
|
script = document->url();
|
|
}
|
|
|
|
if( !err.isEmpty() )
|
|
{
|
|
setError( -3 );
|
|
setErrorText( err );
|
|
return;
|
|
}
|
|
|
|
QString remoteHost = iface->remoteHost( cfg, err );
|
|
if( !err.isEmpty() )
|
|
{
|
|
setError( -4 );
|
|
setErrorText( err );
|
|
return;
|
|
}
|
|
|
|
if( envgrp.isEmpty() )
|
|
{
|
|
kWarning() << "Launch Configuration:" << cfg->name() << i18n("No environment group specified, looks like a broken "
|
|
"configuration, please check run configuration '%1'. "
|
|
"Using default environment group.", cfg->name() );
|
|
envgrp = l.defaultGroup();
|
|
}
|
|
|
|
QStringList arguments = iface->arguments( cfg, err );
|
|
if( !err.isEmpty() )
|
|
{
|
|
setError( -2 );
|
|
setErrorText( err );
|
|
}
|
|
|
|
if( error() != 0 )
|
|
{
|
|
kWarning() << "Launch Configuration:" << cfg->name() << "oops, problem" << errorText();
|
|
return;
|
|
}
|
|
|
|
KDevelop::OutputModel::OutputFilterStrategy currentFilterMode = static_cast<KDevelop::OutputModel::OutputFilterStrategy>( iface->outputFilterModeId( cfg ) );
|
|
|
|
proc = new KProcess( this );
|
|
|
|
lineMaker = new KDevelop::ProcessLineMaker( proc, this );
|
|
|
|
setStandardToolView(KDevelop::IOutputView::RunView);
|
|
setBehaviours(KDevelop::IOutputView::AllowUserClose | KDevelop::IOutputView::AutoScroll);
|
|
KDevelop::OutputModel* m = new KDevelop::OutputModel;
|
|
m->setFilteringStrategy(currentFilterMode);
|
|
setModel( m );
|
|
setDelegate( new KDevelop::OutputDelegate );
|
|
|
|
connect( lineMaker, SIGNAL(receivedStdoutLines(QStringList)), model(), SLOT(appendLines(QStringList)) );
|
|
connect( proc, SIGNAL(error(QProcess::ProcessError)), SLOT(processError(QProcess::ProcessError)) );
|
|
connect( proc, SIGNAL(finished(int,QProcess::ExitStatus)), SLOT(processFinished(int,QProcess::ExitStatus)) );
|
|
|
|
// Now setup the process parameters
|
|
|
|
proc->setEnvironment( l.createEnvironment( envgrp, proc->systemEnvironment()) );
|
|
KUrl wc = iface->workingDirectory( cfg );
|
|
if( !wc.isValid() || wc.isEmpty() )
|
|
{
|
|
wc = KUrl( QFileInfo( script.toLocalFile() ).absolutePath() );
|
|
}
|
|
proc->setWorkingDirectory( wc.toLocalFile() );
|
|
proc->setProperty( "executable", interpreter.first() );
|
|
|
|
QStringList program;
|
|
if (!remoteHost.isEmpty()) {
|
|
program << "ssh";
|
|
QStringList parts = remoteHost.split(":");
|
|
program << parts.first();
|
|
if (parts.length() > 1) {
|
|
program << "-p "+parts.at(1);
|
|
}
|
|
}
|
|
program << interpreter;
|
|
program << script.toLocalFile();
|
|
program << arguments;
|
|
|
|
kDebug() << "setting app:" << program;
|
|
|
|
proc->setOutputChannelMode(KProcess::MergedChannels);
|
|
|
|
proc->setProgram( program );
|
|
|
|
setTitle(cfg->name());
|
|
}
|
|
|
|
|
|
void ScriptAppJob::start()
|
|
{
|
|
kDebug() << "launching?" << proc;
|
|
if( proc )
|
|
{
|
|
startOutput();
|
|
appendLine( i18n("Starting: %1", proc->program().join(" ") ) );
|
|
proc->start();
|
|
} else
|
|
{
|
|
kWarning() << "No process, something went wrong when creating the job";
|
|
// No process means we've returned early on from the constructor, some bad error happened
|
|
emitResult();
|
|
}
|
|
}
|
|
|
|
bool ScriptAppJob::doKill()
|
|
{
|
|
if( proc ) {
|
|
proc->kill();
|
|
appendLine( i18n( "*** Killed Application ***" ) );
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
void ScriptAppJob::processFinished( int exitCode , QProcess::ExitStatus status )
|
|
{
|
|
lineMaker->flushBuffers();
|
|
|
|
if (exitCode == 0 && status == QProcess::NormalExit) {
|
|
appendLine( i18n("*** Exited normally ***") );
|
|
} else if (status == QProcess::NormalExit) {
|
|
appendLine( i18n("*** Exited with return code: %1 ***", QString::number(exitCode)) );
|
|
setError(OutputJob::FailedShownError);
|
|
} else if (error() == KJob::KilledJobError) {
|
|
appendLine( i18n("*** Process aborted ***") );
|
|
setError(KJob::KilledJobError);
|
|
} else {
|
|
appendLine( i18n("*** Crashed with return code: %1 ***", QString::number(exitCode)) );
|
|
setError(OutputJob::FailedShownError);
|
|
}
|
|
kDebug() << "Process done";
|
|
emitResult();
|
|
}
|
|
|
|
void ScriptAppJob::processError( QProcess::ProcessError error )
|
|
{
|
|
kDebug() << proc->readAllStandardError();
|
|
kDebug() << proc->readAllStandardOutput();
|
|
kDebug() << proc->errorString();
|
|
if( error == QProcess::FailedToStart )
|
|
{
|
|
setError( FailedShownError );
|
|
QString errmsg = i18n("*** Could not start program '%1'. Make sure that the "
|
|
"path is specified correctly ***", proc->program().join(" ") );
|
|
appendLine( errmsg );
|
|
setErrorText( errmsg );
|
|
emitResult();
|
|
}
|
|
kDebug() << "Process error";
|
|
}
|
|
|
|
void ScriptAppJob::appendLine(const QString& l)
|
|
{
|
|
if (KDevelop::OutputModel* m = model()) {
|
|
m->appendLine(l);
|
|
}
|
|
}
|
|
|
|
KDevelop::OutputModel* ScriptAppJob::model()
|
|
{
|
|
return dynamic_cast<KDevelop::OutputModel*>( OutputJob::model() );
|
|
}
|
|
|
|
|
|
#include "moc_scriptappjob.cpp"
|