Skip to content

Commit

Permalink
Open and load --code file in C++
Browse files Browse the repository at this point in the history
  • Loading branch information
btzy committed Jan 13, 2025
1 parent b06ccc1 commit 0120199
Show file tree
Hide file tree
Showing 7 changed files with 126 additions and 17 deletions.
20 changes: 3 additions & 17 deletions src/app/main.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1630,27 +1630,13 @@ int main( int argc, char *argv[] )
}
}

if ( !pythonArgs.isEmpty() )
if ( !pythonfile.isEmpty() )
{
if ( !pythonfile.isEmpty() )
if ( !pythonArgs.isEmpty() )
{
#ifdef Q_OS_WIN
//replace backslashes with forward slashes
pythonfile.replace( '\\', '/' );
#endif
pythonArgs.prepend( pythonfile );
}

QgsPythonRunner::run( QStringLiteral( "sys.argv = ['%1']" ).arg( pythonArgs.replaceInStrings( QChar( '\'' ), QStringLiteral( "\\'" ) ).join( "','" ) ) );
}

if ( !pythonfile.isEmpty() )
{
#ifdef Q_OS_WIN
//replace backslashes with forward slashes
pythonfile.replace( '\\', '/' );
#endif
QgsPythonRunner::run( QStringLiteral( "with open('%1','r') as f: exec(f.read())" ).arg( pythonfile ) );
QgsPythonRunner::runFile( pythonfile, pythonArgs );
}

/////////////////////////////////`////////////////////////////////////
Expand Down
15 changes: 15 additions & 0 deletions src/app/qgisapp.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -12101,6 +12101,21 @@ class QgsPythonRunnerImpl : public QgsPythonRunner
return false;
}

bool runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError = QString() ) override
{
#ifdef WITH_BINDINGS
if ( mPythonUtils && mPythonUtils->isEnabled() )
{
return mPythonUtils->runFile( filename, arguments, messageOnError );
}
#else
Q_UNUSED( filename )
Q_UNUSED( arguments )
Q_UNUSED( messageOnError )
#endif
return false;
}

bool evalCommand( QString command, QString &result ) override
{
#ifdef WITH_BINDINGS
Expand Down
14 changes: 14 additions & 0 deletions src/core/qgspythonrunner.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,20 @@ bool QgsPythonRunner::run( const QString &command, const QString &messageOnError
}
}

bool QgsPythonRunner::runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError )
{
if ( sInstance )
{
QgsDebugMsgLevel( "Running " + filename, 3 );
return sInstance->runFile( filename, arguments, messageOnError );
}
else
{
QgsDebugError( QStringLiteral( "Unable to run Python command: runner not available!" ) );
return false;
}
}

bool QgsPythonRunner::eval( const QString &command, QString &result )
{
if ( sInstance )
Expand Down
5 changes: 5 additions & 0 deletions src/core/qgspythonrunner.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,9 @@ class CORE_EXPORT QgsPythonRunner
//! Execute a Python statement
static bool run( const QString &command, const QString &messageOnError = QString() );

//! Execute a Python file, with the given arguments
static bool runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError = QString() );

//! Eval a Python statement
static bool eval( const QString &command, QString &result SIP_OUT );

Expand All @@ -59,6 +62,8 @@ class CORE_EXPORT QgsPythonRunner

virtual bool runCommand( QString command, QString messageOnError = QString() ) = 0;

virtual bool runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError = QString() ) = 0;

virtual bool evalCommand( QString command, QString &result ) = 0;

static QgsPythonRunner *sInstance;
Expand Down
6 changes: 6 additions & 0 deletions src/python/qgspythonutils.h
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,12 @@ class PYTHON_EXPORT QgsPythonUtils
*/
virtual QString runStringUnsafe( const QString &command, bool single = true ) = 0;

/**
* Runs a Python \a filename, showing an error message if one occurred.
* \returns TRUE if no error occurred
*/
virtual bool runFile( const QString &filename, const QStringList &arguments, QString msgOnError = QString(), bool single = true ) = 0;

