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

Issue using API Request in Tool mode from Agent #5325

Open
steveh250 opened this issue Dec 18, 2024 · 3 comments
Open

Issue using API Request in Tool mode from Agent #5325

steveh250 opened this issue Dec 18, 2024 · 3 comments
Labels
enhancement New feature or request

Comments

@steveh250
Copy link

steveh250 commented Dec 18, 2024

Feature Request

When using API Request as a Tool connected to an Agent the API Request component wasn't able to process a cURL request that worked when the API Request was not in Tool mode (grabbed the formatted cURL request from a LangFuse trace to make sure I was testing the exact same request.

Motivation

Fixing the API Request to work as in Tool mode.

Your Contribution

Here is the complete code for the modified API Request that works when running Tool mode. I have not tested this code in non-tool mode.

import asyncio
import json
from typing import Any
from urllib.parse import parse_qsl, urlencode, urlparse, urlunparse

import httpx
from loguru import logger

from langflow.base.curl.parse import parse_context
from langflow.custom import Component
from langflow.io import DataInput, DropdownInput, IntInput, MessageTextInput, NestedDictInput, Output
from langflow.schema import Data
from langflow.schema.dotdict import dotdict


class APIRequestComponent(Component):
    display_name = "API Request - Fixed"
    description = (
        "This component allows you to make HTTP requests to one or more URLs. "
        "You can provide headers and body as either dictionaries or Data objects. "
        "Additionally, you can append query parameters to the URLs.\n\n"
        "**Note:** Check advanced options for more settings."
    )
    icon = "Globe"
    name = "APIRequest"

    inputs = [
        MessageTextInput(
            name="urls",
            display_name="URLs",
            is_list=True,
            info="Enter one or more URLs, separated by commas.",
        ),
        MessageTextInput(
            name="curl",
            display_name="cURL",
            info="Paste a curl command to populate the fields. "
            "This will fill in the dictionary fields for headers and body.",
            advanced=False,
            refresh_button=True,
            real_time_refresh=True,
            tool_mode=True,
        ),
        DropdownInput(
            name="method",
            display_name="Method",
            options=["GET", "POST", "PATCH", "PUT"],
            value="GET",
            info="The HTTP method to use (GET, POST, PATCH, PUT).",
        ),
        NestedDictInput(
            name="headers",
            display_name="Headers",
            info="The headers to send with the request as a dictionary. This is populated when using the CURL field.",
            input_types=["Data"],
        ),
        NestedDictInput(
            name="body",
            display_name="Body",
            info="The body to send with the request as a dictionary (for POST, PATCH, PUT). "
            "This is populated when using the CURL field.",
            input_types=["Data"],
        ),
        DataInput(
            name="query_params",
            display_name="Query Parameters",
            info="The query parameters to append to the URL.",
            tool_mode=True,
        ),
        IntInput(
            name="timeout",
            display_name="Timeout",
            value=5,
            info="The timeout to use for the request.",
        ),
    ]

    outputs = [
        Output(display_name="Data", name="data", method="make_requests"),
    ]

    def parse_curl(self, curl: str, build_config: dotdict) -> dotdict:
        try:
            parsed = parse_context(curl)
            build_config["urls"]["value"] = [parsed.url]
            build_config["method"]["value"] = parsed.method.upper()
            build_config["headers"]["value"] = dict(parsed.headers)

            if parsed.data:
                try:
                    json_data = json.loads(parsed.data)
                    build_config["body"]["value"] = json_data
                except json.JSONDecodeError:
                    logger.exception("Error decoding JSON data")
            else:
                build_config["body"]["value"] = {}
        except Exception as exc:
            msg = f"Error parsing curl: {exc}"
            logger.exception(msg)
            raise ValueError(msg) from exc
        return build_config

    def update_build_config(self, build_config: dotdict, field_value: Any, field_name: str | None = None):
        if field_name == "curl" and field_value:
            build_config = self.parse_curl(field_value, build_config)
        return build_config

    async def make_request(
        self,
        client: httpx.AsyncClient,
        method: str,
        url: str,
        headers: dict | None = None,
        body: dict | None = None,
        timeout: int = 5,
    ) -> Data:
        method = method.upper()
        if method not in {"GET", "POST", "PATCH", "PUT", "DELETE"}:
            msg = f"Unsupported method: {method}"
            raise ValueError(msg)

        if isinstance(body, str) and body:
            try:
                body = json.loads(body)
            except Exception as e:
                msg = f"Error decoding JSON data: {e}"
                logger.exception(msg)
                body = None
                raise ValueError(msg) from e

        data = body or None

        try:
            response = await client.request(method, url, headers=headers, json=data, timeout=timeout)
            try:
                result = response.json()
            except Exception:  # noqa: BLE001
                logger.opt(exception=True).debug("Error decoding JSON response")
                result = response.text
            return Data(
                data={
                    "source": url,
                    "headers": headers,
                    "status_code": response.status_code,
                    "result": result,
                },
            )
        except httpx.TimeoutException:
            return Data(
                data={
                    "source": url,
                    "headers": headers,
                    "status_code": 408,
                    "error": "Request timed out",
                },
            )
        except Exception as exc:  # noqa: BLE001
            logger.opt(exception=True).debug(f"Error making request to {url}")
            return Data(
                data={
                    "source": url,
                    "headers": headers,
                    "status_code": 500,
                    "error": str(exc),
                },
            )

    def add_query_params(self, url: str, params: dict) -> str:
        url_parts = list(urlparse(url))
        query = dict(parse_qsl(url_parts[4]))
        query.update(params)
        url_parts[4] = urlencode(query)
        return urlunparse(url_parts)

    
    async def make_requests(self) -> list[Data]:
        # Parse curl first if present
        curl = self.curl
        if curl:
            try:
                parsed = parse_context(curl)
                # Directly set the values instead of using build_config
                method = parsed.method.upper()
                urls = [parsed.url] if parsed.url else []
                headers = dict(parsed.headers) if parsed.headers else {}
                
                if parsed.data:
                    try:
                        body = json.loads(parsed.data)
                    except json.JSONDecodeError as e:
                        logger.error(f"Error decoding JSON data: {e}")
                        body = {}
                else:
                    body = {}
            except Exception as exc:
                logger.exception(f"Error parsing curl: {exc}")
                # Fall back to regular parameters if curl parsing fails
                method = self.method
                urls = [url.strip() for url in self.urls if url.strip()]
                headers = self.headers or {}
                body = self.body or {}
        else:
            # Use regular parameters if no curl
            method = self.method
            urls = [url.strip() for url in self.urls if url.strip()]
            headers = self.headers or {}
            body = self.body or {}
    
        timeout = self.timeout
    
        # Handle query parameters
        if isinstance(self.query_params, str):
            query_params = dict(parse_qsl(self.query_params))
        else:
            query_params = self.query_params.data if self.query_params else {}
    
        # Handle Data objects
        if isinstance(headers, Data):
            headers = headers.data
    
        if isinstance(body, Data):
            body = body.data
    
        bodies = [body] * len(urls)
        urls = [self.add_query_params(url, query_params) for url in urls]
    
        async with httpx.AsyncClient() as client:
            results = await asyncio.gather(
                *[
                    self.make_request(client, method, u, headers, rec, timeout)
                    for u, rec in zip(urls, bodies, strict=True)
                ]
            )
        self.status = results
        return results
    
    def update_build_config(self, build_config: dotdict, field_value: Any, field_name: str | None = None):
        if field_name == "curl" and field_value:
            try:
                parsed = parse_context(field_value)
                # Update the build_config values directly
                build_config.urls.value = [parsed.url] if parsed.url else []
                build_config.method.value = parsed.method.upper()
                build_config.headers.value = dict(parsed.headers) if parsed.headers else {}
                
                if parsed.data:
                    try:
                        json_data = json.loads(parsed.data)
                        build_config.body.value = json_data
                    except json.JSONDecodeError:
                        logger.exception("Error decoding JSON data")
                        build_config.body.value = {}
                else:
                    build_config.body.value = {}
            except Exception as exc:
                logger.exception(f"Error parsing curl: {exc}")
                # Don't modify build_config if parsing fails
                pass
        return build_config
