# Tool This document guides you on creating custom tools for the LLM Functions framework in Bash, JavaScript, and Python. ## Defining Tool Parameters To define the parameters that your tool accepts, you will use specially formatted comments within your tool's source code. The `Argcfile.sh` script utilizes these comments to automatically generate the function declarations needed by the LLM. ### Json Schema The following JSON schema includes various types of properties. We will use this as an example to see how to write comments in each language so they can be automatically generated. ```json { "name": "demo", "description": "Demonstrate how to create a tool using Javascript and how to use comments.", "parameters": { "type": "object", "properties": { "string": { "type": "string", "description": "Define a required string property" }, "string_enum": { "type": "string", "enum": [ "foo", "bar" ], "description": "Define a required string property with enum" }, "string_optional": { "type": "string", "description": "Define a optional string property" }, "boolean": { "type": "boolean", "description": "Define a required boolean property" }, "integer": { "type": "integer", "description": "Define a required integer property" }, "number": { "type": "number", "description": "Define a required number property" }, "array": { "type": "array", "items": { "type": "string" }, "description": "Define a required string array property" }, "array_optional": { "type": "array", "items": { "type": "string" }, "description": "Define a optional string array property" } }, "required": [ "string", "string_enum", "boolean", "integer", "number", "array" ] } } ``` ### Bash Use `# @describe`, `# @option`, and `# @flag` comments to define your tool's parameters. * `# @describe `: A brief description of your tool's functionality. This is required. * `# @option --[!][] `: Defines an option. * `--`: The name of the option (use kebab-case). * `!`: Indicates a required option. * ``: The data type (e.g., `INT`, `NUM`, ``). If omitted, defaults to `STRING`. * ``: Any constraints (e.g., `[foo|bar]` for an enum). * ``: A description of the option. * `# @flag -- `: Defines a boolean flag. * `--`: The name of the flag (use kebab-case). * ``: A description of the flag. **Example ([tools/demo_sh.sh](https://github.com/sigoden/llm-functions/blob/main/tools/demo_sh.sh)):** ```sh file=tools/demo_sh.sh #!/usr/bin/env bash set -e # @describe Demonstrate how to create a tool using Bash and how to use comment tags. # @option --string! Define a required string property # @option --string-enum![foo|bar] Define a required string property with enum # @option --string-optional Define a optional string property # @flag --boolean Define a boolean property # @option --integer! Define a required integer property # @option --number! Define a required number property # @option --array+ Define a required string array property # @option --array-optional* Define a optional string array property # @env LLM_OUTPUT=/dev/stdout The output path main() { # ... your bash code ... } eval "$(argc --argc-eval "$0" "$@")" ``` ### JavaScript Use JSDoc-style comments to define your tool's parameters. The `@typedef` block defines the argument object, and each property within that object represents a parameter. * `/** ... */`: JSDoc comment block containing the description and parameter definitions. * `@typedef {Object} Args`: Defines the type of the argument object. * `@property {} `: Defines a property (parameter) of the `Args` object. * ``: The data type (e.g., `string`, `boolean`, `number`, `string[]`, `{foo|bar}`). * ``: The name of the parameter. * ``: A description of the parameter. * `[]`: Indicates an optional parameter. **Example ([tools/demo_js.js](https://github.com/sigoden/llm-functions/blob/main/tools/demo_js.js)):** ```js file=tools/demo_js.js /** * Demonstrate how to create a tool using Javascript and how to use comments. * @typedef {Object} Args * @property {string} string - Define a required string property * @property {'foo'|'bar'} string_enum - Define a required string property with enum * @property {string} [string_optional] - Define a optional string property * @property {boolean} boolean - Define a required boolean property * @property {Integer} integer - Define a required integer property * @property {number} number - Define a required number property * @property {string[]} array - Define a required string array property * @property {string[]} [array_optional] - Define a optional string array property * @param {Args} args */ exports.run = function (args) { // ... your JavaScript code ... } ``` Of course, you can also use ESM `export` expressions to export functions. ```js export function run() { // ... your JavaScript code ... } ``` ### Python Use type hints and docstrings to define your tool's parameters. * `def run(...)`: Function definition. * ` : `: Type hints with descriptions in the docstring. * ``: The data type (e.g., `str`, `bool`, `int`, `float`, `List[str]`, `Literal["foo", "bar"]`). * ``: The name of the parameter. * ``: Description of the parameter. * `Optional[...]`: Indicates an optional parameter. **Example ([tools/demo_py.py](https://github.com/sigoden/llm-functions/blob/main/tools/demo_py.py)):** ```py file=tools/demo_py.py def run( string: str, string_enum: Literal["foo", "bar"], boolean: bool, integer: int, number: float, array: List[str], string_optional: Optional[str] = None, array_optional: Optional[List[str]] = None, ): """Demonstrate how to create a tool using Python and how to use comments. Args: string: Define a required string property string_enum: Define a required string property with enum boolean: Define a required boolean property integer: Define a required integer property number: Define a required number property array: Define a required string array property string_optional: Define a optional string property array_optional: Define a optional string array property """ # ... your Python code ... ``` ## Common tools Common tools can be found in `tools/.{sh,js,py}`. Each script defines a single tool. ## Agent tools Agents can possess their own toolset scripts located under `agents//tools.{sh,js,py}`, which can contain multiple tool functions. The following is an example of git agent: ### Bash ```sh file=agents/git/tools.sh # @cmd Shows the working tree status git_status() { # ... your bash code ... } # @cmd Shows differences between branches or commits # @option --target! Shows differences between branches or commits git_diff() { # ... your bash code ... } eval "$(argc --argc-eval "$0" "$@")" ``` > In `tools/.sh`, we use the `@describe` comment tag and a single `main` function, since it has only one function and no subcommands. > In `agent//tools.sh`, we use the `@cmd` comment tag and named functions, since it can have multiple tool functions. ### JavaScript ```js file=agents/git/tools.js /** * Shows the working tree status */ exports.git_status = function() { // ... your JavaScript code ... } /** * Shows differences between branches or commits * @typedef {Object} Args * @property {string} target - Shows differences between branches or commits * @param {Args} args */ exports.git_diff = function() { // ... your JavaScript code ... } ``` ### Python ```py file=agents/git/tools.py def git_status(): """Shows the working tree status""" # ... your Python code ... def git_diff(target: str): """Shows differences between branches or commits Args: target: Shows differences between branches or commits """ # ... your Python code ... ``` ## Quickly Create Tools `Argcfile.sh` provides a tool to quickly create script tools. ``` $ argc create@tool -h Create a boilplate tool script Examples: ./scripts/create-tool.sh _test.py foo bar! baz+ qux* USAGE: create-tool [OPTIONS] [PARAMS]... ARGS: The script file name. [PARAMS]... The script parameters OPTIONS: --description The tool description --force Override the exist tool file -h, --help Print help -V, --version Print version ``` ```sh argc create@tool _test.sh foo bar! baz+ qux* ``` The suffixes after property names represent different meanings. - `!`: The property is required. - `*`: The property value must be an array. - `+`: The property is required, and its value must be an array. - No suffix: The property is optional.