Skip to main content

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


Download files

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

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distribution

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

python_langgraph_equations-0.0.3-py3-none-any.whl (7.3 kB view details)

Uploaded Python 3

File details

Details for the file python_langgraph_equations-0.0.3-py3-none-any.whl.

File metadata

File hashes

Hashes for python_langgraph_equations-0.0.3-py3-none-any.whl
Algorithm Hash digest
SHA256 5f210e8c924fdd68b40035e678f539ea9b27f6f4d43a0b84fa3809147334efda
MD5 0520eff4a31b4c99529b48e4b0eca458
BLAKE2b-256 18f6de1448e6c8cac3b6da98ebf2fcda1e229777c7cdf8df9ea2cef255fce224

See more details on using hashes here.

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