We have a Python API for jobmodules!

Created a Boost.Python module interface in libcalamares.
Added a PythonJob wrapper and exposed it in the Python module.
Rename target calamareslib ==> calamares so in Python it's libcalamares.
Python-related classes in libcalamares that aren't exported as C++
symbols are now in a CalamaresPrivate namespace.
Import the libcalamares python module into every Python script before
running it.
Added Python error handling to PythonJobHelper.
Added some more testing code to dummypython module.
This commit is contained in:
Teo Mrnjavac 2014-07-17 17:52:02 +02:00
parent c13179fdf0
commit a61a36d99d
11 changed files with 217 additions and 25 deletions

View File

@ -53,7 +53,7 @@ endif()
###
### Calamares application info
###
set( CALAMARES_ORGANIZATION_NAME "The Calamares Team" )
set( CALAMARES_ORGANIZATION_NAME "Calamares" )
set( CALAMARES_ORGANIZATION_DOMAIN "github.com/calamares" )
set( CALAMARES_APPLICATION_NAME "Calamares" )
set( CALAMARES_DESCRIPTION_SUMMARY "The distribution-independent installer framework" )
@ -108,7 +108,7 @@ file( COPY CalamaresAddLibrary.cmake DESTINATION "${PROJECT_BINARY_DIR}" )
file( COPY CalamaresAddModuleSubdirectory.cmake DESTINATION "${PROJECT_BINARY_DIR}" )
file( COPY CalamaresAddPlugin.cmake DESTINATION "${PROJECT_BINARY_DIR}" )
set( CALAMARES_LIBRARIES calamareslib )
set( CALAMARES_LIBRARIES calamares )
add_subdirectory( src )
add_subdirectory( tests )
@ -118,8 +118,8 @@ macro_display_feature_log()
# Add all targets to the build-tree export set
set( CMAKE_INSTALL_CMAKEDIR "${CMAKE_INSTALL_LIBDIR}/cmake/Calamares" CACHE PATH "Installation directory for CMake files" )
set( CMAKE_INSTALL_FULL_CMAKEDIR "${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_CMAKEDIR}" )
export(TARGETS calamareslib
FILE "${PROJECT_BINARY_DIR}/CalamaresLibraryDepends.cmake")
export( TARGETS calamares
FILE "${PROJECT_BINARY_DIR}/CalamaresLibraryDepends.cmake" )
# Export the package for use from the build-tree
# (this registers the build-tree with a global CMake-registry)

View File

@ -17,5 +17,5 @@ endif()
include("${CALAMARES_CMAKE_DIR}/CalamaresLibraryDepends.cmake")
# These are IMPORTED targets created by CalamaresLibraryDepends.cmake
set(CALAMARES_LIBRARIES calamareslib)
set(CALAMARES_LIBRARIES calamares)
set(CALAMARES_USE_FILE "${CALAMARES_CMAKE_DIR}/CalamaresUse.cmake")

View File

