Skip to content

Commit 072b5ea

Browse files
authored
Stripe ai sdk example (#93)
* stripe ai with langchain * indentation fix * open ai key to model * services and client * update readme * create paymentlink * more checks * typo
1 parent d6446ea commit 072b5ea

12 files changed

+330
-0
lines changed

email_sender/README.md

+80
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
# Restack AI - Email Sender example
2+
3+
This example showcases how to send emails with a Restack workflow using the sendgrid api. You can easily choose another email provider and update the code.
4+
You can schedule two scenarios of the workflow.
5+
6+
1. It will be successfull and send an email.
7+
2. The email content generation step will fail once to showcase how Restack handles retries automatically. Once failure is caught, step will be retry automatically and rest of workflow will be executed as expected and email will be sent.
8+
9+
10+
## Prerequisites
11+
12+
- Python 3.10 or higher
13+
- Poetry (for dependency management)
14+
- Docker (for running the Restack services)
15+
16+
## Usage
17+
18+
1. Run Restack local engine with Docker:
19+
20+
```bash
21+
docker run -d --pull always --name restack -p 5233:5233 -p 6233:6233 -p 7233:7233 ghcr.io/restackio/restack:main
22+
```
23+
24+
2. Open the web UI to see the workflows:
25+
26+
```bash
27+
http://localhost:5233
28+
```
29+
30+
3. Clone this repository:
31+
32+
```bash
33+
git clone https://github.com/restackio/examples-python
34+
cd examples-python/examples/get-started
35+
```
36+
37+
4. Create .env file with: STRIPE_SECRET_KEY and OPENAI_API_KEY
38+
39+
4. Install dependencies using Poetry:
40+
41+
```bash
42+
poetry env use 3.12
43+
```
44+
45+
```bash
46+
poetry shell
47+
```
48+
49+
```bash
50+
poetry install
51+
```
52+
53+
```bash
54+
poetry env info # Optional: copy the interpreter path to use in your IDE (e.g. Cursor, VSCode, etc.)
55+
```
56+
57+
5. Run the services:
58+
59+
```bash
60+
poetry run services
61+
```
62+
63+
This will start the Restack service with the defined workflows and functions.
64+
65+
6. In a new terminal, schedule the workflow:
66+
67+
```bash
68+
poetry shell
69+
```
70+
71+
```bash
72+
poetry run schedule
73+
```
74+
75+
This will schedule the `SendEmailWorkflow` and print the result.
76+
77+
7. To simulate a flow where the step for sending email fails and the retry is automatically handled by Restack AI use run:
78+
```bash
79+
poetry run schedule_failure
80+
```

stripe-ai/.env.example

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
STRIPE_SECRET_KEY=your-stripe-secret-key
2+
OPENAI_API_KEY=your-openai-api-key
3+
LANGCHAIN_API_KEY=your-langchain-api-key
4+
LANGCHAIN_TRACING_V2=false

stripe-ai/README.md

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
# Restack AI - Stripe Ai example
2+
3+
This repository contains a an example on how restack can use langchain and the stripe ai sdk to create a product with a price and also create a payment link for it.
4+
5+
## Prerequisites
6+
7+
- Python 3.10 or higher
8+
- Poetry (for dependency management)
9+
- Docker (for running the Restack services)
10+
11+
## Usage
12+
13+
1. Run Restack local engine with Docker:
14+
15+
```bash
16+
docker run -d --pull always --name restack -p 5233:5233 -p 6233:6233 -p 7233:7233 ghcr.io/restackio/restack:main
17+
```
18+
19+
2. Open the web UI to see the workflows:
20+
21+
```bash
22+
http://localhost:5233
23+
```
24+
25+
3. Clone this repository:
26+
27+
```bash
28+
git clone https://github.com/restackio/examples-python
29+
cd examples-python/examples/get-started
30+
```
31+
32+
4. Create .env file with: STRIPE_SECRET_KEY, LANGCHAIN_API_KEY and OPENAI_API_KEY
33+
34+
4. Install dependencies using Poetry:
35+
36+
```bash
37+
poetry env use 3.12
38+
```
39+
40+
```bash
41+
poetry shell
42+
```
43+
44+
```bash
45+
poetry install
46+
```
47+
48+
```bash
49+
poetry env info # Optional: copy the interpreter path to use in your IDE (e.g. Cursor, VSCode, etc.)
50+
```
51+
52+
5. Run the services:
53+
54+
```bash
55+
poetry run services
56+
```
57+
58+
This will start the Restack service with the defined workflows and functions.
59+
60+
6. In a new terminal, schedule the workflow:
61+
62+
```bash
63+
poetry shell
64+
```
65+
66+
```bash
67+
poetry run schedule
68+
```
69+
70+
This will schedule the `CreatePaymentLinkWorkflow` and print the result.

stripe-ai/pyproject.toml

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
[tool.poetry]
2+
name = "stripe-ai"
3+
version = "0.0.1"
4+
description = "Send emails with sendgrid"
5+
authors = [
6+
"Restack Team <[email protected]>",
7+
]
8+
readme = "README.md"
9+
packages = [{include = "src"}]
10+
11+
[tool.poetry.dependencies]
12+
python = ">=3.10,<4.0"
13+
pydantic = "2.9.2"
14+
python-dotenv = "1.0.1"
15+
openai = "1.56.2"
16+
restack-ai = "0.0.42"
17+
stripe-agent-toolkit = "^0.1.21"
18+
langchain = "^0.3.9"
19+
stripe = "^11.3.0"
20+
langchain-openai = "^0.2.11"
21+
22+
[build-system]
23+
requires = ["poetry-core"]
24+
build-backend = "poetry.core.masonry.api"
25+
26+
[tool.poetry.scripts]
27+
services = "src.services:run_services"
28+
schedule = "schedule_workflow:run_schedule_workflow"

stripe-ai/schedule_workflow.py

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
import asyncio
2+
import time
3+
from restack_ai import Restack
4+
from dotenv import load_dotenv
5+
6+
load_dotenv()
7+
8+
async def main():
9+
client = Restack()
10+
11+
workflow_id = f"{int(time.time() * 1000)}-CreatePaymentLinkWorkflow"
12+
13+
run_id = await client.schedule_workflow(
14+
workflow_name="CreatePaymentLinkWorkflow",
15+
workflow_id=workflow_id,
16+
)
17+
18+
result = await client.get_workflow_result(
19+
workflow_id=workflow_id,
20+
run_id=run_id
21+
)
22+
23+
print(result)
24+
25+
exit(0)
26+
27+
def run_schedule_workflow():
28+
asyncio.run(main())
29+
30+
if __name__ == "__main__":
31+
run_schedule_workflow()

stripe-ai/src/__init__.py

Whitespace-only changes.

stripe-ai/src/client.py

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import os
2+
from restack_ai import Restack
3+
from restack_ai.restack import CloudConnectionOptions
4+
from dotenv import load_dotenv
5+
6+
# Load environment variables from a .env file
7+
load_dotenv()
8+
9+
10+
engine_id = os.getenv("RESTACK_ENGINE_ID")
11+
address = os.getenv("RESTACK_ENGINE_ADDRESS")
12+
api_key = os.getenv("RESTACK_ENGINE_API_KEY")
13+
14+
connection_options = CloudConnectionOptions(
15+
engine_id=engine_id,
16+
address=address,
17+
api_key=api_key
18+
)
19+
client = Restack(connection_options)

stripe-ai/src/functions/__init__.py

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
from langchain.agents import AgentExecutor, create_structured_chat_agent
2+
from langchain import hub
3+
from stripe_agent_toolkit.langchain.toolkit import StripeAgentToolkit
4+
from restack_ai.function import function, FunctionFailure
5+
from langchain_openai import ChatOpenAI
6+
7+
import os
8+
from dotenv import load_dotenv
9+
from pydantic import SecretStr
10+
11+
load_dotenv()
12+
13+
@function.defn()
14+
async def create_payment_link():
15+
stripe_secret_key = os.getenv("STRIPE_SECRET_KEY")
16+
openai_api_key = os.getenv("OPENAI_API_KEY")
17+
langchain_api_key = os.getenv("LANGCHAIN_API_KEY")
18+
19+
if stripe_secret_key is None:
20+
raise FunctionFailure("STRIPE_SECRET_KEY is not set", non_retryable=True)
21+
22+
if langchain_api_key is None:
23+
raise FunctionFailure("LANGCHAIN_API_KEY is not set", non_retryable=True)
24+
25+
if openai_api_key is None:
26+
raise FunctionFailure("OPENAI_API_KEY is not set", non_retryable=True)
27+
28+
try:
29+
stripe_agent_toolkit = StripeAgentToolkit(
30+
secret_key=stripe_secret_key,
31+
configuration={
32+
"actions": {
33+
"payment_links": {
34+
"create": True,
35+
},
36+
"products": {
37+
"create": True,
38+
},
39+
"prices": {
40+
"create": True,
41+
},
42+
}
43+
},
44+
)
45+
46+
model = ChatOpenAI(api_key=SecretStr(openai_api_key))
47+
48+
prompt = hub.pull("hwchase17/structured-chat-agent")
49+
50+
agent = create_structured_chat_agent(model, stripe_agent_toolkit.get_tools(), prompt)
51+
agent_executor = AgentExecutor(agent=agent, tools=stripe_agent_toolkit.get_tools())
52+
53+
result = agent_executor.invoke({
54+
"input": "Create a payment link for a new product called \"Test\" with a price of $100."
55+
})
56+
57+
return result["output"]
58+
except Exception as e:
59+
raise FunctionFailure(f"Error creating payment link: {e}", non_retryable=True)

stripe-ai/src/services.py

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
import asyncio
2+
from src.client import client
3+
from src.workflows.payment_link import CreatePaymentLinkWorkflow
4+
from src.functions.create_payment_link import create_payment_link
5+
6+
async def main():
7+
await asyncio.gather(
8+
client.start_service(
9+
workflows=[CreatePaymentLinkWorkflow],
10+
functions=[create_payment_link]
11+
)
12+
)
13+
14+
def run_services():
15+
asyncio.run(main())
16+
17+
if __name__ == "__main__":
18+
run_services()

stripe-ai/src/workflows/__init__.py

Whitespace-only changes.
+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
from restack_ai.workflow import workflow, import_functions, log, RetryPolicy
2+
from datetime import timedelta
3+
4+
with import_functions():
5+
from src.functions.create_payment_link import create_payment_link
6+
7+
@workflow.defn()
8+
class CreatePaymentLinkWorkflow:
9+
@workflow.run
10+
async def run(self):
11+
log.info("CreatePaymentLinkWorkflow started", input=input)
12+
13+
result = await workflow.step(
14+
create_payment_link,
15+
retry_policy=RetryPolicy(
16+
initial_interval=timedelta(seconds=10),
17+
backoff_coefficient=1,
18+
),
19+
)
20+
21+
return result

0 commit comments

Comments
 (0)