/**
* Evaluates a Python \a command and stores the result in a the \a result string.
*/
Expand Down
81 changes: 81 additions & 0 deletions src/python/qgspythonutilsimpl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -454,6 +454,87 @@ bool QgsPythonUtilsImpl::runString( const QString &command, QString msgOnError,
return res;
}

QString QgsPythonUtilsImpl::runFileUnsafe( const QString &filename, const QStringList &arguments )
{
// acquire global interpreter lock to ensure we are in a consistent state
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
QString ret;

QFile file( filename );
if ( !file.open( QIODevice::ReadOnly | QIODevice::Text ) )
return "Cannot open file";

if ( !arguments.isEmpty() )
{
PyObject *sysobj = nullptr, *argsobj = nullptr;
bool success = false;
sysobj = PyRun_String( "sys", Py_single_input, mMainDict, mMainDict );
if ( !sysobj )
goto error;
PyObject *argsobj = PyList_New( arguments.size() );
if ( !argsobj )
goto error;
for ( size_t i = 0; i != arguments.size(); ++i )
{
if ( PyList_SET_ITEM( argsobj, i, PyUnicode_FromString( arguments[i].toUtf8().constData() ) ) != 0 )
goto error;
}
if ( PyObject_SetAttrString( sysobj, "argv", argsobj ) != 0 )
goto error;
success = true;
error:
Py_XDECREF( argsobj );
Py_XDECREF( sysobj );
if ( !success )
return "Cannot set sys.argv";
}

PyObject *obj = PyRun_String( file.readAll().constData(), Py_file_input, mMainDict, mMainDict );
PyObject *errobj = PyErr_Occurred();
if ( nullptr != errobj )
{
ret = getTraceback();
}
Py_XDECREF( obj );

// we are done calling python API, release global interpreter lock
PyGILState_Release( gstate );

return ret;
}

bool QgsPythonUtilsImpl::runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError = QString() )
{
const QString traceback = runFileUnsafe( filename, arguments );
if ( traceback.isEmpty() )
return true;

if ( msgOnError.isEmpty() )
{
// use some default message if custom hasn't been specified
msgOnError = QObject::tr( "An error occurred during execution of following file:" ) + "\n<tt>" + filename + "</tt>";
}

QString path, version;
evalString( QStringLiteral( "str(sys.path)" ), path );
evalString( QStringLiteral( "sys.version" ), version );

QString str = "<font color=\"red\">" + msgOnError + "</font><br><pre>\n" + traceback + "\n</pre>"
+ QObject::tr( "Python version:" ) + "<br>" + version + "<br><br>"
+ QObject::tr( "QGIS version:" ) + "<br>" + QStringLiteral( "%1 '%2', %3" ).arg( Qgis::version(), Qgis::releaseName(), Qgis::devVersion() ) + "<br><br>"
+ QObject::tr( "Python path:" ) + "<br>" + path;
str.replace( '\n', QLatin1String( "<br>" ) ).replace( QLatin1String( " " ), QLatin1String( "&nbsp; " ) );

qDebug() << str;
QgsMessageOutput *msg = QgsMessageOutput::createMessageOutput();
msg->setTitle( QObject::tr( "Python error" ) );
msg->setMessage( str, QgsMessageOutput::MessageHtml );
msg->showMessage();

return false;
}


QString QgsPythonUtilsImpl::getTraceback()
{
Expand Down
2 changes: 2 additions & 0 deletions src/python/qgspythonutilsimpl.h
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,8 @@ class QgsPythonUtilsImpl : public QgsPythonUtils
bool isEnabled() final;
bool runString( const QString &command, QString msgOnError = QString(), bool single = true ) final;
QString runStringUnsafe( const QString &command, bool single = true ) final; // returns error traceback on failure, empty QString on success
bool runFile( const QString &filename, const QStringList &arguments, const QString &messageOnError = QString() ) final;
QString runFileUnsafe( const QString &filename, const QStringList &arguments ) final; // returns error traceback on failure, empty QString on success
bool evalString( const QString &command, QString &result ) final;
bool getError( QString &errorClassName, QString &errorText ) final;

Expand Down

0 comments on commit 0120199

Please sign in to comment.