Skip to main content

Civic-Digital-Twins Modeling Framework

Project description

Civic-Digital-Twins Modeling Framework

Build Status codecov PyPI version Python Versions License

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 of Index objects. Declare Inputs, Outputs, and Expose as inner dataclasses to make the inter-model interface explicit. Sub-models are wired via constructor arguments, producing a composable pipeline.
  • ModelVariant — selects among pre-constructed Model implementations 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 by Evaluation.
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):

  1. Make sure the version number in pyproject.toml is correct.

  2. Regenerate the lockfile to record the new version:

    uv lock
    
  3. Update CHANGELOG.md: promote the [Unreleased] heading to [<version>] - <date> and add the corresponding comparison link at the bottom.

  4. Check that documentation Last-Updated dates 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-Updated fields that are out of date.

  5. 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
    
  6. 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


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

civic_digital_twins-0.8.1.tar.gz (224.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

civic_digital_twins-0.8.1-py3-none-any.whl (59.2 kB view details)

Uploaded Python 3

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

Hashes for civic_digital_twins-0.8.1.tar.gz
Algorithm Hash digest
SHA256 6f6c8ad58b4e92db8f04aeeb72121c1ffb994d11de1873ad39c4cebaf7ef3c49
MD5 7f5c09196d359c6f9263e80fd236ca05
BLAKE2b-256 15e344847843226c56d2d67bbed3a5dbac323bc3037c22f8f0defae3f248910c

See more details on using hashes here.

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

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file civic_digital_twins-0.8.1-py3-none-any.whl.

File metadata

File hashes

Hashes for civic_digital_twins-0.8.1-py3-none-any.whl
Algorithm Hash digest
SHA256 7ddd13926c5accf54769f6bc14289c6d2c85c82e78137f75c4e3686adf466ebc
MD5 86cfe9a460bebbc8c3c6dcf0b6a87b27
BLAKE2b-256 ff653a9a162a6bee4528feddd0e527567acd5063ecdf32b52e1ce7c3833ec4dc

See more details on using hashes here.

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

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page