Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Experiment/recursive controller #24

Open
wants to merge 68 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
68 commits
Select commit Hold shift + click to select a range
e1ba772
Imported commits from old repo
mwpb Feb 25, 2025
7132264
Update pyproject.toml
mwpb Feb 25, 2025
bcb4603
Update pyproject.toml
mwpb Feb 25, 2025
f33bed7
Update pyproject.toml
mwpb Feb 25, 2025
6afbb9f
Update pyproject.toml
mwpb Feb 25, 2025
1fc0e03
Fix CI errors
mwpb Feb 25, 2025
4e6c5b0
Fix CI
mwpb Feb 25, 2025
efb551b
Satisfy ruff
mwpb Feb 25, 2025
c04a581
Explicit enum for Python <3.11
mwpb Feb 25, 2025
6bfc5b7
Fix typo
mwpb Feb 25, 2025
14f0c68
Avoid extra read/write
mwpb Feb 25, 2025
9603b76
Link outputs across loop iterations
mwpb Feb 25, 2025
7173649
Link outputs through switch statements
mwpb Feb 25, 2025
a237646
Move numerical-worker
mwpb Feb 26, 2025
3d5342e
Draft nexus-worker and polling loop
mwpb Feb 27, 2025
5d99dd7
Give up on match nodes for controller
mwpb Feb 27, 2025
9d46e7a
Convert storage to use bytes
mwpb Feb 27, 2025
30e4ae4
Create executor interface and ShellExecutor
mwpb Feb 27, 2025
bb4c4e5
Improve debugging ShellExecutor
mwpb Feb 27, 2025
4b4fc83
Make stdout|err path configurable
mwpb Feb 27, 2025
5d7e7eb
Run worker in detached process
mwpb Feb 28, 2025
cb432d2
Create VQE graph
mwpb Feb 28, 2025
33c7fe3
Move some files to tierkreis-chemistry-demo
mwpb Mar 5, 2025
51fea51
Take care of relative paths in generate_protos
mwpb Mar 5, 2025
5adc448
Remove extra imports
mwpb Mar 6, 2025
48e3afa
Include old inputs in loops
mwpb Mar 6, 2025
cab188b
Remove extra changes
mwpb Mar 13, 2025
1fa3ca7
Improve tests
mwpb Mar 14, 2025
a08af46
Remove extra method
mwpb Mar 14, 2025
e5fd210
Improve tests
mwpb Mar 14, 2025
530891f
Fix comment
mwpb Mar 14, 2025
d4e361e
Update .gitignore
mwpb Mar 20, 2025
931e254
Remove extra README
mwpb Mar 20, 2025
368386a
Move logs into checkpoints dir
mwpb Mar 20, 2025
09139ca
Remove extra deps
mwpb Mar 20, 2025
2217507
Revert indentation
mwpb Mar 20, 2025
21dfaed
Revert test improvements
mwpb Mar 20, 2025
a5e8ea0
Remove extra line
mwpb Mar 20, 2025
2dcc1e8
Add extra line
mwpb Mar 20, 2025
f815516
Remove extra print
mwpb Mar 20, 2025
da538b3
Remove extra method
mwpb Mar 20, 2025
e16209a
Remove extra whitespace
mwpb Mar 20, 2025
b8486f0
Add optional logs_path to NodeDefinition
mwpb Mar 20, 2025
4eae340
Add node definitions for non-function nodes
mwpb Mar 20, 2025
89f3c07
Use StrEnum rather than overriding __str__
mwpb Mar 24, 2025
4ca6766
Simplify exception handling
mwpb Mar 24, 2025
b28f87d
Fix typo
mwpb Mar 24, 2025
1cdb461
Improve directory layout
mwpb Mar 24, 2025
72d0734
Try fix CI
mwpb Mar 25, 2025
e90126a
Satisfy ruff
mwpb Mar 25, 2025
cbe74c1
Satisfy ruff
mwpb Mar 25, 2025
3b32fb4
Bump python version
mwpb Mar 25, 2025
2e20caa
Revert StrEnum
mwpb Mar 25, 2025
57a6b3b
Revert Python version bump
mwpb Mar 25, 2025
6b5c775
Update models.py
mwpb Mar 25, 2025
a5074e7
Ignore previous type errors
mwpb Mar 25, 2025
9138b19
Fix test imports
mwpb Mar 25, 2025
5e9e674
Satisfy ruff
mwpb Mar 25, 2025
d86da41
Factor out NodeRunData
mwpb Mar 25, 2025
09acdcb
Factor resume by get_nodes_to_start
mwpb Mar 25, 2025
73a9854
Logging is done via node_definition
mwpb Mar 25, 2025
23478c3
Simplify test_resume
mwpb Mar 25, 2025
d9b3509
Satisfy ruff
mwpb Mar 26, 2025
6d115bd
Simplify user facing code
mwpb Mar 27, 2025
a8e9cc4
Replace custom build system with hatchling
mwpb Mar 27, 2025
fe82a17
Refactor into WalkResult
mwpb Mar 27, 2025
ee02399
Improve function names
mwpb Mar 27, 2025
8a1de38
Satisfy ruff
mwpb Mar 27, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
/target
.idea

# Byte-compiled / optimized / DLL files
__pycache__/
Expand Down
1 change: 0 additions & 1 deletion python/.gitignore
Original file line number Diff line number Diff line change
@@ -1,2 +1 @@
./venv
tierkreis/core/protos/
5 changes: 5 additions & 0 deletions python/examples/launchers/numerical-worker
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
#!/usr/bin/env bash
printf "\n\n------ NUMERICAL WORKER ------"

LAUNCHER_DIR=$(dirname "$0")
uv run "$LAUNCHER_DIR/../numerical-worker" $1
96 changes: 96 additions & 0 deletions python/examples/numerical-worker/__main__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
import json
from logging import getLogger
from pathlib import Path
from sys import argv

from pydantic import BaseModel

logger = getLogger(__name__)


class NodeDefinition(BaseModel):
function_name: str
inputs: dict[str, Path]
outputs: dict[str, Path]
done_path: Path


def iadd(a: int, b: int) -> int:
logger.debug(f"iadd {a} {b}")
return a + b


def itimes(a: int, b: int) -> int:
logger.debug(f"itimes {a} {b}")
return a * b


def igt(a: int, b: int) -> bool:
logger.debug(f"igt {a} {b}")
return a > b


def run(node_definition: NodeDefinition):
logger.debug(node_definition)
if node_definition.function_name == "iadd":
with open(node_definition.inputs["a"], "rb") as fh:
a: int = json.loads(fh.read())
with open(node_definition.inputs["b"], "rb") as fh:
b: int = json.loads(fh.read())

c = iadd(a, b)

with open(node_definition.outputs["value"], "w+") as fh:
fh.write(json.dumps(c))

elif node_definition.function_name == "itimes":
with open(node_definition.inputs["a"], "rb") as fh:
a: int = json.loads(fh.read())
with open(node_definition.inputs["b"], "rb") as fh:
b: int = json.loads(fh.read())

c = itimes(a, b)

with open(node_definition.outputs["value"], "w+") as fh:
fh.write(json.dumps(c))

elif node_definition.function_name == "igt":
with open(node_definition.inputs["a"], "rb") as fh:
a: int = json.loads(fh.read())
with open(node_definition.inputs["b"], "rb") as fh:
b: int = json.loads(fh.read())

c = igt(a, b)

with open(node_definition.outputs["value"], "w+") as fh:
fh.write(json.dumps(c))

elif node_definition.function_name == "and":
with open(node_definition.inputs["a"], "rb") as fh:
a = json.loads(fh.read())
with open(node_definition.inputs["b"], "rb") as fh:
b = json.loads(fh.read())

c = a and b

with open(node_definition.outputs["value"], "w+") as fh:
fh.write(json.dumps(c))

elif node_definition.function_name == "id":
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would it make sense for id to be a builtin so you can make use of the linking functionality? Similarly with copy

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess there isn't much use in id usually

Copy link
Contributor Author

@mwpb mwpb Mar 24, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes I just needed id because it is in the sample graph. I don't expect the id function to be used in real applications.

with open(node_definition.inputs["value"], "rb") as fh:
value = json.loads(fh.read())

with open(node_definition.outputs["value"], "w+") as fh:
fh.write(json.dumps(value))

else:
raise ValueError(f"function name {node_definition.function_name} not found")
node_definition.done_path.touch()


if __name__ == "__main__":
worker_definition_path = argv[1]
with open(worker_definition_path, "r") as fh:
node_definition = NodeDefinition(**json.loads(fh.read()))

run(node_definition)
7 changes: 7 additions & 0 deletions python/examples/numerical-worker/pyproject.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[project]
name = "numerical-worker"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.10, <3.13"
dependencies = []
7 changes: 7 additions & 0 deletions python/examples/numerical-worker/uv.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions python/examples/sc22_example.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,14 +30,14 @@

def runtime_client_from_args(args: list[str]) -> Optional[RuntimeClient]:
if len(args) == 0:
import sc22_worker.main
import examples.sc22_worker.main

