python-langgraph-equations
Project description
python-langgraph-equations
This library provides integration between LangGraph and Category Equations, providing a set of primitives to construct, evaluate and simplify workflows.
For about python-category-equations see: https://github.com/kummahiih/python-category-equations
get_primitives(graph: StateGraph, debug: bool = False)
Function get_primitives returns a tuple (I, O, C)
that represents the set of equation construction primitives
for the given langraph StateGraph. The connect_edge function
is used to add edges to the graph, optionally printing them if debug is True.
The created primitives can be used to manipulate workflows as equations and to optimize and compare their structure.
Usage example:
Set up a StateGraph and nodes representing various text processing steps:
>>> from langgraph.graph import StateGraph, END, START
>>> from typing import TypedDict, List
>>> from langgraph_equations import get_primitives
>>> class State(TypedDict):
... text: str
... summary: str
... category: str
... embedding: List[float]
>>> def process_input(state: State):
... return {"text": state["text"].strip()}
>>> def summarize(state: State):
... text = state["text"]
... return {"summary": f"Summary of: {text[:30]}..."}
>>> def classify(state: State):
... text = state["text"]
... category = "long" if len(text) > 50 else "short"
... return {"category": category}
>>> def embed(state: State):
... # Fake embedding for demonstration
... return {"embedding": [len(state["text"]), 1.0, 0.5]}
>>> def combine_results(state: State):
... return state
>>> graph = StateGraph(State)
>>> graph = graph.add_node("process_input", process_input)
>>> graph = graph.add_node("summarize", summarize)
>>> graph = graph.add_node("classify", classify)
>>> graph = graph.add_node("embed", embed)
>>> graph = graph.add_node("combine_results", combine_results)
Now we can use the get_primitives function to obtain the equation primitives and construct a workflow
>>> I, O, C = get_primitives(graph, debug=True)
>>> term = C(START) * C("process_input") * C("summarize", "classify", "embed") * C("combine_results") * C(END)
On calling evaluate, the term will print the edges (debug=True) and add the edges to the graph:
>>> term.evaluate()
__start__ → process_input
classify → combine_results
combine_results → __end__
embed → combine_results
process_input → classify
process_input → embed
process_input → summarize
summarize → combine_results
Now we can compile and invoke the graph with an input state:
>>> app = graph.compile()
>>> app.invoke({"text": "LangGraph makes parallel workflows easy!"})
{'text': 'LangGraph makes parallel workflows easy!', 'summary': 'Summary of: LangGraph makes parallel workf...', 'category': 'short', 'embedding': [40, 1.0, 0.5]}
How to use category_equations.simplify with LangGraph
Lets setup the workflow graph first:
>>> from langgraph.graph import StateGraph, END, START
>>> from typing import TypedDict, List
>>> from langgraph_equations import get_primitives
>>> from category_equations import simplify, EquationMap
>>> class State(TypedDict):
... text: str
... summary: str
... category: str
... embedding: List[float]
>>> def process_input(state: State):
... return {"text": state["text"].strip()}
>>> def summarize(state: State):
... text = state["text"]
... return {"summary": f"Summary of: {text[:30]}..."}
>>> def classify(state: State):
... text = state["text"]
... category = "long" if len(text) > 50 else "short"
... return {"category": category}
>>> def embed(state: State):
... # Fake embedding for demonstration
... return {"embedding": [len(state["text"]), 1.0, 0.5]}
>>> def combine_results(state: State):
... return state
>>> graph = StateGraph(State)
>>> graph = graph.add_node("process_input", process_input)
>>> graph = graph.add_node("summarize", summarize)
>>> graph = graph.add_node("classify", classify)
>>> graph = graph.add_node("embed", embed)
>>> graph = graph.add_node("combine_results", combine_results)
Now we can use the get_primitives function to obtain the equation primitives and construct a workflow in a way
that can be simplified with category_equations.simplify:
>>> I, O, C = get_primitives(graph, debug=True)
>>> term = (
... O * C(START) * C("process_input") * O +
... O * C("process_input") * C("summarize") * O +
... O * C("process_input") * C("classify") * O +
... O * C("process_input") * C("embed") * O +
... O * C("summarize") * C("combine_results") * O +
... O * C("classify") * C("combine_results") * O +
... O * C("embed") * C("combine_results") * O +
... O * C("combine_results") * C(END) * O
... )
Now we can use the category_equations.simplify function to simplify the workflow.
First we need to create an EquationMap where the search of simplifications will be done:
>>> m = EquationMap(I, O, C)
Then we can try to simplify the term:
>>> max_depth = 2000
>>> simplified_term, path = simplify(term, max_depth, m)
Because the equation solver quality is questionable, it is always a good idea to test the resulting simplified term to see if it is equivalent to the original term.
>>> term == simplified_term
True
All seems to be good, so lets see how the simplified term looks like:
>>> simplified_term
(O * C(__start__) * C(process_input) * C(summarize) * O + O * C(process_input) * C(classify) * C(combine_results) * O) + O * C(process_input) * C(embed) * C(combine_results) * C(__end__) * O
To see the steps taken during simplification, we can iterate over the paths, but we skip that because it is boring.
>>> len(path)
40
It is pretty easy perform better in simplifying than this, which indicates that the equation solver is not good enough. You can however use this to test your own terms and see if you did any mistakes in constructing them:
>>> simplified_manually = O * C(START) * C("process_input") * C("summarize", "classify", "embed") * C("combine_results") * C(END) * O
>>> term == simplified_manually
True
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 Distributions
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 python_langgraph_equations-0.0.3-py3-none-any.whl.
File metadata
- Download URL: python_langgraph_equations-0.0.3-py3-none-any.whl
- Upload date:
- Size: 7.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5f210e8c924fdd68b40035e678f539ea9b27f6f4d43a0b84fa3809147334efda
|
|
| MD5 |
0520eff4a31b4c99529b48e4b0eca458
|
|
| BLAKE2b-256 |
18f6de1448e6c8cac3b6da98ebf2fcda1e229777c7cdf8df9ea2cef255fce224
|