use display_error_chain::DisplayErrorChain;
use gemini_rust::{Content, FunctionCallingMode, FunctionDeclaration, Gemini, Part};
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use std::env;
use std::process::ExitCode;
use tracing::info;
#[derive(Default, Debug, Clone, Serialize, Deserialize, JsonSchema)]
#[schemars(description = "The unit of temperature")]
#[serde(rename_all = "lowercase")]
enum Unit {
#[default]
Celsius,
Fahrenheit,
}
impl std::fmt::Display for Unit {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Unit::Celsius => write!(f, "Celsius"),
Unit::Fahrenheit => write!(f, "Fahrenheit"),
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize, JsonSchema)]
#[serde(default)]
struct Weather {
location: String,
unit: Option<Unit>,
}
impl Default for Weather {
fn default() -> Self {
Weather {
location: "".to_string(),
unit: Some(Unit::Celsius),
}
}
}
impl std::fmt::Display for Weather {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}",
serde_json::to_string_pretty(self).unwrap_or_default()
)
}
}
#[derive(Debug, Clone, Serialize, Deserialize, JsonSchema)]
struct WeatherResponse {
temperature: i32,
unit: Unit,
condition: String,
location: String,
}
#[tokio::main]
async fn main() -> ExitCode {
tracing_subscriber::fmt()
.with_env_filter(
tracing_subscriber::EnvFilter::builder()
.with_default_directive(tracing::level_filters::LevelFilter::INFO.into())
.from_env_lossy(),
)
.init();
match do_main().await {
Ok(()) => ExitCode::SUCCESS,
Err(e) => {
let error_chain = DisplayErrorChain::new(e.as_ref());
tracing::error!(error.debug = ?e, error.chained = %error_chain, "execution failed");
ExitCode::FAILURE
}
}
}
async fn do_main() -> Result<(), Box<dyn std::error::Error>> {
let api_key = env::var("GEMINI_API_KEY")?;
let client = Gemini::new(api_key).expect("unable to create Gemini API client");
let get_weather = FunctionDeclaration::new(
"get_weather",
"Get the current weather for a location",
None,
)
.with_parameters::<Weather>()
.with_response::<WeatherResponse>();
info!("sending function call request");
let response = client
.generate_content()
.with_user_message("What's the weather like in Tokyo right now?")
.with_function(get_weather)
.with_function_calling_mode(FunctionCallingMode::Any)
.execute()
.await?;
let Some(function_call) = response.function_calls().first().cloned() else {
tracing::error!(
response = response.text(),
"no function calls in the response"
);
return Ok(());
};
let result = serde_json::from_value::<Weather>(function_call.args.clone())?;
info!(
function_name = function_call.name,
args = serde_json::to_string_pretty(&result)?,
"function call received"
);
let location = result.location;
let unit = result.unit.unwrap_or_default();
let weather_response = WeatherResponse {
temperature: 22,
unit,
condition: "sunny".to_string(),
location: location.clone(),
};
info!("sending function response");
let mut final_request = client
.generate_content()
.with_user_message("What's the weather like in Tokyo right now?");
let call_content = Content {
parts: Some(vec![Part::FunctionCall {
function_call: (*function_call).clone(),
thought_signature: None,
}]),
..Default::default()
};
final_request.contents.push(call_content);
final_request = final_request.with_function_response("get_weather", weather_response)?;
let final_response = final_request.execute().await?;
info!(final_response = final_response.text(), "final response");
Ok(())
}