Civic-Digital-Twins Modeling Framework
Project description
Civic-Digital-Twins Modeling Framework
This repository contains a Python package implementing a Civic-Digital-Twins modeling framework. The framework is designed to support defining digital twins models and evaluating them in simulated environments with varying contextual conditions. We develop this package at @fbk-most, a research unit at Fondazione Bruno Kessler.
Note: this package is currently in an early development stage.
Conceptual Overview
The framework is organised in three layers.
Engine layer
The engine (civic_digital_twins.dt_model.engine) is an embedded DSL
compiler. The programmer builds a computation graph (DAG) by composing
typed nodes — constants, placeholders, and operations — using ordinary Python
expressions. The graph is then linearised by topological sorting and
evaluated by a NumPy-based interpreter that maps each node to the
corresponding numpy operation.
from civic_digital_twins.dt_model.engine.frontend import graph, linearize
from civic_digital_twins.dt_model.engine.numpybackend import executor
import numpy as np
a = graph.placeholder("a")
b = graph.placeholder("b")
c = a * 2 + b
state = executor.State(values={a: np.asarray(3.0), b: np.asarray(1.0)})
executor.evaluate_nodes(state, *linearize.forest(c))
print(state.get_node_value(c)) # 7.0
See docs/design/dd-cdt-engine.md for a full description of the engine.
Model / simulation layer
The model layer (civic_digital_twins.dt_model) provides higher-level
abstractions built on top of the engine:
Index/TimeseriesIndex— named wrappers around graph nodes. An index can be a constant, a distribution (sampled at evaluation time), or a formula.Model— a named collection ofIndexobjects. DeclareInputs,Outputs, andExposeas inner dataclasses to make the inter-model interface explicit. Sub-models are wired via constructor arguments, producing a composable pipeline.ModelVariant— selects among pre-constructedModelimplementations sharing the same I/O contract. The active variant is resolved at construction time via a string key.Evaluation— evaluates a model over a sequence of weighted scenarios, each of which maps every abstract index to a concrete value.Ensemble/WeightedScenario— a protocol and type alias that define the scenario contract consumed byEvaluation.
from civic_digital_twins.dt_model import Evaluation, Model, DistributionIndex
from civic_digital_twins.dt_model.model.index import Index
from scipy import stats
# Two distribution-backed indexes
x = DistributionIndex("x", stats.uniform, {"loc": 0.0, "scale": 1.0})
y = DistributionIndex("y", stats.uniform, {"loc": 0.0, "scale": 1.0})
result = Index("result", x + y)
model = Model("example", [x, y, result])
See docs/design/dd-cdt-model.md for the full
reference: index types, Model API, ModelVariant, Evaluation, and the
vertical extension pattern.
Usage patterns
Three concrete usage patterns are illustrated in the examples/ directory.
Direct pattern (examples/mobility_bologna/) — the model consists
entirely of Index and TimeseriesIndex objects;
DistributionEnsemble samples each distribution-backed index to produce
weighted scenarios; Evaluation.evaluate() runs the engine and returns an
EvaluationResult.
Vertical extension pattern (examples/overtourism_molveno/) — the
domain-specific layer introduces ContextVariable, PresenceVariable,
Constraint, and OvertourismModel on top of the core types.
OvertourismEnsemble samples context variables and produces weighted
scenarios. Evaluation.evaluate(axes={pv: array, …}) evaluates the model
on a multi-dimensional grid, returning arrays of shape (N₀, …, Nₖ, S)
where S is the number of scenarios and each Nᵢ corresponds to one
presence axis.
Model modularity — for larger models, the dataclass-based I/O API
(Inputs, Outputs, Expose) makes inter-model wiring explicit and
machine-checkable. ModelVariant enables swapping between alternative
implementations at construction time. Both the Bologna and Molveno examples
are rewritten using this API. See
docs/design/dd-cdt-modularity.md
for the full guide.
Installation
The package name is civic-digital-twins on PyPi. Install
using pip:
pip install civic-digital-twins
or, using uv:
uv add civic-digital-twins
The main package name is civic_digital_twins:
import civic_digital_twins
or
from civic_digital_twins import dt_model
Minimum Python Version
Python 3.11. Tested against Python 3.11, 3.12, 3.13, and 3.14.
API Stability Guarantees
The package is currently in an early development stage. We do not anticipate breaking APIs without a good reason to do so, yet, breaking changes may occur from time to time. We generally expect subpackages within the top-level package to change more frequently.
Development Setup
We use uv for managing the development environment.
To get started, run:
git clone https://github.com/fbk-most/civic-digital-twins
cd civic-digital-twins
uv venv
source .venv/bin/activate
uv sync --dev
We use pytest for testing. To run tests use this command (from inside the virtual environment):
pytest
Each pull request is automatically tested using GitHub Actions. The workflow
is defined in .github/workflows/test.yml.
Updating Dependencies
uv self update
uv sync --upgrade
Releasing
Per-release checklist (six manual steps):
-
Make sure the version number in
pyproject.tomlis correct. -
Regenerate the lockfile to record the new version:
uv lock -
Update
CHANGELOG.md: promote the[Unreleased]heading to[<version>] - <date>and add the corresponding comparison link at the bottom. -
Check that documentation
Last-Updateddates are in sync with actual commit dates:git log -1 --format="%ai" -- docs/design/dd-cdt-engine.md git log -1 --format="%ai" -- docs/design/dd-cdt-model.md git log -1 --format="%ai" -- docs/design/dd-cdt-modularity.md git log -1 --format="%ai" -- docs/getting-started.md
Update any
Last-Updatedfields that are out of date. -
Verify that the runnable doc scripts in
examples/doc/are in sync with the code snippets in the corresponding documentation files, and that they all execute without errors:uv run python examples/doc/doc_engine.py uv run python examples/doc/doc_model.py uv run python examples/doc/doc_modularity.py uv run python examples/doc/doc_getting_started.py
-
Commit the changes above, then create and push a version tag:
git add pyproject.toml uv.lock CHANGELOG.md docs/ git commit -m "chore: prepare v<version> release" git tag v<version> && git push origin main v<version>
After the tag is pushed, go to the repository's Releases page, review the
auto-created draft, write release notes, and click Publish release. This
triggers the publish.yml workflow, which
builds the sdist + wheel, runs twine check, and publishes to PyPI via OIDC —
no manual build or upload step is needed.
Precondition: PyPI's Trusted Publisher must be configured for this repository before the first release. See the PyPI Trusted Publishers documentation for setup instructions.
Documentation
| Document | Description |
|---|---|
| Getting Started | Step-by-step guide covering the direct and vertical extension usage patterns. |
| dd-cdt-engine.md | DSL compiler engine — graph nodes, topological sorting, NumPy executor. |
| dd-cdt-model.md | Model / simulation layer — Model, Evaluation, WeightedScenario, and the vertical extension pattern. |
| dd-cdt-modularity.md | Model modularity concept guide — dataclass I/O API, ModelVariant, decomposition patterns, and Bologna worked example. |
License
SPDX-License-Identifier: Apache-2.0
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file civic_digital_twins-0.8.1.tar.gz.
File metadata
- Download URL: civic_digital_twins-0.8.1.tar.gz
- Upload date:
- Size: 224.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6f6c8ad58b4e92db8f04aeeb72121c1ffb994d11de1873ad39c4cebaf7ef3c49
|
|
| MD5 |
7f5c09196d359c6f9263e80fd236ca05
|
|
| BLAKE2b-256 |
15e344847843226c56d2d67bbed3a5dbac323bc3037c22f8f0defae3f248910c
|
Provenance
The following attestation bundles were made for civic_digital_twins-0.8.1.tar.gz:
Publisher:
publish.yml on fbk-most/civic-digital-twins
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
civic_digital_twins-0.8.1.tar.gz -
Subject digest:
6f6c8ad58b4e92db8f04aeeb72121c1ffb994d11de1873ad39c4cebaf7ef3c49 - Sigstore transparency entry: 1214404802
- Sigstore integration time:
-
Permalink:
fbk-most/civic-digital-twins@46f0549fa512427fc08e7dba39e353ec9f6abfe2 -
Branch / Tag:
refs/tags/v0.8.1 - Owner: https://github.com/fbk-most
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@46f0549fa512427fc08e7dba39e353ec9f6abfe2 -
Trigger Event:
release
-
Statement type:
File details
Details for the file civic_digital_twins-0.8.1-py3-none-any.whl.
File metadata
- Download URL: civic_digital_twins-0.8.1-py3-none-any.whl
- Upload date:
- Size: 59.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7ddd13926c5accf54769f6bc14289c6d2c85c82e78137f75c4e3686adf466ebc
|
|
| MD5 |
86cfe9a460bebbc8c3c6dcf0b6a87b27
|
|
| BLAKE2b-256 |
ff653a9a162a6bee4528feddd0e527567acd5063ecdf32b52e1ce7c3833ec4dc
|
Provenance
The following attestation bundles were made for civic_digital_twins-0.8.1-py3-none-any.whl:
Publisher:
publish.yml on fbk-most/civic-digital-twins
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
civic_digital_twins-0.8.1-py3-none-any.whl -
Subject digest:
7ddd13926c5accf54769f6bc14289c6d2c85c82e78137f75c4e3686adf466ebc - Sigstore transparency entry: 1214404912
- Sigstore integration time:
-
Permalink:
fbk-most/civic-digital-twins@46f0549fa512427fc08e7dba39e353ec9f6abfe2 -
Branch / Tag:
refs/tags/v0.8.1 - Owner: https://github.com/fbk-most
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@46f0549fa512427fc08e7dba39e353ec9f6abfe2 -
Trigger Event:
release
-
Statement type: