Adding functions to Brightway processes
You can install bw-functional via [pip] from [PyPI]:
$ pip install bw-functional
Multifunctional activities can lead to linear algebra problems which don't have exactly one solution. Therefore, we commonly need to apply a handling function to either partition such activities, or otherwise manipulate their data such that they allow for the creation of a square and non-singular technosphere matrix.
This library is designed around the following workflow:
Users create and register a bw_functional.FunctionalSQLiteDatabase
. Registering this database must include the database metadata key default_allocation
, which refers to an allocation strategy function present in bw_functional.allocation_strategies
.
import bw_functional
mf_db = bw_functional.FunctionalSQLiteDatabase("emojis FTW")
mf_db.register()
Multifunctional process(es) are created and written to the FunctionalSQLiteDatabase
. A multifunctional process is any process with multiple "functions", either outputs (products) and/or input (reducts).
mf_data = {
("emojis FTW", "😼"): {
"type": "product",
"name": "meow",
"unit": "kg",
"processor": ("emojis FTW", "1"),
"properties": {
"price": {'unit': 'EUR', 'amount': 7, 'normalize': True},
"mass": {'unit': 'kg', 'amount': 1, 'normalize': True},
},
},
("emojis FTW", "🐶"): {
"type": "product",
"name": "woof",
"unit": "kg",
"processor": ("emojis FTW", "1"),
"properties": {
"price": {'unit': 'EUR', 'amount': 12, 'normalize': True},
"mass": {'unit': 'kg', 'amount': 4, 'normalize': True},
},
},
("emojis FTW", "1"): {
"name": "process - 1",
"location": "somewhere",
"exchanges": [
{
"type": "production",
"input": ("emojis FTW", "😼"),
"amount": 4,
},
{
"type": "production",
"input": ("emojis FTW", "🐶"),
"amount": 6,
},
],
}
}
LCA calculations can then be done as normal. See dev/basic_example.ipynb
for a simple example.
WORK IN PROGRESS
bw-functional
includes the following built-in allocation functions:
manual_allocation
: Does allocation based on the "allocation" field of the Product. Doesn't normalize by amount of production exchange.equal
: Splits burdens equally among all functional edges.
You can also do property-based allocation by specifying the property label in the allocation
field of the Process.
Individual processes can override the default database allocation by specifying their own allocation
:
import bw2data as bd
node = bd.get_activity(database="emojis FTW", code="1")
node["allocation"] = "mass"
node.save()
Recent Brightway versions allow users to specify which graph nodes types should be used when building matrices, and which types can be ignored. We create a multifunctional process node with the type multifunctional
, which will be ignored when creating processed datapackages. However, in our database class FunctionalSQLiteDatabase
we change the function which creates these processed datapackages to load the multifunctional processes, perform whatever strategy is needed to handle multifunctionality, and then use the results of those handling strategies (e.g. monofunctional processes) in the processed datapackage.
We also tell MultifunctionalDatabase
to load a new ReadOnlyProcess
process class instead of the standard Activity
class when interacting with the database. This new class is read only because the data is generated from the multifunctional process itself - if updates are needed, either that input process or the allocation function should be modified.
Contributions are very welcome. To learn more, see the Contributor Guide.
Distributed under the terms of the BSD 3 Clause license, multifunctional is free and open source software.
If you encounter any problems, please file an issue along with a detailed description.
You can build the documentation locally by installing the documentation Conda environment:
conda env create -f docs/environment.yml
activating the environment
conda activate sphinx_multifunctional
and running the build command:
sphinx-build docs _build/html --builder=html --jobs=auto --write-all; open _build/html/index.html