import pytket_worker.main # type: ignore
from tierkreis.pyruntime import PyRuntime

return PyRuntime(
[
sc22_worker.main.root,
examples.sc22_worker.main.root,
pytket_worker.main.root,
]
)
Expand Down
14 changes: 3 additions & 11 deletions python/pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
name = "tierkreis"
description = "Python client and utilities for tierkreis."
readme = "README.md"
dynamic = ["version"]
version = "0.7.8"
authors = [
{ name = "Seyon Sivarajah", email = "[email protected]" },
{ name = "Lukas Heidemann", email = "[email protected]" },
Expand Down Expand Up @@ -42,17 +42,9 @@ build = ["build[uv]"]
[tool.pytest.ini_options]
markers = ["pytket"]


[build-system]
requires = ["setuptools>=65.5.0", "betterproto[compiler]==2.0.0b6"]
build-backend = "generate_protos"
backend-path = ["tierkreis/_build"]

[tool.setuptools.dynamic]
version = { attr = "tierkreis._version.__version__" }

[tool.setuptools.packages.find]
include = ["tierkreis*", "tierkreis/py.typed"]
requires = ["hatchling"]
build-backend = "hatchling.build"


[tool.ruff]
Expand Down
Empty file added python/tests/__init__.py
Empty file.
2 changes: 1 addition & 1 deletion python/tests/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,8 @@

import pytest
from grpclib.client import Channel
from test_worker import main

from tests.test_worker import main
from tierkreis.builder import Namespace
from tierkreis.client.runtime_client import RuntimeClient
from tierkreis.client.server_client import ServerRuntime
Expand Down
42 changes: 42 additions & 0 deletions python/tests/controller/test_models.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import pytest

from tierkreis.controller.models import NodeLocation

node_location_1 = NodeLocation(location=[])
node_location_1 = node_location_1.append_node(0)
node_location_1 = node_location_1.append_loop(0)
node_location_1 = node_location_1.append_node(3)
node_location_1 = node_location_1.append_loop(2)
node_location_1 = node_location_1.append_node(0)
node_location_1 = node_location_1.append_map(8)
node_location_1 = node_location_1.append_node(0)


node_location_2 = NodeLocation(location=[])
node_location_2 = node_location_2.append_node(0)
node_location_2 = node_location_2.append_loop(0)
node_location_2 = node_location_2.append_node(3)
node_location_2 = node_location_2.append_node(8)
node_location_2 = node_location_2.append_node(0)

node_location_3 = NodeLocation(location=[])
node_location_3 = node_location_3.append_node(0)

node_location_4 = NodeLocation(location=[])


@pytest.mark.parametrize(
["node_location", "loc_str"],
[
(node_location_1, "N0.L0.N3.L2.N0.M8.N0"),
(node_location_2, "N0.L0.N3.N8.N0"),
(node_location_3, "N0"),
(node_location_4, ""),
],
)
def test_to_from_str(node_location: NodeLocation, loc_str: str):
node_location_str = str(node_location)
assert node_location_str == loc_str

new_loc = NodeLocation.from_str(node_location_str)
assert new_loc == node_location
29 changes: 29 additions & 0 deletions python/tests/controller/test_resume.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import json
from pathlib import Path
from uuid import UUID

from tests.sample_graph import sample_graph_without_match
from tierkreis.controller import run_graph
from tierkreis.controller.executor.shell_executor import ShellExecutor
from tierkreis.controller.models import NodeLocation
from tierkreis.controller.storage.filestorage import ControllerFileStorage
from tierkreis.core import Labels


def test_resume_sample_graph():
g = sample_graph_without_match()
storage = ControllerFileStorage(UUID(int=0))
executor = ShellExecutor(
Path("./python/examples/launchers"), logs_path=storage.logs_path
)
inputs = {
"inp": json.dumps(4).encode(),
Labels.VALUE: json.dumps(2).encode(),
Labels.THUNK: g.to_proto().SerializeToString(),
}

storage.clean_graph_files()
run_graph(storage, executor, g, inputs)

c = storage.read_output(NodeLocation(location=[]), "loop_out")
assert c == b"6"
65 changes: 65 additions & 0 deletions python/tests/sample_graph.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,3 +66,68 @@ def sample_graph() -> TierkreisGraph:
)["value"],
)
return tg


def sample_graph_without_match() -> TierkreisGraph:
def loop_graph() -> TierkreisGraph:
ifg = TierkreisGraph()
ifg.set_outputs(value=ifg.add_tag(Labels.BREAK, value=ifg.input["x"]))