@ -1,4 +1,4 @@
project( calamareslib )
project( libcalamares )
add_definitions( ${QT_DEFINITIONS} )
add_definitions( -DQT_SHARED )
@ -36,6 +36,7 @@ if( WITH_PYTHON )
set( libSources
${libSources}
PythonJob.cpp
PythonJobApi.cpp
PythonJobHelper.cpp
)
@ -53,19 +54,18 @@ if( WITH_PYTHON )
endif()
add_library( calamareslib SHARED ${libSources} )
set_target_properties( calamareslib
add_library( calamares SHARED ${libSources} )
set_target_properties( calamares
PROPERTIES
AUTOMOC TRUE
VERSION ${CALAMARES_VERSION_SHORT}
SOVERSION ${CALAMARES_VERSION_SHORT}
OUTPUT_NAME "calamares"
)
qt5_use_modules( calamareslib Core )
qt5_use_modules( calamares Core )
target_link_libraries( calamareslib
target_link_libraries( calamares
LINK_PRIVATE
# internal deps, if any
${OPTIONAL_PRIVATE_LIBRARIES}
@ -75,7 +75,7 @@ target_link_libraries( calamareslib
Qt5::Core
)
install( TARGETS calamareslib
install( TARGETS calamares
EXPORT CalamaresLibraryDepends
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}

View File

@ -38,7 +38,7 @@ public:
, m_queue( queue )
{
#ifdef WITH_PYTHON
new PythonJobHelper( this );
new CalamaresPrivate::PythonJobHelper( this );
#endif
}

View File

@ -26,8 +26,26 @@
#undef slots
#include <boost/python.hpp>
#include "PythonJobApi.h"
namespace bp = boost::python;
BOOST_PYTHON_MODULE( libcalamares )
{
bp::scope().attr( "ORGANIZATION_NAME" ) = CALAMARES_ORGANIZATION_NAME;
bp::scope().attr( "ORGANIZATION_DOMAIN" ) = CALAMARES_ORGANIZATION_DOMAIN;
bp::scope().attr( "APPLICATION_NAME" ) = CALAMARES_APPLICATION_NAME;
bp::scope().attr( "VERSION" ) = CALAMARES_VERSION;
bp::scope().attr( "VERSION_SHORT" ) = CALAMARES_VERSION_SHORT;
bp::class_< CalamaresPrivate::PythonJobInterface >( "job", bp::init< const Calamares::PythonJob* >() )
.def( "prettyName", &CalamaresPrivate::PythonJobInterface::prettyName )
.def( "workingPath", &CalamaresPrivate::PythonJobInterface::workingPath );
}
namespace Calamares {
@ -84,6 +102,11 @@ PythonJob::exec()
{
bp::object scriptNamespace = helper()->createCleanNamespace();
bp::object calamaresModule = bp::import( "libcalamares" );
bp::dict calamaresNamespace = bp::extract< bp::dict >( calamaresModule.attr( "__dict__" ) );
calamaresNamespace[ "job" ] = CalamaresPrivate::PythonJobInterface( this );
bp::object result = bp::exec_file( scriptFI.absoluteFilePath().toLocal8Bit().data(),
scriptNamespace,
scriptNamespace );
@ -94,19 +117,28 @@ PythonJob::exec()
cDebug() << "Python job" << prettyName() << "finished with message" << message;
}
catch ( bp::error_already_set e )
catch ( bp::error_already_set )
{
return JobResult::error( tr( "Boost.Python error" ) );
QString msg;
if ( PyErr_Occurred() )
{
msg = helper()->handleLastError();
}
bp::handle_exception();
PyErr_Clear();
return JobResult::error( tr( "Boost.Python error" ),
msg );
}
return JobResult::ok();
}
PythonJobHelper*
CalamaresPrivate::PythonJobHelper*
PythonJob::helper()
{
return PythonJobHelper::s_instance;
return CalamaresPrivate::PythonJobHelper::s_instance;
}

View File

@ -21,9 +21,13 @@
#include "Job.h"
namespace Calamares {
namespace CalamaresPrivate
{
class PythonJobInterface;
class PythonJobHelper;
}
namespace Calamares {
class PythonJob : public Job
{
@ -38,8 +42,9 @@ public:
JobResult exec() override;
private:
friend class PythonJobHelper;
PythonJobHelper* helper();
friend class CalamaresPrivate::PythonJobHelper;
friend class CalamaresPrivate::PythonJobInterface;
CalamaresPrivate::PythonJobHelper* helper();
QString m_scriptFile;
QString m_workingPath;
};

View File

@ -0,0 +1,42 @@
/* === This file is part of Calamares - <http://github.com/calamares> ===
*
* Copyright 2014, Teo Mrnjavac <teo@kde.org>
*
* Calamares 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 3 of the License, or
* (at your option) any later version.
*
* Calamares 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 Calamares. If not, see <http://www.gnu.org/licenses/>.
*/
#include "PythonJobApi.h"
namespace CalamaresPrivate
{
PythonJobInterface::PythonJobInterface( const Calamares::PythonJob* parent )
: m_parent( parent )
{}
std::string
PythonJobInterface::prettyName() const
{
return m_parent->prettyName().toStdString();
}
std::string
PythonJobInterface::workingPath() const
{
return m_parent->m_workingPath.toStdString();
}
}

View File

@ -0,0 +1,44 @@
/* === This file is part of Calamares - <http://github.com/calamares> ===
*
* Copyright 2014, Teo Mrnjavac <teo@kde.org>
*
* Calamares 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 3 of the License, or
* (at your option) any later version.
*
* Calamares 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 Calamares. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef PYTHONJOBAPI_H
#define PYTHONJOBAPI_H
#include "CalamaresVersion.h"
#include "PythonJob.h"
namespace CalamaresPrivate
{
class PythonJobInterface
{
public:
explicit PythonJobInterface( const Calamares::PythonJob* parent );
std::string prettyName() const;
std::string workingPath() const;
private:
const Calamares::PythonJob* m_parent;
};
}
#endif // PYTHONJOBAPI_H

View File

@ -18,13 +18,18 @@
#include "PythonJobHelper.h"
#include "utils/CalamaresUtils.h"
#include "utils/Logger.h"
#include <QDir>
#include <QFileInfo>
#undef slots
#include <boost/python.hpp>
namespace bp = boost::python;
namespace Calamares {
namespace CalamaresPrivate {
PythonJobHelper* PythonJobHelper::s_instance = nullptr;
@ -38,6 +43,34 @@ PythonJobHelper::PythonJobHelper( QObject* parent )
m_mainModule = bp::import( "__main__" );
m_mainNamespace = m_mainModule.attr( "__dict__" );
// If we're running from the build dir
QFileInfo fi( QDir::current().absoluteFilePath( "libcalamares.so" ) );
if ( fi.exists() && fi.isReadable() )
{
m_pythonPaths.append( fi.dir().absolutePath() );
}
QDir calaPythonPath( CalamaresUtils::systemLibDir().absolutePath() +
QDir::separator() + "calamares" );
if ( calaPythonPath.exists() &&
calaPythonPath.isReadable() )
{
QFileInfo fi( calaPythonPath.absoluteFilePath( "libcalamares.so" ) );
if ( fi.exists() && fi.isReadable() )
{
m_pythonPaths.append( fi.dir().absolutePath() );
}
}
bp::object sys = bp::import( "sys" );
foreach ( QString path, m_pythonPaths )
{
bp::str dir = path.toLocal8Bit().data();
sys.attr( "path" ).attr( "append" )( dir );
}
}
else
{
@ -64,4 +97,27 @@ PythonJobHelper::createCleanNamespace()
}
QString
PythonJobHelper::handleLastError()
{
using namespace boost::python;
using namespace boost;
PyObject *exc,*val,*tb;
object formatted_list, formatted;
PyErr_Fetch(&exc,&val,&tb);
handle<> hexc(exc),hval(allow_null(val)),htb(allow_null(tb));
object traceback(import("traceback"));
if (!tb) {
object format_exception_only(traceback.attr("format_exception_only"));
formatted_list = format_exception_only(hexc,hval);
} else {
object format_exception(traceback.attr("format_exception"));
formatted_list = format_exception(hexc,hval,htb);
}
formatted = str("\n").join(formatted_list);
return QString::fromStdString( extract<std::string>(formatted) );
}
} // namespace Calamares

View File

@ -21,10 +21,12 @@
#include "PythonJob.h"
#include <QStringList>
#undef slots
#include <boost/python/object.hpp>
namespace Calamares {
namespace CalamaresPrivate {
class PythonJobHelper : public QObject
{
@ -35,12 +37,16 @@ public:
boost::python::object createCleanNamespace();
QString handleLastError();
private:
friend PythonJobHelper* PythonJob::helper();
friend PythonJobHelper* Calamares::PythonJob::helper();
static PythonJobHelper* s_instance;
boost::python::object m_mainModule;
boost::python::object m_mainNamespace;
QStringList m_pythonPaths;
};
} // namespace Calamares

View File

@ -16,9 +16,16 @@
# You should have received a copy of the GNU General Public License
# along with Calamares. If not, see <http://www.gnu.org/licenses/>.
import sys
import libcalamares
import os
from time import gmtime, strftime
def calamares_main():
os.system( "/bin/sh -c \"touch ~/calamares-dummypython\"" )
return strftime( "%Y-%m-%d %H:%M:%S", gmtime() )
accumulator = strftime( "%Y-%m-%d %H:%M:%S", gmtime() ) + "\n"
accumulator += "Calamares version: " + libcalamares.VERSION_SHORT + "\n"
accumulator += "This job's name: " + libcalamares.job.prettyName() + "\n"
accumulator += "This job's path: " + libcalamares.job.workingPath() + "\n"
return accumulator