use crate::shared::models::UserSession; use crate::shared::state::AppState; use log::debug; use rhai::{Dynamic, Engine}; use std::sync::Arc; pub fn instr_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("INSTR", |haystack: &str, needle: &str| -> i64 { instr_impl(1, haystack, needle) }); engine.register_fn("instr", |haystack: &str, needle: &str| -> i64 { instr_impl(1, haystack, needle) }); engine.register_fn("INSTR", |start: i64, haystack: &str, needle: &str| -> i64 { instr_impl(start, haystack, needle) }); engine.register_fn("instr", |start: i64, haystack: &str, needle: &str| -> i64 { instr_impl(start, haystack, needle) }); debug!("Registered INSTR keyword"); } pub fn instr_impl(start: i64, haystack: &str, needle: &str) -> i64 { if haystack.is_empty() || needle.is_empty() { return 0; } let start_idx = if start < 1 { 0 } else { (start - 1) as usize }; if start_idx >= haystack.len() { return 0; } match haystack[start_idx..].find(needle) { Some(pos) => (start_idx + pos + 1) as i64, None => 0, } } pub fn is_numeric_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("IS_NUMERIC", |value: &str| -> bool { is_numeric_impl(value) }); engine.register_fn("is_numeric", |value: &str| -> bool { is_numeric_impl(value) }); engine.register_fn("ISNUMERIC", |value: &str| -> bool { is_numeric_impl(value) }); engine.register_fn("isnumeric", |value: &str| -> bool { is_numeric_impl(value) }); engine.register_fn("IS_NUMERIC", |value: Dynamic| -> bool { match value.clone().into_string() { Ok(s) => is_numeric_impl(&s), Err(_) => { value.is::() || value.is::() } } }); debug!("Registered IS_NUMERIC keyword"); } pub fn is_numeric_impl(value: &str) -> bool { let trimmed = value.trim(); if trimmed.is_empty() { return false; } if trimmed.parse::().is_ok() { return true; } if trimmed.parse::().is_ok() { return true; } false } pub fn not_operator(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("NOT", |value: bool| -> bool { !value }); engine.register_fn("not", |value: bool| -> bool { !value }); debug!("Registered NOT operator"); } pub fn logical_operators(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("OR", |a: bool, b: bool| -> bool { a || b }); engine.register_fn("or", |a: bool, b: bool| -> bool { a || b }); engine.register_fn("AND", |a: bool, b: bool| -> bool { a && b }); engine.register_fn("and", |a: bool, b: bool| -> bool { a && b }); debug!("Registered logical operators (OR, AND)"); } pub fn lower_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("LOWER", |s: &str| -> String { s.to_lowercase() }); engine.register_fn("lower", |s: &str| -> String { s.to_lowercase() }); engine.register_fn("LCASE", |s: &str| -> String { s.to_lowercase() }); debug!("Registered LOWER/LCASE keyword"); } pub fn upper_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("UPPER", |s: &str| -> String { s.to_uppercase() }); engine.register_fn("upper", |s: &str| -> String { s.to_uppercase() }); engine.register_fn("UCASE", |s: &str| -> String { s.to_uppercase() }); debug!("Registered UPPER/UCASE keyword"); } pub fn len_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("LEN", |s: &str| -> i64 { s.len() as i64 }); engine.register_fn("len", |s: &str| -> i64 { s.len() as i64 }); debug!("Registered LEN keyword"); } pub fn trim_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("TRIM", |s: &str| -> String { s.trim().to_string() }); engine.register_fn("trim", |s: &str| -> String { s.trim().to_string() }); engine.register_fn("LTRIM", |s: &str| -> String { s.trim_start().to_string() }); engine.register_fn("ltrim", |s: &str| -> String { s.trim_start().to_string() }); engine.register_fn("RTRIM", |s: &str| -> String { s.trim_end().to_string() }); engine.register_fn("rtrim", |s: &str| -> String { s.trim_end().to_string() }); debug!("Registered TRIM/LTRIM/RTRIM keywords"); } pub fn left_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("LEFT", |s: &str, count: i64| -> String { let count = count.max(0) as usize; s.chars().take(count).collect() }); engine.register_fn("left", |s: &str, count: i64| -> String { let count = count.max(0) as usize; s.chars().take(count).collect() }); debug!("Registered LEFT keyword"); } pub fn right_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("RIGHT", |s: &str, count: i64| -> String { let count = count.max(0) as usize; let len = s.chars().count(); if count >= len { s.to_string() } else { s.chars().skip(len - count).collect() } }); engine.register_fn("right", |s: &str, count: i64| -> String { let count = count.max(0) as usize; let len = s.chars().count(); if count >= len { s.to_string() } else { s.chars().skip(len - count).collect() } }); debug!("Registered RIGHT keyword"); } pub fn mid_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("MID", |s: &str, start: i64| -> String { let start_idx = if start < 1 { 0 } else { (start - 1) as usize }; s.chars().skip(start_idx).collect() }); engine.register_fn("MID", |s: &str, start: i64, length: i64| -> String { let start_idx = if start < 1 { 0 } else { (start - 1) as usize }; let len = length.max(0) as usize; s.chars().skip(start_idx).take(len).collect() }); engine.register_fn("mid", |s: &str, start: i64| -> String { let start_idx = if start < 1 { 0 } else { (start - 1) as usize }; s.chars().skip(start_idx).collect() }); engine.register_fn("mid", |s: &str, start: i64, length: i64| -> String { let start_idx = if start < 1 { 0 } else { (start - 1) as usize }; let len = length.max(0) as usize; s.chars().skip(start_idx).take(len).collect() }); debug!("Registered MID keyword"); } pub fn replace_keyword(_state: &Arc, _user: UserSession, engine: &mut Engine) { engine.register_fn("REPLACE", |s: &str, find: &str, replace: &str| -> String { s.replace(find, replace) }); engine.register_fn("replace", |s: &str, find: &str, replace: &str| -> String { s.replace(find, replace) }); debug!("Registered REPLACE keyword"); } pub fn register_string_functions(state: Arc, user: UserSession, engine: &mut Engine) { instr_keyword(&state, user.clone(), engine); is_numeric_keyword(&state, user.clone(), engine); not_operator(&state, user.clone(), engine); logical_operators(&state, user.clone(), engine); lower_keyword(&state, user.clone(), engine); upper_keyword(&state, user.clone(), engine); len_keyword(&state, user.clone(), engine); trim_keyword(&state, user.clone(), engine); left_keyword(&state, user.clone(), engine); right_keyword(&state, user.clone(), engine); mid_keyword(&state, user.clone(), engine); replace_keyword(&state, user, engine); }