2025-10-11 20:02:14 -03:00
|
|
|
use crate::shared::state::AppState;
|
2025-10-12 20:12:49 -03:00
|
|
|
use crate::{channels::ChannelAdapter, shared::models::UserSession};
|
2025-10-11 12:29:03 -03:00
|
|
|
use log::info;
|
|
|
|
|
use rhai::{Dynamic, Engine, EvalAltResult};
|
|
|
|
|
|
2025-10-12 20:12:49 -03:00
|
|
|
pub fn hear_keyword(state: &AppState, user: UserSession, engine: &mut Engine) {
|
2025-10-11 12:29:03 -03:00
|
|
|
let session_id = user.id;
|
2025-10-12 20:12:49 -03:00
|
|
|
let cache = state.redis_client.clone();
|
2025-10-11 20:02:14 -03:00
|
|
|
|
2025-10-11 12:29:03 -03:00
|
|
|
engine
|
2025-10-11 20:02:14 -03:00
|
|
|
.register_custom_syntax(&["HEAR", "$ident$"], true, move |_context, inputs| {
|
|
|
|
|
let variable_name = inputs[0]
|
|
|
|
|
.get_string_value()
|
|
|
|
|
.expect("Expected identifier as string")
|
|
|
|
|
.to_string();
|
|
|
|
|
|
|
|
|
|
info!(
|
|
|
|
|
"HEAR command waiting for user input to store in variable: {}",
|
|
|
|
|
variable_name
|
|
|
|
|
);
|
|
|
|
|
|
2025-10-12 20:12:49 -03:00
|
|
|
let cache_clone = cache.clone();
|
|
|
|
|
let session_id_clone = session_id;
|
|
|
|
|
let var_name_clone = variable_name.clone();
|
|
|
|
|
|
2025-10-11 12:29:03 -03:00
|
|
|
tokio::spawn(async move {
|
2025-10-11 20:02:14 -03:00
|
|
|
log::debug!(
|
|
|
|
|
"HEAR: Starting async task for session {} and variable '{}'",
|
2025-10-12 20:12:49 -03:00
|
|
|
session_id_clone,
|
|
|
|
|
var_name_clone
|
2025-10-11 20:02:14 -03:00
|
|
|
);
|
2025-10-12 20:12:49 -03:00
|
|
|
|
|
|
|
|
if let Some(cache_client) = &cache_clone {
|
|
|
|
|
let mut conn = match cache_client.get_multiplexed_async_connection().await {
|
|
|
|
|
Ok(conn) => conn,
|
|
|
|
|
Err(e) => {
|
|
|
|
|
log::error!("Failed to connect to cache: {}", e);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let key = format!("hear:{}:{}", session_id_clone, var_name_clone);
|
|
|
|
|
let _: Result<(), _> = redis::cmd("SET")
|
|
|
|
|
.arg(&key)
|
|
|
|
|
.arg("waiting")
|
|
|
|
|
.query_async(&mut conn)
|
|
|
|
|
.await;
|
|
|
|
|
}
|
2025-10-11 20:02:14 -03:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Err(Box::new(EvalAltResult::ErrorRuntime(
|
|
|
|
|
"Waiting for user input".into(),
|
|
|
|
|
rhai::Position::NONE,
|
|
|
|
|
)))
|
2025-10-11 12:29:03 -03:00
|
|
|
})
|
|
|
|
|
.unwrap();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn talk_keyword(state: &AppState, user: UserSession, engine: &mut Engine) {
|
2025-10-11 20:02:14 -03:00
|
|
|
use crate::shared::models::BotResponse;
|
|
|
|
|
|
2025-10-11 12:29:03 -03:00
|
|
|
let state_clone = state.clone();
|
2025-10-11 20:02:14 -03:00
|
|
|
let user_clone = user.clone();
|
2025-10-11 12:29:03 -03:00
|
|
|
|
|
|
|
|
engine
|
|
|
|
|
.register_custom_syntax(&["TALK", "$expr$"], true, move |context, inputs| {
|
|
|
|
|
let message = context.eval_expression_tree(&inputs[0])?.to_string();
|
|
|
|
|
|
|
|
|
|
info!("TALK command executed: {}", message);
|
|
|
|
|
|
2025-10-11 20:02:14 -03:00
|
|
|
let response = BotResponse {
|
2025-10-11 12:29:03 -03:00
|
|
|
bot_id: "default_bot".to_string(),
|
2025-10-11 20:02:14 -03:00
|
|
|
user_id: user_clone.user_id.to_string(),
|
|
|
|
|
session_id: user_clone.id.to_string(),
|
2025-10-11 12:29:03 -03:00
|
|
|
channel: "basic".to_string(),
|
|
|
|
|
content: message,
|
2025-10-12 14:39:23 -03:00
|
|
|
message_type: 1,
|
2025-10-11 12:29:03 -03:00
|
|
|
stream_token: None,
|
|
|
|
|
is_complete: true,
|
|
|
|
|
};
|
|
|
|
|
|
2025-10-12 20:12:49 -03:00
|
|
|
let state_for_spawn = state_clone.clone();
|
2025-10-11 12:29:03 -03:00
|
|
|
tokio::spawn(async move {
|
2025-10-12 20:12:49 -03:00
|
|
|
if let Err(e) = state_for_spawn.web_adapter.send_message(response).await {
|
|
|
|
|
log::error!("Failed to send TALK message: {}", e);
|
|
|
|
|
}
|
2025-10-11 12:29:03 -03:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Ok(Dynamic::UNIT)
|
|
|
|
|
})
|
|
|
|
|
.unwrap();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn set_context_keyword(state: &AppState, user: UserSession, engine: &mut Engine) {
|
2025-10-12 20:12:49 -03:00
|
|
|
let cache = state.redis_client.clone();
|
2025-10-11 12:29:03 -03:00
|
|
|
|
|
|
|
|
engine
|
|
|
|
|
.register_custom_syntax(
|
|
|
|
|
&["SET", "CONTEXT", "$expr$"],
|
|
|
|
|
true,
|
|
|
|
|
move |context, inputs| {
|
|
|
|
|
let context_value = context.eval_expression_tree(&inputs[0])?.to_string();
|
|
|
|
|
|
|
|
|
|
info!("SET CONTEXT command executed: {}", context_value);
|
|
|
|
|
|
|
|
|
|
let redis_key = format!("context:{}:{}", user.user_id, user.id);
|
|
|
|
|
|
2025-10-12 20:12:49 -03:00
|
|
|
let cache_clone = cache.clone();
|
2025-10-11 20:02:14 -03:00
|
|
|
|
2025-10-11 12:29:03 -03:00
|
|
|
tokio::spawn(async move {
|
2025-10-12 20:12:49 -03:00
|
|
|
if let Some(cache_client) = &cache_clone {
|
|
|
|
|
let mut conn = match cache_client.get_multiplexed_async_connection().await {
|
2025-10-11 12:29:03 -03:00
|
|
|
Ok(conn) => conn,
|
|
|
|
|
Err(e) => {
|
2025-10-12 20:12:49 -03:00
|
|
|
log::error!("Failed to connect to cache: {}", e);
|
2025-10-11 12:29:03 -03:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let _: Result<(), _> = redis::cmd("SET")
|
|
|
|
|
.arg(&redis_key)
|
|
|
|
|
.arg(&context_value)
|
|
|
|
|
.query_async(&mut conn)
|
|
|
|
|
.await;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Ok(Dynamic::UNIT)
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
.unwrap();
|
|
|
|
|
}
|