A simple, extensible build system for use with Scons. Cuppa is designed to leverage the capabilities of Scons, while allowing developers to focus on the task of describing what needs to be built. In general cuppa supports make
like usage on the command-line. That is developers can simply write:
scons -D
and have Scons "do the right thing"; building targets for any sconscript
files found in the current directory.
Cuppa can be installed as a normal python package or installed locally into a site_scons
directory allowing it to be effortlessly integrated into any Scons setup.
Note:
-D
tellsscons
to look for ansconstruct
file in the current or parent directories and if it finds one execute thesconscript
files as if called from the starting directory. This ensures everything works as expected. For more details refer to the Scons documentation
- Quick Intro
- Design Principles
- Installation and Dependencies
- Reference
- Supported Dependencies
- Creating your own Dependencies
- Profiles
- Acknowledgements
The simpest way to get cuppa is to pip install
it using:
pip install cuppa
however there are a few approaches that can be used as described in the Installation and Dependencies section.
Let's look at a minimal sconstruct
that makes use of cuppa. It could look like this:
# Pull in all the Cuppa goodies..
import cuppa
# Call sconscripts to do the work
cuppa.run()
Calling the run
method in the cuppa
module starts the build process calling sconscript
files.
Here is an example sconscript
file that builds all *.cpp files in the directory where it resides:
Import( 'env' )
# Build all *.cpp source files as executables
for Source in env.GlobFiles('*.cpp'):
env.Build( str(Source)[:-4], Source )
The env.Build()
method is provided by cuppa and does essentially what env.Program()
does but in addition is both toolchain and variant aware, and further can provide notifications on progress.
Note: Source[:-4] simply strips off the file extension
.cpp
, that is, the last 4 characters of the file name.
If our sconscript
file was for a directory containing *.cpp files that are actually tests then we could instead write the sconscript
file as:
Import( 'env' )
# Build all *.cpp source files as executables to be run as tests
for Source in env.GlobFiles('*.cpp'):
env.BuildTest( str(Source)[:-4], Source )
The env.BuildTest()
method is provided by cuppa and builds the sources specified as env.Build()
does.
However, in addition, passing --test
on the command-line will also result in the executable produced being run by a runner. The default test runner simply treats each executable as a test case and each directory of executables as a test suite. If the process executes cleanly the test passed, if not it failed.
To run this on the command-line we would write:
scons -D --test
If we only want to build and test debug executables we can instead write this:
scons -D --dbg --test
Or for release only pass --rel
.
cuppa also makes it easy to work with dependencies. For example, if boost was a default dependency for all your sconscript
files you could write your sconstruct file as follows:
import cuppa
cuppa.run(
default_options = {
'boost-location': '<Location of Boost>'
},
default_dependencies = [
'boost'
]
)
This will automatically ensure that necessary includes and other compile options are set for the boost version that is found at boost-location
.
Note: cuppa will even attempt to retrieve the dependency if a version or URL is supplied. Including from source control assuming you have the necessary source control system installed on your machine.
If you need to link against specific boost libraries this can also be done in the sconscript file as follows:
Import('env')
Test = 'my_complex_test'
Sources = [
Test + '.cpp'
]
env.AppendUnique( STATICLIBS =
env.BoostStaticLibs( [
'system',
'log',
'thread',
'timer',
'chrono',
'filesystem'
] )
)
env.BuildTest( Test, Sources )
The BoostStaticLibs()
method ensures that the library is built in the correct build variant as required. If you preferred to use dynamic linking then that can also be achieved using BoostSharedLibs()
.
The point is the complexities of using boost as a dependency are encapsulated and managed separately from the scontruct and sconscript files allowing developers to focus on intent, not method.
cuppa has been written primarily to provide a clean and structured way to leverage the power of Scons without the usual problems of hugely complex scontruct
files that diverge between projects. Key goals of cuppa are:
- minimise the need for adding logic into
sconscript
files, keeping them as declarative as possible. - allow declarative
sconscript
s that are both much clearer and significantly simpler than the equivalentmake
file, without the need to learn a whole new scripting language likemake
orcmake
. - provide a clear structure for extending the facilities offered by cuppa
- provide a clear vocabulary for building projects (dependencies, methods, profiles, runners, variants etc.)
- codify Scons best practices into cuppa itself so that users just need to call appropriate methods knowing that cuppa will do the right thing with their intent
- provide a framework that allows experts to focus on providing facilities for others to use. Write once, use everywhere. For example one person who knows how best to make boost available as a dependency can manage and maintain that dependency and allow others to use it seamlessly.
cuppa can be made available as a normal python package and this is the preferred method of installation. It can be added directly to a site_scons
folder, placed appropriately so Scons will find it. For global use add it to your home directory or for use with a specific project place it beside (or sym-link site_scons
beside) the top-level sconstruct
file. For more details on using a site_scons
folder refer to the Scons man page.
The following sections summarise some of the ways you can get cuppa.
Use pip install
to get the latest:
pip install cuppa
Install locally in the same folder as your sconstruct
file using:
pip install cuppa -t .
Adding this to your sconstruct
file would pip install
cuppa if it was not found:
try:
import cuppa
except ImportError:
print "Cuppa not found, installing..."
import subprocess, shlex
subprocess.call( shlex.split( 'pip install cuppa -t .' ) )
import cuppa
To make use of the colourisation cuppa uses the colorama package.
cuppa uses the gcovr python library to help post-process the coverage files that gcov
produces (used by both the GCC and CLANG toolchains). This produces a nice coverage.html
file in your final build folder that links to HTML files for all files for which coverage information was produced.
cuppa uses the following terms to mean specific aspects of a build. Understanding these will remove ambiguity in understanding the facilities that cuppa provides.
Term | Meaning |
---|---|
Methods | cuppa provides a number of build methods that can be called inside your sconscript files. Methods such as Build() , BuildTest() , BuildWith() , Compile() and so on. These are in addition to the methods already provided by Scons. A key difference between these methods and the basic Scons methods such as Program() and Object() is that these methods are aware of build variants, toolchains, progress notifications and so on. |
Dependencies | Projects can have an arbitrary number of dependencies, for example boost or Qt. Typically a dependency requires compiler and linker flags to be updated, or in some cases pulled from a remote repository and built. Wrapping that complexity up into a dependency makes it easy for developers to re-use that effort cleanly across projects. |
Profiles | Profiles are a collection of build modifications that are commonly made to builds and might bring in one or more dependencies. Profiles provide an easy way to say, "I always do this with my builds" and make that available to others if you want to. |
Variants and Actions | Variants and Actions allow the specification of a specific builds, such as debug and release builds. Actions allow additional actions to be taken for a build, such as executing tests and analysing code coverage. |
Target Architecture | The Target Architecture is the platform target of any build output. If none is specified a default for the current platform and toolchain is used. By tracking this information it allows multiple builds to be completed for different target architectures, such as x86 and amd64 on Windows. |
Toolchains | Toolchains allow custom build settings for different toolchains making it easy to build for any available toolchain on a specific platform, or even different versions of the same underlying toolchain. For example you might want to build and test against several versions of GCC and Clang at the same time. This allows you to do so. |
--show-conf Show the current values in the configuration
file if one exists
--save-conf Save the current command-line a configuration
file
--update-conf Update the configuration file with the current
command-line
--remove-settings=REMOVE_SETTINGS
Remove the listed settings from the
configuration file
--clear-conf Clear the configuration file
--raw-output Disable output processing like colourisation of
output
--standard-output Perform standard output processing but not
colourisation of output
--minimal-output Show only errors and warnings in the output
--ignore-duplicates Do not show repeated errors or warnings
--projects=PROJECTS Projects to build (alias for scripts)
--scripts=SCRIPTS Sconscripts to run
--thirdparty=DIR Thirdparty directory
--build-root=BUILD_ROOT The root directory for build output. If not
specified then .build is used
--download-root=DOWNLOAD_ROOT
The root directory for downloading external
libraries to. If not specified then .cuppa is
used
--runner=RUNNER The test runner to use for executing tests. The
default is the process test runner
--dump Dump the default environment and exit
--parallel Enable parallel builds utilising the available
concurrency. Translates to -j N with N chosen
based on the current hardware
--show-test-output When executing tests display all outout to
stdout and stderr as appropriate
--verbosity=VERBOSITY The The verbosity level that you wish to run
cuppa at. The default level is "info".
VERBOSITY may be one of ('trace', 'debug',
'info', 'warn', 'error')
--decider=DECIDER The decider to use for determining if a
dependency has changed. Refer to the Scons
manual for more details. By default
"MD5-timestamp" is used. DECIDER may be one of
('timestamp-newer', 'timestamp-match', 'MD5',
'MD5-timestamp')
--stdcpp=STDCPP Use this option to override the default language
compliance of your cpp compiler which by
dafault is the highest compliance available.
Value may be one of ('c++98', 'c++03',
'c++0x', 'c++11', 'c++1y', 'c++14')
--cov Build an instrumented binary
--dbg Build a debug binary
--rel Build a release (optimised) binary
--test Run the binary as a test
--toolchains=TOOLCHAINS The Toolchains you wish to build against
cuppa places all builds outside of the source tree under the BUILD_ROOT
which by default is the folder _build
beside the sconstruct
file used when Scons is executed. You can change this by specifying the --build-root
option.
Build variants and the output from each sconscript
is kept separate using the following convention:
<build_root>/<sconscript_path>/<sconscript_name>/<toolchain>/<build_variant>/<target_arch>/final
If <sconscript_name>
is "sconscript" then it is omitted from the path. The assumption is that a single sconscript
file is being used for the given folder and therefore the folder name is sufficient to differentiate from other sconscript
s.
cuppa allows you to save commonly used or local settings to a conf file so that they can be re-applied when you execute scons
from anywhere in your Sconscript tree. The basic approach is to pass --save-conf
along with the options you wish to save.
No builds will be performed at this time and your effective command-line will be dispalyed for you to review. Next time you simply execute scons
(or more typically scons -D
) your previous options will be applied automatically. For example, passing --save-conf
alongside --boost-home=~/boost/boost_1_55
would result in --boost-home
being applied on subsequent builds.
In addition to --save-conf
there are a few other options that allow, updating, removing, clearing and reviewing of options. All the options are summarised below:
Option | Description |
---|---|
--save-conf |
Saves all the current command-line settings to the conf file - overwriting any that existed previously. |
--update-conf |
Passing --update-conf which will save any new settings passed and overwrite the values of existing settings. Other existing settings will remain unchanged. |
--clear-conf |
Removes the conf file - clearing all settings. |
--remove-settings |
--remove-settings takes a comma-separate list of settings to be removed from the currrent conf file. Non-existant settings are ignored. |
--show-conf |
Echoes the equivalent command-line that would result from the application of the existing conf file. |
Calling the run()
method of the cuppa
module in your sconstruct file is used to start the build process. run()
is defined as follows:
run( base_path = os.path.abspath( '.' ),
branch_root = None,
default_options = {},
default_projects = [],
default_variants = [],
default_dependencies = [],
default_profiles = [],
dependencies = [],
profiles = [],
default_runner = None,
configure_callback = None,
tools = [] )
Overview: Starts the build process.
Effects: Executes the build using the defaults supplied. That is each sconscript
file will be executed using the defaults specified in the sconstruct
file. Passing options on the command-line will override any defaults specified here. If no --scripts
or --projects
are specified on the command-line cuppa attempts to find and run sconscript
s present in the lauch directory from where Scons was executed.
Argument | Usage |
---|---|
base_path |
You may override this to force a different base path other than the path where the sconstruct is located. You might want to do this to compensate for situations were your sconstruct file resides beside your project files in the filesystem. |
branch_root |
In some project structures, for example those using subversion style folder branches, you may want to specify where your branch root is. The purpose of this would be to allow specification of full branch names when referencing other source code in your sconstruct files. For example, if you had project code that relied on a specific branch of shared code (using folder based branches as in subversion) you could refer to the branch explicitly in your sconstrcut file as an offset to the branch_root . |
default_options |
default_options expects a dictionary of options. The allowable options are the same as the command-line options with the leading -- . For example changing the default build_root from .build to /tmp/builds could be achieved by writing default_options = { 'build-root': '/tmp/builds' } |
default_variants |
default_variants takes a list of variants, for example [ 'dbg', 'rel', 'cov' ] . By default the dbg and rel variants are built. If you only wanted to build release variants you might set default_variants = ['rel'] for example. |
default_dependencies |
default_dependencies takes a list of dependencies you want to always apply to the build environment and ensures that they are already applied for each build. You may pass the name of a supported dependency, such as 'boost' or a callable object taking ( env, toolchain, variant ) as parameters. |
default_profiles |
default_profiles takes a list of profiles you want to always apply to the build environment and ensures that they are already applied for each build. You may pass the name of a supported profile or a callable object taking ( env, toolchain, variant ) as parameters. |
default_runner |
By default the runner used is 'process' however you my specify your own test runner or use one of the other runners provided, such as 'boost' . |
configure_callback |
This allows you to specify a callback to be executed during part of a configure process. This callback should be any callable object that takes the following parameter ( configure_context ) . Refer to the Scons Multi-Platform Configuration documentation for details on how to make use of the configure_context . |
dependencies |
dependencies takes a dictionary of dependency classes as "name":class_name pairs. For example you might |
tools |
tools takes a list of Scons Tools that should be added by default to all environments |
env.Build(
target,
source,
final_dir = None,
append_variant = False )
Overview: env.Build()
performs the same task as env.Program()
but with the additional benefit of reporting progress and the ability to specify where the target is placed and named.
Effects: Builds the target from the sources specified writing the output as target_name
where target_name
is:
target_name = os.path.join(
final_dir,
target,
( ( append_variant and env['variant'] != 'rel' ) and '_' + env['variant'] or '' )
)
If final_dir
is not specified then it is ../final
, relative to the working directory
In addition to adding dynamic libraries to the environment using:
env.AppendUnique( DYNAMICLIBS = env['LIBS'] )
env.Build()
essentially performs as:
env.Program(
target_name,
source,
LIBS = env['DYNAMICLIBS'] + env['STATICLIBS'],
CPPPATH = env['SYSINCPATH'] + env['INCPATH']
)
It can do this because the build variants and toolchains have taken care to ensure that env is configured with the correct values in the variables referenced.
env.Test(
source,
final_dir = None,
data = None,
runner = None,
expected = 'passed' )
Overview: Uses the specified runner
to execute source
as a test using any data
provided. The runner
can report on the progress of the test with respect to the expected
outcome.
Example:
Import('env')
test_program = env.Build( 'my_program', [ 'main.cpp', 'utility.cpp' ] )
env.Test( test_program )
Effects: The runner
will be used to execute the source
program with any supplied data
treated as a source dependency for the target test outcome from running the test. Any output from the test will be placed in final_dir
.
env.BuildTest(
target,
source,
final_dir = None,
data = None,
append_variant = None,
runner = None,
expected = 'passed' )
Overview: Builds the target from the specified sources and allows it to be executed as a test.
Effects: As if:
program = env.Build( target, sources )
env.Test( program )
env.Compile( sources )
Overview: Compile the specified sources
into object files.
Effects: Returns objects
nodes that represent the outcome of compiling the specified sources
. As if:
objects = env.Object( sources, CPPPATH = env['SYSINCPATH'] + env['INCPATH'] )
Typically env.Compile()
is not needed and instead you should directly use env.Build()
to directly produce the required program or library being built. However in some cases, such as when using env.CreateVersion()
you need to break dependency cycles and then env.Compile()
is needed.
env.BuildWith( dependencies )
Overview: Updates the current env
with any modifications required to allow the build to work with the given dependencies
.
Effects: env
will be updated with all variable and method updates provided by each dependency
in dependencies
.
env.BuildProfile( profiles )
Overview: Updates the current env
with any modifications specified in the profiles
listed.
Effects: env
will be updated as dictated by each profile
in profiles
.
env.Use( dependency )
Overview: Updates the current env
with specified dependency
.
Effects: env
will be updated as per the dependency
.
env.CreateVersion( version_file, sources, namespaces, version, location )
Overview: Creates a version cpp and hpp file that can be included in other files in your project while ensuring dependencies between files are correctly handled and that the cpp file is built correctly.
Effects: Creates a version_file
and a header file for the target file that depends on sources
. The version file will have a class identity
nested inside namespaces
with an interface as follows:
#ifndef INCLUDED__FIRST__SECOND__N_BUILD_GENERATED_VERSION_HPP
#define INCLUDED__FIRST__SECOND__N_BUILD_GENERATED_VERSION_HPP
namespace _first {
namespace _second {
namespace _n {
class identity
{
public:
typedef std::string string_t;
typedef std::vector< string_t > revisions_t;
private:
struct dependency
{
string_t name;
string_t version;
revisions_t revisions;
};
public:
typedef dependency dependency_t;
typedef std::map< string_t, dependency > dependencies_t;
public:
static const char* const product_version();
static const char* const product_revision();
static const char* const build_variant();
static const char* const build_time();
static const char* const build_user();
static const char* const build_host();
static const dependencies_t& dependencies();
static const char* const report();
};
} // end namespace _n
} // end namespace _second
} // end namespace _first
#endif
The version
provided is used to populate the result of product_version()
method and the location
is used to specify what directories should be read to determine revision information based on the source control method used. For example if the source under location
is from a subversion repository the revision()
method will return the revision number of the source code. In addition to details about the build system are also included as well as the time of the build and the build variant, such as debug or release. The report()
method provide a single string containing all the information.
Typically env.CreateVersion()
is used with the env.Compile()
method to allow dependencie between intermediate objects to be established as shown in the example that follows.
Example:
Import('env')
Version = "Product 00.01.00"
Sources = [
'main.cpp',
]
# We add this intermediary step to get the nodes representing the objects
# after compilation so that we can make the version file depend on this.
# Otherwise we could pass the source files directly to the Build() method
ProductObjects = env.Compile( Sources )
# If anything in the application changes we want a new version file with
# new build times and so on. We therefore make the version file depend
# on the result of compiling all our sources apart from the version file
# itself. This ensures that changes that cause the source to recompile will
# also cause the version file to be recompiled.
VersionFile = env.CreateVersion(
'version.cpp', # The name of the version file. This will be
# in the current directory.
ProductObjects, # What the version file depends on.
['company', 'product'], # The namespaces that should be used to nest
# the version info in.
Version, # The product version string.
env['base_path'] # The location below which all revision information
# should be gathered in this case basically all source
# from sconsctuct and below
)
# The program itself will depend on all objects and the version file
Objects = ProductObjects + VersionFile
env.Build( 'product_name', Objects )
Specifies the creation of a debug variant of the build. Usually including debug symbols, no optimisations and so on. What exactly is done depends on the toolchain and normal settings for it.
Specifies the creation of a release variant of the build. Usually with full optimisations turned on. What exactly is done depends on the toolchain and normal settings for it.
Specifies the creation of an instrumented variant of the build which allows coverage metrics to be gathered when the program is run. Usually including debug symbols this typically produces a fully instrumented build so that metrics can be obtained. Depending on the toolchain and coverage summariser used HTML or XML coverage output can be produced.
In order for the coverage to be determined for a given build target the program must be executed and therefore specifying --cov
implies --test
. To indicated that a build target is executable it should be built with the BuildTest()
method. If only Build()
is used then intermediate instrumented files will be produced, but the target will not be executed and no coverage data will be generated.
The test
variant does not actually produce an output directly. Instead it executes any target build using the BuildTest()
method. The runner
specified in the call to BuildTest()
(or the default if none is specified) is used to execute the target and interpret success or failure.
The following toolchains are currently supported:
Toolchain | Description |
---|---|
gcc101 |
g++ 10.1 |
gcc93 |
g++ 9.3 |
gcc92 |
g++ 9.2 |
gcc91 |
g++ 9.1 |
gcc83 |
g++ 8.3 |
gcc82 |
g++ 8.2 |
gcc81 |
g++ 8.1 |
gcc74 |
g++ 7.4 |
gcc73 |
g++ 7.3 |
gcc72 |
g++ 7.2 |
gcc71 |
g++ 7.1 |
gcc70 |
g++ 7.0 |
gcc64 |
g++ 6.4 |
gcc63 |
g++ 6.3 |
gcc62 |
g++ 6.2 |
gcc61 |
g++ 6.1 |
gcc60 |
g++ 6.0 |
gcc54 |
g++ 5.4 |
gcc53 |
g++ 5.3 |
gcc52 |
g++ 5.2 |
gcc51 |
g++ 5.1 |
gcc50 |
g++ 5.0 |
gcc49 |
g++ 4.9 |
gcc48 |
g++ 4.8 |
gcc47 |
g++ 4.7 |
gcc46 |
g++ 4.6 |
gcc45 |
g++ 4.5 |
gcc44 |
g++ 4.4 |
gcc43 |
g++ 4.3 |
gcc42 |
g++ 4.2 |
gcc41 |
g++ 4.1 |
gcc40 |
g++ 4.0 |
gcc34 |
g++ 3.4 |
Toolchain | Description |
---|---|
clang100 |
clang 10.0 |
clang90 |
clang 9.0 |
clang80 |
clang 8.0 |
clang70 |
clang 7.0 |
clang60 |
clang 6.0 |
clang50 |
clang 5.0 |
clang40 |
clang 4.0 |
clang39 |
clang 3.9 |
clang38 |
clang 3.8 |
clang37 |
clang 3.7 |
clang36 |
clang 3.6 |
clang35 |
clang 3.5 |
clang34 |
clang 3.4 |
clang33 |
clang 3.3 |
clang32 |
clang 3.2 |
Toolchain | Description |
---|---|
vc140 |
Visual C++ 14.0 (Visual C++ 2015) |
vc140e |
Visual C++ 14.0 (Visual C++ 2015 Exp) |
vc120 |
Visual C++ 12.0 (Visual C++ 2013) |
vc120e |
Visual C++ 12.0 (Visual C++ 2013 Exp) |
vc110 |
Visual C++ 11.0 (Visual C++ 2012) |
vc110e |
Visual C++ 11.0 (Visual C++ 2012 Exp) |
vc100 |
Visual C++ 10.0 (Visual C++ 2010) |
vc100e |
Visual C++ 10.0 Express (Visual C++ 2010 Exp) |
vc90 |
Visual C++ 9.0 (Visual C++ 2008) |
vc90e |
Visual C++ 9.0 Express (Visual C++ 2008 Exp) |
vc80 |
Visual C++ 8.0 (Visual C++ 2005) |
vc80e |
Visual C++ 8.0 Express (Visual C++ 2005 Exp) |
vc71 |
Visual C++ 7.1 (Visual C++ .NET 2003) |
vc70 |
Visual C++ 7.0 (Visual C++ .NET 2002) |
vc60 |
Visual C++ 6.0 |
It is not necessary to specify a toolchain when building. If none is specified the default toolchain for the current platform will be used. However if more toolchains are available and you want to use one or more then pass the --toolchains
option with a comma-separated list of toolchains from the list. For example to build with both GCC 4.9 and CLANG 3.4 you would add:
--toolchains=gcc49,clang35
to the command-line. It is not necessary to specify the versions of the toolchains if you just want the default version. For example you could write:
--toochains=gcc,clang
You can also use *
as a wildcard so all available GCC toolchains would be:
--toolchains=gcc*
The following platforms are supported:
- Linux
- Darwin (Mac)
- Windows
In order to make use of a dependency in your code it must both exist and be added to the current environment.
Typically a dependency is created by indicating a version or location of the dependency. It is up to each dependency how they interpret this information however there are usually sensible defaults and any dependency that is based on the supplied To then use the dependency you can either make it a default dependency by passing it as a list member to the default_dependencies
argument to cuppa.run
or by using the BuildWith()
or Use()
methods.
Out of the box cuppa supports the following dependencies.
Dependency | Description |
---|---|
boost |
The Boost Libraries |
qt4 |
The QT Libraries (version 4.8) for building applications with user interfaces |
qt5 |
The QT Libraries (version 5) for building applications with user interfaces |
quince |
The Quince (QUeries IN C++) Library a high quality library for accessing databases |
The boost
dependency simplifies the use of the Boost C++ Libraries.
The dependency provides the following options to specify which Boost source tree you wish to build against.
Option | Description |
---|---|
--boost-home |
Use this option to specify the location of your Boost source tree which contains the Boost boost and lib folders. For example if you downloaded Boost 1.55 and extracted it to ~/boost/boost_1_55 so that boost and lib are in that folder then you can make boost available to your builds by specifying --boost-home=~/boost/boost_1_55 on the command-line |
--boost-version |
If you specified a --thirdparty option then you can use --boost-version to indicate that you want to build with a particular version of boost under the thirdparty folder. The version should be in the format major_minor for example to specify using boost version 1.55 you would write --boost-version=1_55 . The boost dependency then tries to find a matching version of Boost under the thirdparty folder. |
--boost-build-always |
By default b2 is only executed once to build each library once. That is, it is assumed that the sourec is static and will not change. If it is possible or likely that the the boost sources will be changed then specifying --boost-build-always will ensure that b2 is executed every time to check if the library needs rebuilt or not. |
--boost-verbose-build |
By default only warning and error messages are displayed when boost is being built and even then only the summary (file and location) of the warning or error is shown. Passing this flag will show full build output allowing problems to be diagnosed. |
--boost-verbose-config |
By default no boost configuration messages are shown, such as performance configuration checks or configuration file reading. Passing this flag will show all configuration output allowing problems to be diagnosed. |
This dependency provides the following additional build methods to make using Boost easier.
Method | Description |
---|---|
BoostStaticLibs |
Specifies Boost libraries to lazily build for statically linking against. The libraries are specified using the libraries' names in a list. For example, to create a static version of the Boost.thread library for statically linking against you would write, BoostStaticLibs( [ 'thread' ] ) . For more details see BoostStaticLibs. |
BoostSharedLibs |
Specifies Boost libraries to lazily build for dynamically linking against. The libraries are specified using the libraries' names in a list. For example, to create a shared (dynamic) version of the Boost.thread library for dynamically linking against you would write, BoostSharedLibs( [ 'thread' ] ) . For more details see BoostSharedLibs. |
It is important to note that these methods return a node representing the built libraries. To link against the libraries you need to append the library to the environment's STATICLIBS
or DYNAMICLIBS
as appropriate.
Note: These methods will automatically add any required dependent libraries to the list of libraries built. For example if you want to link against Boost.Coroutine then these methods will also build Boost.Context and other libraries that Boost.Coroutine requires.
Note: Calling either
BoostStaticLibs
orBoostSharedLibs
will automatically implyBuildWith( ['boost'] )
ifboost
has not already been added as a dependency.
Use this method to specify a Boost library that you want to link statically with your application. For example, if you want to use Boost.System and Boost.Thread you would add this to your sconscript
file:
env.AppendUnique( STATICLIBS =
env.BoostStaticLibs( [
'system',
'thread'
] )
)
This is all that is required to ensure that the libraries are built correctly and linked with your target.
Use this method to specify a Boost library that you want to link statically with your application. For example, if you want to use Boost.Coroutine and Boost.Chrono you would add this to your sconscript
file:
env.AppendUnique( DYNAMICLIBS =
env.BoostSharedLibs( [
'system',
'chrono',
'coroutine'
] )
)
This is all that is required to ensure that the libraries are built correctly and linked with your target. It is important to note this will also "Do The Right Thing" in the presence of existing Boost installations. In other words this will pick up the correct shared library.
The qt4
and qt5
dependencies make use of the SCons_qt4
tool and SCons_qt5
tool respectively.
Both tools require a significant amount of setup however using cuppa reduces that down to a bare minimum.
In essence you simply require the necessary Qt development files and libraries installed on your system, pkg-config
if on Linux, enable the required version of Qt as a dependency and then "enable" the required components of Qt that you wish to use.
For example, on a Debian-based system you might install the required Qt development packages as follows if you required the Qt base components and QT Multimedia components:
sudo apt-get install pkg-config qtbase5-dev qtmultimedia5-dev qt5-default
Your sconstruct
file might look like this:
import cuppa
cuppa.run(
default_dependencies = [
'boost' # Oh look, we're using boost too
'qt5',
],
)
and then in your sconscript
file you might write something like this:
Import('env')
Product = "My System"
Major = "00"
Minor = "01"
Version = "{product_name} {major_ver}.{minor_ver}".format(
product_name = Product,
major_ver = Major,
minor_ver = Minor
)
env.EnableQt5Modules( [
'QtCore',
'QtWidgets',
'QtGui',
'QtMultimedia'
] )
# Let's separate out our source files from our test files
import re
sources = env.RecursiveGlob( re.compile( r'.*(?<!_test)[.]cpp$' ) )
env.AppendUnique( STATICLIBS =
env.BoostStaticLibs( [
'program_options',
'filesystem'
] )
)
# If we have UI files let's deal with those
uifiles = env.RecursiveGlob( "*.ui" )
env.Uic5( uifiles )
# Now identify the object files
objects = env.Compile( sources )
# Create a version file versioned against both objects and UI files
import os
version_file = env.CreateVersion(
'version.cpp',
[ objects, uifiles ],
[ 'my_namespace' ],
Version,
os.path.join( env['base_path'], ".." )
)
# Build the system
env.Build( 'my_system', [ objects, version_file ] )
# Now build the tests
tests = env.RecursiveGlob( "*_test.cpp" )
for test in tests:
env.BuildTest( env.TargetFrom( test ), test )
The sample sconscript
file shown illustrates what a project might look like using cuppa and something like Qt5. The documentation for the Qt4 and Qt5 tools are a good starting point to learn more about the facilities offered by these dependencies.
The Quince library (on Github https://github.com/mshepanski/quince) is a sophisticated, but very easy to use ORM library for C++. From the website:
Quince is a library that allows C++ programs to create SQL commands, and to have them executed on a relational DBMS. When data accompanies an outgoing command, quince first converts the data from C++ types (including user-defined classes and structs) to the DBMS's formats; and when data comes back in reply, quince makes the reverse conversion. So quince is an Object Relational Mapper (ORM).
The difference between quince and existing ORMs is this: Quince offers more support for sophisticated queries. A user who knows the power of a relational database will want to harness it, by issuing commands that employ server-side calculations, WHERE clauses, SELECT clauses, JOINs, DISTINCT, GROUP BY, INTERSECT, and so on, all combined and interconnected according to the user's own design. Quince treats such a user as the normal case, not an outlier. Of course it is also sometimes necessary to store or retrieve a single record, and quince makes that very easy to do; but it is only one point on a wide spectrum of possible uses. This reflects an underlying belief that an RDBMS is an engine of data processing as much as it is an engine of data storage.
To use quince
in your program you need to specify not only that you want to use quince
but also which backend you want to use. Out of the box Quince supports PostrgreSQL (on Github https://github.com/mshepanski/quince_postgresql and SQLite (on Github https://github.com/mshepanski/quince_sqlite).
A typical sconstruct
might be:
import cuppa
cuppa.run(
default_options = {
'quince-location' : "git+https://github.com/mshepanski/quince.git@dev",
'quince-postgresql-location': "git+https://github.com/mshepanski/quince_postgresql.git@dev",
},
default_dependencies = [
'quince',
'quince-postgresql',
],
)
As it happens Quince requires some libraries from the boost
dependency and this are automatically built as required. In addition if Quince is listed as a dependency all the Quince libraries themselves are built and all required libaries are statically linked to the final executable.
Each of the backends requires the appropriate development files and tools such as pg_config
to be available.
Cuppa makes building with header only libraries easy if all you need to do is add the libraries to your include path. To support this scenario cuppa provides the cuppa.location_dependency()
factory which can be used directly in this scenario or can be used as the foundation for more sophisticated dependencies. The remainder of this section describes how to use this class factory to define your own simple header library dependencies. For more sophisticated uses refer to the Custom Location Dependencies section.
The cuppa.location_dependency()
class factory takes one required argument, the name of the dependency, and returns a class that can be given to cuppa for later use.
For example, let's consider adding the non-boost version of the asio library as a project dependency. We can download a release and put it somewhere convenient (or directly point to the repository where the code is stored). Then we could write our sconstruct
file as follows:
# Specify where to find 'asio'
options['asio-location'] = "<location-of-asio>"
# Specify the include folder needed to allow compilation
options['asio-include'] = "asio/include"
cuppa.run(
# Add 'asio' as a dependency
dependencies = [
cuppa.location_dependency( 'asio' )
],
# Ensure the options we've added for 'asio' are added to the defaults
default_options = options,
# Make this a default dependency in all sconscripts
default_dependencies = [ 'asio' ]
)
We can simplify the above as follows:
cuppa.run(
# Make this a default dependency in all sconscripts
default_dependencies = [
cuppa.location_dependency( 'asio', location="<location-of-asio>", include="asio/include" )
]
)
Now we can compile against asio as expected by adding:
#include <asio.hpp>
As remarked above we are not limited to specifying location dependencies on local disk. It is also possible to specify remote locations. For example, again considering asio, if we want to build against a specfic release tag of the source code directly from the source repository we could write [asio-location]
as follows:
[asio-location] = "git+https://github.com/chriskohlhoff/asio.git@asio-1-10-4"
Of course being an ordinary option we could override this on the command-line if we wanted to, for example, to try building against master
. In that case we might write this while invoking scons
:
scons -D --asio-location="git+https://github.com/chriskohlhoff/asio.git"
As you might expect building against a branch rather than a specific tag or archived release automatically updates as the branch is updated.
As shown previously the cuppa.location_dependency()
factory function takes a name (referred to as <dependency-name>
below) for the dependency and returns a class that can be passed as a dependency to cuppa.
Classes created the factory function provide the following Scons options.
Option | Description |
---|---|
--<dependency-name>-location |
Specify the location of the dependency. The location can be any local directory, or any URL to a remote directory or archive, as well any version controlled location supported by pip. |
--<dependency-name>-include |
This allows you to specify a subfolder which should be added to the INCPATH Scons variable. |
--<dependency-name>-sys-include |
This allows you to specify a subfolder which should be added to the SYSINCPATH Scons variable. |
--<dependency-name>-branch |
You may use the --<dependency-name>-branch option with local directories to specify that a subfolder (as a branch) is added to the location given. This follows the approach used with Subversion to allow branches to be specfied as folders. If you are using a remote location branch information is typically provided as part of the URL and so this option is not needed. The option also finds use for informational purpores if the given location is not under version control and branch or revision information is not available. |
Typically only --<dependency-name>-location
and --<dependency-name>-include
are needed when used with remote URLs.
When a remote URL is specified, and the files have not previously been obtained, then cuppa attemtps to either download, checkout
or pull
them. Once retrieved compressed archives are expanded into a suitable location.
Cuppa stores any retrieved files under a sub-folder under _cuppa
. The sub-folder name is derived from the URL so that it is unique for a given URL. That means different branches of the same repository will be checked out in to different folders.
If the files are already present and they were not retrieved from version control then nothing is done. However, if the files were retrieved from version control then cuppa will attempt to update the files to the latest revision as allowed by the original URL specified.
It is possibly to create your own dependencies, like the boost
dependency. As far as cuppa is concerned a dependency is any class that provides the following:
import os
from cuppa.log import logger
from cuppa.colourise import as_notice, as_error
class <dependency>:
_name = <dependency>
_cached_locations = {}
@classmethod
def add_options( cls, add_option ):
# Specify any options here
location_name = cls._name + "-location"
add_option( '--' + location_name,
dest=location_name,
type='string', nargs=1,
action='store',
help = cls._name + ' location to build against' )
@classmethod
def add_to_env( cls, env, add_dependency ):
add_dependency( cls._name, cls.create )
@classmethod
def name( cls ):
return cls._name
# Helper function not required as part of interface
@classmethod
def _location_id( cls, env ):
location = env.get_option( cls._name + "-location" )
if not location and env['thirdparty']:
location = os.path.join( env['thirdparty'], cls._name )
if not location:
logger.debug( "No location specified for dependency [{}]."
" Dependency not available."
.format( cls._name.title() ) )
return None
return location
# Helper function not required as part of interface
@classmethod
def _get_location( cls, env ):
location_id = cls._location_id( env )
if not location_id:
return None
if location_id not in cls._cached_locations:
try:
cls._cached_locations[location_id] = cuppa.location.Location( env, location )
except cuppa.location.LocationException as error:
logger.error(
"Could not get location for [{}] at [{}]."
" Failed with error [{}]"
.format( as_notice( cls._name.title() ),
as_notice( str(location) ),
as_error( error ) )
)
return None
return cls._cached_locations[location_id]
@classmethod
def create( cls, env ):
location = cls._get_location( env )
if not location:
return None
return cls( env, location )
# The following methods are not strictly required
def __init__( self, env, location ):
self._location = location
def __call__( self, env, toolchain, variant ):
# Update the environment
pass
def version( self ):
return str(self._location.version())
def repository( self ):
return self._location.repository()
def branch( self ):
return self._location.branch()
def revisions( self ):
return self._location.revisions()
Only add_to_env()
, create()
, __call__
and name()
are strictly required but it makes sense to provide the others as using cuppa.location.Location
makes this trivial.
Once the basic dependency is written abitrarily complex relationships can be built by making use of Scons builders or other dependency related tools. It is worth looking at the boost
dependency as an example of a complex dependency.
Most dependencies will be much more trivial than the boost
dependency and many can be built directly on top of the cuppa.location_dependency()
factory. For example, let's look again at using asio as a dependency. We've already seen how we can create a basic dependency by writing:
asio_dependency = cuppa.location_dependency( 'asio' )
We can take this a step further and create our dependency by inheriting from the class returned by the factory and overriding the __call__
method:
import cuppa
class asio( cuppa.location_dependency( 'asio', location="<location-of-asio>", sys_include="asio/include" ) ):
def __call__( self, env, toolchain, variant ):
super(asio,self).__call__( env, toolchain, variant )
# Update the environment as we need to, for example...
# Perhaps we remove deprecated features
env.AppendUnique( CPPDEFINES = [
'ASIO_NO_DEPRECATED',
] )
cuppa.run(
# Add 'asio' as a dependency and make this a default dependency in all sconscripts
default_dependencies = [ asio ]
)
Profiles provide a simpler approach to modifying a build environment in a repeatable controlled fashion than a full-blown dependency. The only real difference between a profile and a dependency is that a profile typically has not related code or any "state". The easiest way to write a profile is to use the cuppa profile
class factory. For example to write a profile
that adds some flags to the build you could write:
import cuppa
class my_profile( cuppa.profile( 'my_profile' ) ):
def __call__( self, env, toolchain, variant ):
# Update the environment as we need to, for example...
env.AppendUnique( CPPDEFINES = [
'MY_IMPORTANT_DEFINE',
] )
cuppa.run(
# Add 'my_profile' as a profile and make this a default profile in all sconscripts
default_profiles = [ my_profile ]
)
This work is based on the build system used in clearpool.io during development of its next generation exchange platform.