Skip to main content

Wrapper around a NetworkX Digraph to model and visualize tasks/todos, their duration and interdependencies

Project description

License: MIT Python Versions (officially) supported Pypi status badge

Task Dependency Graph

Task Dependency Graph is a Python package that allows to model tasks and dependencies between tasks as a directed graph. It also supports visualizing the graph with dot for a graph-like view or mermaid for Gantt charts.

The package is built on networkx and under the hood the task dependency graph is just a networkx DiGraph. For visualization, it uses GraphViz via kroki (in a Docker container).

Example / How to use

Install the package from PyPI

pip install taskdependencygraph

Imagine the following scenario:

You and your partner are invited to a birthday party. You promised to bring a cake. Baking the cake and the recipe can be divided into atomic tasks, all of which have a duration. Those are the nodes of the task dependency graph (TDG).

Tasks are created like this:

import uuid
from datetime import timedelta

from taskdependencygraph.models import TaskNode

my_task = TaskNode(
    id=uuid.uuid4(),  # boilerplate only, but you need the ID to find nodes in your graph later on
    name="Shop groceries",  # human readable description
    external_id="some unique string",  # technical ID for those who don't like uuids ;)
    planned_duration=timedelta(minutes=15)  # how long it probably takes
    # You may also add an assignee, phase, tags, earliest_starttime, is_milestone, or execution_status
)

The tasks depend on each other: You cannot prepare the cake without buying the ingredients first. You cannot decorate the cake before you've made it. Which task has which mandatory predecessor tasks is defined in task dependencies. Those are the edges of our task dependency graph.

Task dependencies are created like this:

import uuid

from taskdependencygraph.models import TaskDependencyEdge, TaskNode

shopping_groceries = TaskNode(...)
mixing_flour_and_sugar = TaskNode(...)
baking_in_the_oven = TaskNode(...)

buy_ingredients_before_mixing_them = TaskDependencyEdge(
    id=uuid.uuid4(),  # boilerplate
    predecessor_task=shopping_groceries.id,
    successor_task=mixing_flour_and_sugar.id
)
mix_ingredients_before_baking_the_cake = TaskDependencyEdge(
    id=uuid.uuid4(),  # boilerplate
    predecessor_task=mixing_flour_and_sugar.id,
    successor_task=baking_in_the_oven.id
)

The graph is made out of tasks (nodes), task dependencies (edges) and a start datetime.

from datetime import datetime, UTC

from taskdependencygraph import TaskDependencyGraph
from taskdependencygraph.models import TaskNode, TaskDependencyEdge

# nodes
shopping_groceries = TaskNode(...)
mixing_flour_and_sugar = TaskNode(...)
baking_in_the_oven = TaskNode(...)

# edges
buy_ingredients_before_mixing_them = TaskDependencyEdge(...)
mix_ingredients_before_baking_the_cake = TaskDependencyEdge(...)

# graph
tdg = TaskDependencyGraph(
    task_list=[shopping_groceries, mixing_flour_and_sugar, baking_in_the_oven],
    dependency_list=[buy_ingredients_before_mixing_them, mix_ingredients_before_baking_the_cake],
    starting_time_of_run=datetime(2025, 1, 1, 12, 0, 0, tzinfo=UTC)
)

Once you have a graph, you can:

  • Validate the definition before construction — TaskDependencyGraph.validate_definition(task_list, dependency_list) checks for duplicate task/dependency IDs, duplicate external IDs, missing edge endpoints, invalid milestone durations, duplicate edge pairs, and cycles, returning a GraphDefinitionValidationResult.
  • Get the full scheduletdg.create_schedule_report() returns a ScheduleReport with planned start/finish, critical-path flag, and total slack for every task.
  • Inspect the critical pathtdg.get_critical_path_tasks() returns the ordered list of TaskNode objects on the critical path.
  • Calculate total slacktdg.calculate_total_slack_of_task(task_id) returns how much a task can slip without affecting the deadline.
  • Query finish timestdg.calculate_planned_finish_time_of_task(task_id) and tdg.calculate_planned_finish_time_of_graph().
  • Assign persons to tasks and check if any person has more than one task assigned at a time.

Find a complete working example in the demo unittest. This demo test is also the basis for the following visualization examples.

Visualization with Kroki