elg = TierkreisGraph()
elg.set_outputs(
value=elg.add_tag(
Labels.CONTINUE,
value=elg.add_func(
"numerical-worker.iadd", a=elg.input["x"], b=elg.add_const(1)
),
)
)

tg = TierkreisGraph()
v1, v2 = tg.copy_value(tg.input["value"])
tg.set_outputs(
value=tg.add_func(
"eval",
thunk=tg.add_func(
"switch",
pred=tg.add_func("numerical-worker.igt", a=v1, b=tg.add_const(5)),
if_true=tg.add_const(ifg),
if_false=tg.add_const(elg),
),
x=v2,
)["value"]
)
return tg

one_graph = TierkreisGraph()
one_graph.set_outputs(
value=one_graph.add_func(
"numerical-worker.iadd",
a=one_graph.input["value"],
b=one_graph.input["other"],
)
)
many_graph = TierkreisGraph()
many_graph.discard(many_graph.input["other"])
many_graph.set_outputs(
value=many_graph.add_func(
"numerical-worker.id", value=many_graph.input["value"]
)
)

tg = TierkreisGraph()
tg.set_outputs(
out=tg.input["inp"],
b=tg.add_func("numerical-worker.iadd", a=tg.add_const(1), b=tg.add_const(3)),
tag=tg.add_tag("boo", value=tg.add_const("world")),
add=tg.add_func(
"numerical-worker.iadd", a=tg.add_const(23), b=tg.add_const(123)
),
_and=tg.add_func(
"numerical-worker.and", a=tg.add_const(True), b=tg.add_const(False)
),
loop_out=tg.add_func(
"loop", body=tg.add_const(loop_graph()), value=tg.add_const(2)
)["value"],
)
return tg
2 changes: 1 addition & 1 deletion python/tests/test_frontend.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@
from typing import Any, Dict, List, Optional, Tuple, Type

import pytest
from utils import nint_adder

from tests.utils import nint_adder
from tierkreis import TierkreisGraph
from tierkreis.client import RuntimeClient
from tierkreis.core.graphviz import _merge_copies
Expand Down
4 changes: 2 additions & 2 deletions python/tests/test_generics.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,11 @@

import pydantic as pyd
import pytest
from test_worker.main import (

from tests.test_worker.main import (
MyGeneric,
MyGenericList,
)

from tierkreis.builder import Const, Output, graph
from tierkreis.client.runtime_client import RuntimeClient
from tierkreis.core._internal import generic_origin
Expand Down
2 changes: 1 addition & 1 deletion python/tests/test_pyruntime.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import pytest
from sample_graph import sample_graph as sample_g

from tests.sample_graph import sample_graph as sample_g
from tierkreis.core.tierkreis_graph import TierkreisEdge, TierkreisGraph
from tierkreis.core.values import TierkreisValue, VariantValue
from tierkreis.pyruntime import PyRuntime
Expand Down
2 changes: 1 addition & 1 deletion python/tests/test_variants.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@

import pydantic as pyd
import pytest
from test_worker.main import EmptyStruct, IntStruct, MyGeneric

from tests.test_worker.main import EmptyStruct, IntStruct, MyGeneric
from tierkreis.builder import Const, Output, UnionConst, graph
from tierkreis.client.runtime_client import RuntimeClient
from tierkreis.core.opaque_model import OpaqueModel
Expand Down
8 changes: 5 additions & 3 deletions python/tierkreis/_build/generate_protos.py
Original file line number Diff line number Diff line change
@@ -1,13 +1,14 @@
import glob
import os
from pathlib import Path
import subprocess

from setuptools import build_meta as default_backend # type : ignore

# PEP 517 / 518 local (aka in-tree) backend for pip/build/etc.
# We expect to be run from the project root, i.e. python/
PROTO_INPUT_DIR = "./protos"
PROTO_OUTPUT_DIR = "./tierkreis/core/protos"
tierkreis_python_dir = Path(__file__).parent.parent.parent
PROTO_INPUT_DIR = str(tierkreis_python_dir / "protos")
PROTO_OUTPUT_DIR = str(tierkreis_python_dir / "tierkreis/core/protos")


def generate_proto_code():
Expand Down Expand Up @@ -55,6 +56,7 @@ def build_wheel(*args, **kwargs):

prepare_metadata_for_build_wheel = default_backend.prepare_metadata_for_build_wheel


# PEP 660 defines additional hooks to support "pip install --editable":
def build_editable(*args, **kwargs):
generate_proto_code()
Expand Down
Loading