Tutorial: Contributing a New Tool to Gofannon

This guide will walk you through the process of contributing a new tool to the
Gofannon repository. We’ll focus on creating a simple REST API function that
doesn’t require authentication, making it ideal for first-time contributors.

For a comprehensive overview of what tools are, see the functions and tools overview

Step 1: Add an Issue to the Repository

Before starting work on your contribution, create an issue in the Gofannon repository to discuss your proposed tool with the maintainers. This ensures your contribution aligns with the project’s goals and avoids duplication of effort. Include:

Wait for feedback from the maintainers before proceeding to implementation.

Step 2: Find a Simple REST API

Once your proposal is approved, identify a simple REST API that doesn’t require authentication (requiring
authentication introduces a degree of complexity and is covered in the appendix
below). Some good examples include:

For this tutorial, we’ll use the OpenNotify ISS Location API.
Note: This has already been implemented.

If you’re having a difficult time choosing a REST API, see Appendix C below.

Step 3: Create a New Python File

Create a new Python file in the appropriate directory. For our ISS location function, we’ll create:

gofannon/open_notify_space/iss_locator.py    

Handling New APIs

If you’re working with a new API that doesn’t have an existing folder in the repository, you’ll need to:

  1. Create a new directory for your API:
gofannon/new_api_name/    
  1. Add an empty __init__.py file:
gofannon/new_api_name/__init__.py    
  1. Create your function file in the new directory:
gofannon/new_api_name/your_function.py    

Important Notes:

Example Directory Structure

For our ISS Locator example, the structure would look like:

gofannon/    
├── open_notify_space/    
│   ├── __init__.py    
│   └── iss_locator.py    

Step 4: Implement the Function

Follow this pattern to implement your function:

from ..base import BaseTool    
from ..config import FunctionRegistry    
import logging    
import requests

logger = logging.getLogger(__name__)

@FunctionRegistry.register    
class IssLocator(BaseTool):    
    def __init__(self, name="iss_locator"):    
        super().__init__()    
        self.name = name

    @property    
    def definition(self):    
        return {    
            "type": "function",    
            "function": {    
                "name": self.name,    
                "description": "Get current location of the International Space Station",    
                "parameters": {    
                    "type": "object",    
                    "properties": {},    
                    "required": []    
                }    
            }    
        }    
    
    def fn(self):    
        logger.debug("Fetching ISS location")    
        response = requests.get("http://api.open-notify.org/iss-now.json")    
        response.raise_for_status()    
        return response.json()    

Key Components Explained:

  1. BaseTool Inheritance: Your class must inherit from BaseTool to get core functionality
  2. FunctionRegistry.register: This decorator registers your function with Gofannon’s function registry
  3. definition Property: Defines the function’s interface using OpenAI’s function calling schema
  4. fn Method: Contains the actual implementation of your function

Step 5: Add Documentation

Add a docstring to your class explaining what it does:

class IssLocator(BaseTool):    
    """Get the current location of the International Space Station (ISS)

    Uses the OpenNotify API to retrieve the ISS's current latitude and longitude.    
    Returns a dictionary with the ISS's position and timestamp.    
    """    

Step 6: Test Your Function

1. Create a test file:

tests/test_iss_locator.py    

2. Add basic tests:

import pytest    
from gofannon.open_notify_space.iss_locator import IssLocator

def test_iss_locator():    
    tool = IssLocator()    
    result = tool.fn()

    assert isinstance(result, dict)    
    assert 'iss_position' in result    
    assert 'latitude' in result['iss_position']    
    assert 'longitude' in result['iss_position']    

3. Run the tests locally using Poetry:

# Install dependencies (if you haven't already)
poetry install

# Run the tests
poetry run pytest tests/test_iss_locator.py -v    

4. Check the test output to ensure your function works as expected.

Step 7: Submit Your Pull Request

  1. Fork the repository
  2. Create a new branch for your feature
  3. Commit your changes
  4. Push to your fork
  5. Open a pull request

Appendix A: Understanding the Decorators

The @FunctionRegistry.register decorator does several important things:

  1. Registers your function with Gofannon’s central registry
  2. Makes your function available to the orchestration system
  3. Enables automatic documentation generation
  4. Allows your function to be exported to different platforms (LangChain, Bedrock, etc.)

By following this pattern, you ensure your function integrates seamlessly with the rest of the Gofannon ecosystem.

Next Steps

Once you’re comfortable with simple REST functions, you can explore:

Appendix B: Adding an API with Authentication

If you’re working with an API that requires authentication, follow these additional steps:

