mirror of
https://github.com/Significant-Gravitas/AutoGPT.git
synced 2026-02-11 15:25:16 -05:00
The Library Agent credentials UX (#9789) currently doesn't work for sub-graphs. ### Changes 🏗️ - Include sub-graphs in generating `Graph.credentials_input_schema` - Propagate `node_credentials_input_map` into `AgentExecutionBlock` executions - Fix: also apply `node_credentials_input_map` in `_enqueue_next_nodes` ### Checklist 📋 #### For code changes: - [x] I have clearly listed my changes in the PR description - [x] I have made a test plan - [x] I have tested my changes according to the test plan: - Import a graph with sub-graphs that need credentials - Run this agent from the Library - [x] -> Should work
142 lines
4.6 KiB
Python
142 lines
4.6 KiB
Python
from datetime import datetime, timezone
|
|
|
|
import pytest
|
|
from prisma.enums import CreditTransactionType
|
|
from prisma.models import CreditTransaction
|
|
|
|
from backend.blocks.llm import AITextGeneratorBlock
|
|
from backend.data.block import get_block
|
|
from backend.data.credit import BetaUserCredit, UsageTransactionMetadata
|
|
from backend.data.execution import NodeExecutionEntry
|
|
from backend.data.user import DEFAULT_USER_ID
|
|
from backend.executor.utils import block_usage_cost
|
|
from backend.integrations.credentials_store import openai_credentials
|
|
from backend.util.test import SpinTestServer
|
|
|
|
REFILL_VALUE = 1000
|
|
user_credit = BetaUserCredit(REFILL_VALUE)
|
|
|
|
|
|
async def disable_test_user_transactions():
|
|
await CreditTransaction.prisma().delete_many(where={"userId": DEFAULT_USER_ID})
|
|
|
|
|
|
async def top_up(amount: int):
|
|
await user_credit._add_transaction(
|
|
DEFAULT_USER_ID,
|
|
amount,
|
|
CreditTransactionType.TOP_UP,
|
|
)
|
|
|
|
|
|
async def spend_credits(entry: NodeExecutionEntry) -> int:
|
|
block = get_block(entry.block_id)
|
|
if not block:
|
|
raise RuntimeError(f"Block {entry.block_id} not found")
|
|
|
|
cost, matching_filter = block_usage_cost(block=block, input_data=entry.inputs)
|
|
await user_credit.spend_credits(
|
|
entry.user_id,
|
|
cost,
|
|
UsageTransactionMetadata(
|
|
graph_exec_id=entry.graph_exec_id,
|
|
graph_id=entry.graph_id,
|
|
node_id=entry.node_id,
|
|
node_exec_id=entry.node_exec_id,
|
|
block_id=entry.block_id,
|
|
block=entry.block_id,
|
|
input=matching_filter,
|
|
reason=f"Ran block {entry.block_id} {block.name}",
|
|
),
|
|
)
|
|
|
|
return cost
|
|
|
|
|
|
@pytest.mark.asyncio(loop_scope="session")
|
|
async def test_block_credit_usage(server: SpinTestServer):
|
|
await disable_test_user_transactions()
|
|
await top_up(100)
|
|
current_credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
|
|
spending_amount_1 = await spend_credits(
|
|
NodeExecutionEntry(
|
|
user_id=DEFAULT_USER_ID,
|
|
graph_id="test_graph",
|
|
node_id="test_node",
|
|
graph_exec_id="test_graph_exec",
|
|
node_exec_id="test_node_exec",
|
|
block_id=AITextGeneratorBlock().id,
|
|
inputs={
|
|
"model": "gpt-4-turbo",
|
|
"credentials": {
|
|
"id": openai_credentials.id,
|
|
"provider": openai_credentials.provider,
|
|
"type": openai_credentials.type,
|
|
},
|
|
},
|
|
),
|
|
)
|
|
assert spending_amount_1 > 0
|
|
|
|
spending_amount_2 = await spend_credits(
|
|
NodeExecutionEntry(
|
|
user_id=DEFAULT_USER_ID,
|
|
graph_id="test_graph",
|
|
node_id="test_node",
|
|
graph_exec_id="test_graph_exec",
|
|
node_exec_id="test_node_exec",
|
|
block_id=AITextGeneratorBlock().id,
|
|
inputs={"model": "gpt-4-turbo", "api_key": "owned_api_key"},
|
|
),
|
|
)
|
|
assert spending_amount_2 == 0
|
|
|
|
new_credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
assert new_credit == current_credit - spending_amount_1 - spending_amount_2
|
|
|
|
|
|
@pytest.mark.asyncio(loop_scope="session")
|
|
async def test_block_credit_top_up(server: SpinTestServer):
|
|
await disable_test_user_transactions()
|
|
current_credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
|
|
await top_up(100)
|
|
|
|
new_credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
assert new_credit == current_credit + 100
|
|
|
|
|
|
@pytest.mark.asyncio(loop_scope="session")
|
|
async def test_block_credit_reset(server: SpinTestServer):
|
|
await disable_test_user_transactions()
|
|
month1 = 1
|
|
month2 = 2
|
|
|
|
# set the calendar to month 2 but use current time from now
|
|
user_credit.time_now = lambda: datetime.now(timezone.utc).replace(
|
|
month=month2, day=1
|
|
)
|
|
month2credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
|
|
# Month 1 result should only affect month 1
|
|
user_credit.time_now = lambda: datetime.now(timezone.utc).replace(
|
|
month=month1, day=1
|
|
)
|
|
month1credit = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
await top_up(100)
|
|
assert await user_credit.get_credits(DEFAULT_USER_ID) == month1credit + 100
|
|
|
|
# Month 2 balance is unaffected
|
|
user_credit.time_now = lambda: datetime.now(timezone.utc).replace(
|
|
month=month2, day=1
|
|
)
|
|
assert await user_credit.get_credits(DEFAULT_USER_ID) == month2credit
|
|
|
|
|
|
@pytest.mark.asyncio(loop_scope="session")
|
|
async def test_credit_refill(server: SpinTestServer):
|
|
await disable_test_user_transactions()
|
|
balance = await user_credit.get_credits(DEFAULT_USER_ID)
|
|
assert balance == REFILL_VALUE
|