You can visualize the dependencies either as rather simple technical graph or as Gantt chart, when you start kroki in a docker container:

# docker-compose.yaml
services:
  kroki: # see https://docs.kroki.io/kroki/setup/use-docker-or-podman/#_run_multiple_kroki_containers_together
    image: yuzutech/kroki:0.30.1
    depends_on:
      - mermaid
    environment:
      - KROKI_MERMAID_HOST=mermaid
    ports:
      - "8123:8000"
  mermaid:
    image: yuzutech/kroki-mermaid
# run
# docker-compose up -d
# and kroki is ready at localhost:8123
import asyncio

from taskdependencygraph import TaskDependencyGraph
from taskdependencygraph.plotting import KrokiClient, KrokiConfig


async def plot_a_graph() -> None:
    tdg = TaskDependencyGraph(...)  # with all nodes and edges and stuff

    config = KrokiConfig(host="http://localhost:8123")  # w/o docker, you may also use kroki.io, but it's rate limited
    kroki_client = KrokiClient(config=config)

    await kroki_client.plot_as_svg(tdg, mode="gantt")  # or mode="dot"


if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(plot_a_graph())

The result may look like this:

Gantt Chart (Mermaid)

A Gantt chart

The tasks marked in red mark the critical path on which delays affect the finishing time. The 🔶 are milestones which mark important moments in your project (often you want to have a group of tasks like ' Shopping' done before starting with the next step, even though there's no "real" dependency between e.g. Cake Base and buying the strawberries.) The Gantt chart is useful to get an overview of your project and to identify which tasks are crucial.

The Gantt output is customizable via MermaidGanttConfig — you can set the title, axis format, tick interval, and group tasks into sections by their phase field.

Raw Graph ("dot" engine)

The raw graph helps you to understand the tasks and dependencies setup in a not so shiny but verbose fashion.

Storing the Graph in a Database

You can store the nodes and edges on a database. We suggest to just use two tables: One for the edges, one for the nodes. You can even add trigger-based DB constraints to prevent loops in the graph which are faster than you might guess, even for hundreds of tasks.

Maintainers/ Further Development / Professional Support

This library was built for and then cut out of a mainly internal project by @hf-crings, @OLILHR, @hf-sheese and @hf-kklein, but we decided to publish it, because it might be useful to someone. This is why some things are hardcoded here and there and why some features might seem unintuitive at first glance.

We at Hochfrequenz also built a SQLAlchemy+FastAPI+htmx web application around this library in which you can plan and schedule time-critical tasks and projects in the browser. It's ready to use, but not pretty enough to publish it yet ;) Just ping us if interested.

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

taskdependencygraph-0.2.0.tar.gz (32.5 kB view details)

Uploaded Source

Built Distribution

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

taskdependencygraph-0.2.0-py3-none-any.whl (31.4 kB view details)

Uploaded Python 3

File details

Details for the file taskdependencygraph-0.2.0.tar.gz.

File metadata

  • Download URL: taskdependencygraph-0.2.0.tar.gz
  • Upload date:
  • Size: 32.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for taskdependencygraph-0.2.0.tar.gz
Algorithm Hash digest
SHA256 456a0b43b2c837f58ec2ea54a5c785a9efaf4e3141bacbc022f2577d0b535e2f
MD5 835fa92b1a26466b1ff982e6c4b59432
BLAKE2b-256 2f4bf5d4abc614a37541372bcf6a0337fc89569f510fc4c3b30a9909ff329815

See more details on using hashes here.

Provenance

The following attestation bundles were made for taskdependencygraph-0.2.0.tar.gz:

Publisher: python-publish.yml on Hochfrequenz/task-dependency-graph

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

File details

Details for the file taskdependencygraph-0.2.0-py3-none-any.whl.

File metadata

File hashes

Hashes for taskdependencygraph-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 8865840af781dfa877b2e753cf6b080984596174074314e041d5b13a64015984
MD5 56d63458b21cd798d95ec3913cd1ad30
BLAKE2b-256 3ba30bddfe781a06d875608bdb7a9729b7d754a9c96ed56bb0b5f4c9a6f04ff7

See more details on using hashes here.

Provenance

The following attestation bundles were made for taskdependencygraph-0.2.0-py3-none-any.whl:

Publisher: python-publish.yml on Hochfrequenz/task-dependency-graph

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