#[cfg(feature = "python")]
use oxibase::Database;
#[cfg(feature = "python")]
mod python_function_tests {
use super::*;
use std::f64::consts::{PI, TAU};
#[test]
fn test_python_function_creation_and_execution() {
let db = Database::open("memory://python_create_test").unwrap();
db.execute(
r#"
CREATE FUNCTION compute_sum(a INTEGER, b INTEGER) RETURNS INTEGER
LANGUAGE PYTHON AS 'return arguments[0] + arguments[1]'
"#,
(),
)
.unwrap();
let result: i64 = db.query_one("SELECT compute_sum(10, 20)", ()).unwrap();
assert_eq!(result, 30);
}
#[test]
fn test_python_function_with_different_types() {
let db = Database::open("memory://python_types_test").unwrap();
db.execute(
r#"
CREATE FUNCTION get_answer() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 42'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION get_message() RETURNS TEXT
LANGUAGE PYTHON AS 'return "Hello from Python"'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION get_truth() RETURNS BOOLEAN
LANGUAGE PYTHON AS 'return True'
"#,
(),
)
.unwrap();
let int_result: i64 = db.query_one("SELECT get_answer()", ()).unwrap();
assert_eq!(int_result, 42);
let text_result: String = db.query_one("SELECT get_message()", ()).unwrap();
assert_eq!(text_result, "Hello from Python");
let bool_result: bool = db.query_one("SELECT get_truth()", ()).unwrap();
assert!(bool_result);
}
#[test]
fn test_python_function_with_multiple_arguments() {
let db = Database::open("memory://python_multi_args_test").unwrap();
db.execute(r#"
CREATE FUNCTION format_person(name TEXT, age INTEGER, active BOOLEAN) RETURNS TEXT
LANGUAGE PYTHON AS 'return f"{arguments[0]} is {arguments[1]} years old, active: {arguments[2]}"'
"#, ()).unwrap();
let result: String = db
.query_one("SELECT format_person('Alice', 30, true)", ())
.unwrap();
assert_eq!(result, "Alice is 30 years old, active: True");
}
#[test]
fn test_python_function_drop() {
let db = Database::open("memory://python_drop_test").unwrap();
db.execute(
r#"
CREATE FUNCTION temp_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 123'
"#,
(),
)
.unwrap();
let result: i64 = db.query_one("SELECT temp_func()", ()).unwrap();
assert_eq!(result, 123);
db.execute("DROP FUNCTION temp_func", ()).unwrap();
let result: Result<i64, _> = db.query_one("SELECT temp_func()", ());
assert!(result.is_err());
}
#[test]
fn test_python_function_without_return() {
let db = Database::open("memory://python_no_result_test").unwrap();
db.execute(
r#"
CREATE FUNCTION no_return_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'x = 42'
"#,
(),
)
.unwrap();
let result: Option<i64> = db.query_one("SELECT no_return_func()", ()).unwrap();
assert!(result.is_none());
}
#[test]
fn test_python_string_manipulation() {
let db = Database::open("memory://python_string_test").unwrap();
db.execute(
r#"
CREATE FUNCTION greet(name TEXT)
RETURNS TEXT
LANGUAGE PYTHON AS 'return f"Hello, {arguments[0]}!"'
"#,
(),
)
.unwrap();
let result: String = db.query_one("SELECT greet('World')", ()).unwrap();
assert_eq!(result, "Hello, World!");
}
#[test]
fn test_python_math_operations() {
let db = Database::open("memory://python_math_test").unwrap();
db.execute(
r#"
CREATE FUNCTION power(base INTEGER, exp INTEGER)
RETURNS INTEGER
LANGUAGE PYTHON AS 'return arguments[0] ** arguments[1]'
"#,
(),
)
.unwrap();
let result: i64 = db.query_one("SELECT power(2, 3)", ()).unwrap();
assert_eq!(result, 8);
}
#[test]
fn test_python_type_conversion() {
let db = Database::open("memory://python_types_test").unwrap();
db.execute(
r#"
CREATE FUNCTION double_int(x INTEGER)
RETURNS INTEGER
LANGUAGE PYTHON AS 'return x * 2'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION double_float(x FLOAT)
RETURNS FLOAT
LANGUAGE PYTHON AS 'return arguments[0] * 2.0'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION negate_bool(x BOOLEAN)
RETURNS BOOLEAN
LANGUAGE PYTHON AS 'return not arguments[0]'
"#,
(),
)
.unwrap();
let int_result: i64 = db.query_one("SELECT double_int(21)", ()).unwrap();
assert_eq!(int_result, 42);
let float_result: f64 = db.query_one("SELECT double_float(3.14)", ()).unwrap();
assert!((float_result - TAU).abs() < 0.01);
let bool_result: bool = db.query_one("SELECT negate_bool(true)", ()).unwrap();
assert!(!bool_result);
}
#[test]
fn test_python_invalid_syntax() {
let db = Database::open("memory://python_invalid_test").unwrap();
db.execute(
r#"
CREATE FUNCTION invalid_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 1 +'
"#,
(),
)
.unwrap();
let result: Result<i64, _> = db.query_one("SELECT invalid_func()", ());
assert!(result.is_err());
}
#[test]
fn test_python_runtime_error() {
let db = Database::open("memory://python_runtime_error_test").unwrap();
db.execute(
r#"
CREATE FUNCTION error_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 1 / 0'
"#,
(),
)
.unwrap();
let result: Result<i64, _> = db.query_one("SELECT error_func()", ());
assert!(result.is_err());
let err_msg = result.unwrap_err().to_string();
assert!(
err_msg.contains("error")
|| err_msg.contains("Error")
|| err_msg.contains("ZeroDivisionError")
);
}
#[test]
fn test_python_syntax_validation() {
let db = Database::open("memory://python_syntax_test").unwrap();
let result = db.execute(
r#"
CREATE FUNCTION valid_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 1'
"#,
(),
);
assert!(result.is_ok());
db.execute(
r#"
CREATE FUNCTION invalid_func() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 1 + (invalid_syntax'
"#,
(),
)
.unwrap();
let result: Result<i64, _> = db.query_one("SELECT invalid_func()", ());
assert!(result.is_err());
}
#[test]
fn test_python_security_isolation() {
let db = Database::open("memory://python_security_test").unwrap();
let result = db.execute(
r#"
CREATE FUNCTION file_access() RETURNS TEXT
LANGUAGE PYTHON AS 'try:
with open("/dev/null", "r") as f: return "File access allowed"
except:
return "File access failed"'
"#,
(),
);
if result.is_ok() {
let exec_result: Result<String, _> = db.query_one("SELECT file_access()", ());
let _ = exec_result; }
let result = db.execute(
r#"
CREATE FUNCTION subprocess_test() RETURNS TEXT
LANGUAGE PYTHON AS 'try:
import subprocess
return "Subprocess available"
except:
return "Subprocess blocked"'
"#,
(),
);
if result.is_ok() {
let exec_result: Result<String, _> = db.query_one("SELECT subprocess_test()", ());
let _ = exec_result; }
}
#[test]
fn test_python_basic_execution() {
let db = Database::open("memory://python_exec_test").unwrap();
db.execute(
r#"
CREATE FUNCTION is_even(n INTEGER)
RETURNS BOOLEAN
LANGUAGE PYTHON AS 'return arguments[0] % 2 == 0'
"#,
(),
)
.unwrap();
let result: bool = db.query_one("SELECT is_even(4)", ()).unwrap();
assert!(result);
let result: bool = db.query_one("SELECT is_even(5)", ()).unwrap();
assert!(!result);
}
#[test]
fn test_python_argument_types() {
let db = Database::open("memory://python_args_test").unwrap();
db.execute(r#"
CREATE FUNCTION process_args(i INTEGER, f FLOAT, t TEXT, b BOOLEAN)
RETURNS TEXT
LANGUAGE PYTHON AS 'return f"{arguments[0]}, {arguments[1]}, {arguments[2]}, {arguments[3]}"'
"#, ()).unwrap();
let result: String = db
.query_one("SELECT process_args(42, 3.14, 'hello', true)", ())
.unwrap();
assert_eq!(result, "42, 3.14, hello, True");
}
#[test]
fn test_python_return_types() {
let db = Database::open("memory://python_return_test").unwrap();
db.execute(
r#"
CREATE FUNCTION return_int() RETURNS INTEGER
LANGUAGE PYTHON AS 'return 42'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION return_float() RETURNS FLOAT
LANGUAGE PYTHON AS 'return 3.14'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION return_text() RETURNS TEXT
LANGUAGE PYTHON AS 'return "hello"'
"#,
(),
)
.unwrap();
db.execute(
r#"
CREATE FUNCTION return_bool() RETURNS BOOLEAN
LANGUAGE PYTHON AS 'return True'
"#,
(),
)
.unwrap();
let int_result: i64 = db.query_one("SELECT return_int()", ()).unwrap();
assert_eq!(int_result, 42);
let float_result: f64 = db.query_one("SELECT return_float()", ()).unwrap();
assert!((float_result - PI).abs() < 0.01);
let text_result: String = db.query_one("SELECT return_text()", ()).unwrap();
assert_eq!(text_result, "hello");
let bool_result: bool = db.query_one("SELECT return_bool()", ()).unwrap();
assert!(bool_result);
}
}