Step 1: Add API Key to Configuration

1. Add your API key to the .env file:

NEW_API_KEY=your_api_key_here    

2. Update the ToolConfig class in gofannon/config.py:

class ToolConfig:    
    _instance = None

    def __init__(self):    
        load_dotenv()    
        self.config = {    
            # Existing keys...    
            'new_api_key': os.getenv('NEW_API_KEY'),    
        }    

Step 2: Modify Your Function Class

Update your function class to handle authentication:

class AuthenticatedApiTool(BaseTool):    
    def __init__(self, api_key=None, name="authenticated_tool"):    
        super().__init__()    
        self.api_key = api_key or ToolConfig.get("new_api_key")    
        self.name = name    
        self.API_SERVICE = 'new_api'  # Add this for API-specific configuration

    @property    
    def definition(self):    
        return {    
            "type": "function",    
            "function": {    
                "name": self.name,    
                "description": "Description of your authenticated API function",    
                "parameters": {    
                    "type": "object",    
                    "properties": {    
                        # Add your parameters here    
                    },    
                    "required": []  # Add required parameters here    
                }    
            }    
        }    
    
    def fn(self, **kwargs):    
        headers = {    
            'Authorization': f'Bearer {self.api_key}',    
            'Content-Type': 'application/json'    
        }    
            
        # Make your authenticated API call    
        response = requests.get(    
            "https://api.example.com/endpoint",    
            headers=headers,    
            params=kwargs    
        )    
        response.raise_for_status()    
        return response.json()    

Step 3: Add Error Handling

Add robust error handling for authentication failures:

def fn(self, **kwargs):    
    try:    
        headers = {    
            'Authorization': f'Bearer {self.api_key}',    
            'Content-Type': 'application/json'    
        }

        response = requests.get(    
            "https://api.example.com/endpoint",    
            headers=headers,    
            params=kwargs    
        )    
            
        # Handle authentication errors    
        if response.status_code == 401:    
            raise ValueError("Invalid API key")    
                
        response.raise_for_status()    
        return response.json()    
            
    except requests.exceptions.RequestException as e:    
        logger.error(f"API request failed: {e}")    
        raise    

Step 4: Update Documentation

Update your function’s documentation to include authentication requirements:

class AuthenticatedApiTool(BaseTool):    
    """Description of your authenticated API function

    Requires an API key from https://api.example.com    
    Set the API key in your .env file as NEW_API_KEY=your_key    
    """    

Step 5: Add Authentication Tests

Add tests for authentication scenarios:

def test_authentication_failure():    
    tool = AuthenticatedApiTool(api_key="invalid_key")

    with pytest.raises(ValueError) as exc_info:    
        tool.fn()    
            
    assert "Invalid API key" in str(exc_info.value)    

Best Practices for Authentication

  1. Never hard-code API keys in your code
  2. Use environment variables for configuration
  3. Handle authentication errors gracefully
  4. Consider adding rate limiting for API calls
  5. Document authentication requirements clearly

Appendix C: Choosing a REST API to Wrap a Tool: A Guide for New Contributors

If you’re new to contributing tools for large language models (LLMs) or agentic systems, one of the most impactful ways to get started is by wrapping a REST API into a tool. REST APIs are widely used, well-documented, and provide access to a wealth of data and functionality. Here’s a quick guide to help you choose the right REST API for your tool:

  1. Identify a Common Use Case: Start by thinking about tasks that users frequently ask LLMs or agentic systems to perform. For example, retrieving weather data, fetching stock prices, or searching for information are common use cases. Choose an API that addresses one of these needs.

  2. Look for Well-Documented APIs: A good REST API should have clear and comprehensive documentation. This makes it easier to understand how to use the API, what parameters it requires, and what kind of output it provides.

  3. Check for Accessibility: Ensure the API is publicly accessible or provides an easy way to obtain an API key. Some APIs may require registration or have usage limits, so choose one that aligns with your project’s goals.

  4. Consider Data Relevance: The API should provide data or functionality that is relevant and useful to the end user. For example, a weather API is more universally useful than a niche API for a specific industry.

  5. Evaluate Performance and Reliability: Choose an API that is reliable and has low latency. Users expect quick responses, so the API should be able to handle requests efficiently.

  6. Start Simple: If you’re new to this, start with a simple API that has a straightforward interface. For example, wrapping a currency conversion API or a public transportation schedule API is a great way to get started.

By following these guidelines, you can choose a REST API that will make a meaningful contribution to the ecosystem of tools for LLMs and agentic systems. Happy coding!