mirror of
https://github.com/aljazceru/goose.git
synced 2025-12-21 16:14:21 +01:00
feat: V1.0 (#734)
Co-authored-by: Michael Neale <michael.neale@gmail.com> Co-authored-by: Wendy Tang <wendytang@squareup.com> Co-authored-by: Jarrod Sibbison <72240382+jsibbison-square@users.noreply.github.com> Co-authored-by: Alex Hancock <alex.hancock@example.com> Co-authored-by: Alex Hancock <alexhancock@block.xyz> Co-authored-by: Lifei Zhou <lifei@squareup.com> Co-authored-by: Wes <141185334+wesrblock@users.noreply.github.com> Co-authored-by: Max Novich <maksymstepanenko1990@gmail.com> Co-authored-by: Zaki Ali <zaki@squareup.com> Co-authored-by: Salman Mohammed <smohammed@squareup.com> Co-authored-by: Kalvin C <kalvinnchau@users.noreply.github.com> Co-authored-by: Alec Thomas <alec@swapoff.org> Co-authored-by: lily-de <119957291+lily-de@users.noreply.github.com> Co-authored-by: kalvinnchau <kalvin@block.xyz> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Rizel Scarlett <rizel@squareup.com> Co-authored-by: bwrage <bwrage@squareup.com> Co-authored-by: Kalvin Chau <kalvin@squareup.com> Co-authored-by: Alice Hau <110418948+ahau-square@users.noreply.github.com> Co-authored-by: Alistair Gray <ajgray@stripe.com> Co-authored-by: Nahiyan Khan <nahiyan.khan@gmail.com> Co-authored-by: Alex Hancock <alexhancock@squareup.com> Co-authored-by: Nahiyan Khan <nahiyan@squareup.com> Co-authored-by: marcelle <1852848+laanak08@users.noreply.github.com> Co-authored-by: Yingjie He <yingjiehe@block.xyz> Co-authored-by: Yingjie He <yingjiehe@squareup.com> Co-authored-by: Lily Delalande <ldelalande@block.xyz> Co-authored-by: Adewale Abati <acekyd01@gmail.com> Co-authored-by: Ebony Louis <ebony774@gmail.com> Co-authored-by: Angie Jones <jones.angie@gmail.com> Co-authored-by: Ebony Louis <55366651+EbonyLouis@users.noreply.github.com>
This commit is contained in:
152
crates/mcp-macros/src/lib.rs
Normal file
152
crates/mcp-macros/src/lib.rs
Normal file
@@ -0,0 +1,152 @@
|
||||
use convert_case::{Case, Casing};
|
||||
use proc_macro::TokenStream;
|
||||
use quote::{format_ident, quote};
|
||||
use std::collections::HashMap;
|
||||
use syn::{
|
||||
parse::Parse, parse::ParseStream, parse_macro_input, punctuated::Punctuated, Expr, ExprLit,
|
||||
FnArg, ItemFn, Lit, Meta, Pat, PatType, Token,
|
||||
};
|
||||
|
||||
struct MacroArgs {
|
||||
name: Option<String>,
|
||||
description: Option<String>,
|
||||
param_descriptions: HashMap<String, String>,
|
||||
}
|
||||
|
||||
impl Parse for MacroArgs {
|
||||
fn parse(input: ParseStream) -> syn::Result<Self> {
|
||||
let mut name = None;
|
||||
let mut description = None;
|
||||
let mut param_descriptions = HashMap::new();
|
||||
|
||||
let meta_list: Punctuated<Meta, Token![,]> = Punctuated::parse_terminated(input)?;
|
||||
|
||||
for meta in meta_list {
|
||||
match meta {
|
||||
Meta::NameValue(nv) => {
|
||||
let ident = nv.path.get_ident().unwrap().to_string();
|
||||
if let Expr::Lit(ExprLit {
|
||||
lit: Lit::Str(lit_str),
|
||||
..
|
||||
}) = nv.value
|
||||
{
|
||||
match ident.as_str() {
|
||||
"name" => name = Some(lit_str.value()),
|
||||
"description" => description = Some(lit_str.value()),
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
}
|
||||
Meta::List(list) if list.path.is_ident("params") => {
|
||||
let nested: Punctuated<Meta, Token![,]> =
|
||||
list.parse_args_with(Punctuated::parse_terminated)?;
|
||||
|
||||
for meta in nested {
|
||||
if let Meta::NameValue(nv) = meta {
|
||||
if let Expr::Lit(ExprLit {
|
||||
lit: Lit::Str(lit_str),
|
||||
..
|
||||
}) = nv.value
|
||||
{
|
||||
let param_name = nv.path.get_ident().unwrap().to_string();
|
||||
param_descriptions.insert(param_name, lit_str.value());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(MacroArgs {
|
||||
name,
|
||||
description,
|
||||
param_descriptions,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
#[proc_macro_attribute]
|
||||
pub fn tool(args: TokenStream, input: TokenStream) -> TokenStream {
|
||||
let args = parse_macro_input!(args as MacroArgs);
|
||||
let input_fn = parse_macro_input!(input as ItemFn);
|
||||
|
||||
// Extract function details
|
||||
let fn_name = &input_fn.sig.ident;
|
||||
let fn_name_str = fn_name.to_string();
|
||||
|
||||
// Generate PascalCase struct name from the function name
|
||||
let struct_name = format_ident!("{}", { fn_name_str.to_case(Case::Pascal) });
|
||||
|
||||
// Use provided name or function name as default
|
||||
let tool_name = args.name.unwrap_or(fn_name_str);
|
||||
let tool_description = args.description.unwrap_or_default();
|
||||
|
||||
// Extract parameter names, types, and descriptions
|
||||
let mut param_defs = Vec::new();
|
||||
let mut param_names = Vec::new();
|
||||
|
||||
for arg in input_fn.sig.inputs.iter() {
|
||||
if let FnArg::Typed(PatType { pat, ty, .. }) = arg {
|
||||
if let Pat::Ident(param_ident) = &**pat {
|
||||
let param_name = ¶m_ident.ident;
|
||||
let param_name_str = param_name.to_string();
|
||||
let description = args
|
||||
.param_descriptions
|
||||
.get(¶m_name_str)
|
||||
.map(|s| s.as_str())
|
||||
.unwrap_or("");
|
||||
|
||||
param_names.push(param_name);
|
||||
param_defs.push(quote! {
|
||||
#[schemars(description = #description)]
|
||||
#param_name: #ty
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Generate the implementation
|
||||
let params_struct_name = format_ident!("{}Parameters", struct_name);
|
||||
let expanded = quote! {
|
||||
#[derive(serde::Deserialize, schemars::JsonSchema)]
|
||||
struct #params_struct_name {
|
||||
#(#param_defs,)*
|
||||
}
|
||||
|
||||
#input_fn
|
||||
|
||||
#[derive(Default)]
|
||||
struct #struct_name;
|
||||
|
||||
#[async_trait::async_trait]
|
||||
impl mcp_core::handler::ToolHandler for #struct_name {
|
||||
fn name(&self) -> &'static str {
|
||||
#tool_name
|
||||
}
|
||||
|
||||
fn description(&self) -> &'static str {
|
||||
#tool_description
|
||||
}
|
||||
|
||||
fn schema(&self) -> serde_json::Value {
|
||||
mcp_core::handler::generate_schema::<#params_struct_name>()
|
||||
.expect("Failed to generate schema")
|
||||
}
|
||||
|
||||
async fn call(&self, params: serde_json::Value) -> Result<serde_json::Value, mcp_core::handler::ToolError> {
|
||||
let params: #params_struct_name = serde_json::from_value(params)
|
||||
.map_err(|e| mcp_core::handler::ToolError::InvalidParameters(e.to_string()))?;
|
||||
|
||||
// Extract parameters and call the function
|
||||
let result = #fn_name(#(params.#param_names,)*).await
|
||||
.map_err(|e| mcp_core::handler::ToolError::ExecutionError(e.to_string()))?;
|
||||
|
||||
Ok(serde_json::to_value(result).expect("should serialize"))
|
||||
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
TokenStream::from(expanded)
|
||||
}
|
||||
Reference in New Issue
Block a user