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:
- The API you plan to wrap
- A brief description of the tool’s functionality
- Any authentication requirements (if applicable)
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:
- Public APIs like OpenNotify
- Public datasets like NASA APIs
- Simple utility APIs like JSONPlaceholder
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:
- Create a new directory for your API:
gofannon/new_api_name/
- Add an empty
__init__.py
file:
gofannon/new_api_name/__init__.py
- Create your function file in the new directory:
gofannon/new_api_name/your_function.py
Important Notes:
- The
__init__.py
file should remain empty - Do not import your function in the
__init__.py
file - The function registration happens automatically through the decorator
- Follow the same pattern as existing functions for consistency
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:
- BaseTool Inheritance: Your class must inherit from
BaseTool
to get core functionality - FunctionRegistry.register: This decorator registers your function with Gofannon’s function registry
- definition Property: Defines the function’s interface using OpenAI’s function calling schema
- 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
- Fork the repository
- Create a new branch for your feature
- Commit your changes
- Push to your fork
- Open a pull request
Appendix A: Understanding the Decorators
The @FunctionRegistry.register
decorator does several important things:
- Registers your function with Gofannon’s central registry
- Makes your function available to the orchestration system
- Enables automatic documentation generation
- 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:
- Adding authentication
- Creating more complex parameter schemas
- Implementing error handling
- Adding caching or rate limiting
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
- Never hard-code API keys in your code
- Use environment variables for configuration
- Handle authentication errors gracefully
- Consider adding rate limiting for API calls
- 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:
-
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.
-
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.
-
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.
-
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.
-
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.
-
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!