@steveh250 steveh250 added the enhancement New feature or request label Dec 18, 2024
@edwinjosechittilappilly
Copy link
Collaborator

@carlosrcoelho We might need to look into it and later see if it could be a useful update to the component.

@edwinjosechittilappilly
Copy link
Collaborator

@steveh250 Can you also explain the issue that you have addressed?

@steveh250
Copy link
Author

@edwinjosechittilappilly - absolutely, happy to.

At a high level it's explained in the summary "When using API Request as a Tool connected to an Agent the API Request component wasn't able to process a cURL request that worked when the API Request was not in Tool mode (grabbed the formatted cURL request from a LangFuse trace to make sure I was testing the exact same request."

At a more detailed level this is what I experienced when building the following flow (a really simple flow):

Flow

I created a cURL command to call a Flask API that I have wrapped around an OS binary via the API Request component. When I paste the cURL command into the API Request component the cURL command is parsed and Method, Headers and Body are correctly extracted. Then, switch the API Request component into Tool Mode and link it to an Agent, have the Agent generate a cURL command and the Method, Headers and Body are not correctly extracted (the code I pasted works for a cURL command passed from an Agent). I have tested this new code in Tool Mode, and passed it to the user and it is worked flawlessly.

Anything else I can help explain - need more detail on anything?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

No branches or pull requests

2 participants