nautilus_core/python/
params.rs1use pyo3::{
19 conversion::IntoPyObjectExt,
20 prelude::*,
21 types::{PyDict, PyList, PyModule},
22};
23use serde_json::Value;
24
25use crate::{
26 params::Params,
27 python::{serialization::from_pyobject_pyo3, to_pyvalue_err},
28};
29
30pub fn pydict_to_params(py: Python<'_>, dict: &Py<PyDict>) -> PyResult<Option<Params>> {
38 let dict_bound = dict.bind(py);
39 if dict_bound.is_empty() {
40 return Ok(None);
41 }
42
43 from_pyobject_pyo3(py, dict_bound.as_any()).map(Some)
44}
45
46pub fn value_to_pyobject(py: Python<'_>, val: &Value) -> PyResult<Py<PyAny>> {
55 match val {
56 Value::Null => Ok(py.None()),
57 Value::Bool(b) => b.into_py_any(py),
58 Value::String(s) => s.into_py_any(py),
59 Value::Number(n) => {
60 if n.is_i64() {
61 n.as_i64()
62 .ok_or_else(|| to_pyvalue_err("JSON number could not be read as i64"))?
63 .into_py_any(py)
64 } else if n.is_u64() {
65 n.as_u64()
66 .ok_or_else(|| to_pyvalue_err("JSON number could not be read as u64"))?
67 .into_py_any(py)
68 } else if n.is_f64() {
69 n.as_f64()
70 .ok_or_else(|| to_pyvalue_err("JSON number could not be read as f64"))?
71 .into_py_any(py)
72 } else {
73 Err(to_pyvalue_err("Unsupported JSON number type"))
74 }
75 }
76 Value::Array(arr) => {
77 let py_list = PyList::new(py, &[] as &[Py<PyAny>])?;
78 for item in arr {
79 let py_item = value_to_pyobject(py, item)?;
80 py_list.append(py_item)?;
81 }
82 py_list.into_py_any(py)
83 }
84 Value::Object(_) => {
85 let json_str = serde_json::to_string(val).map_err(to_pyvalue_err)?;
87 let py_dict: Py<PyDict> = PyModule::import(py, "json")?
88 .call_method("loads", (json_str,), None)?
89 .extract()?;
90 py_dict.into_py_any(py)
91 }
92 }
93}
94
95pub fn params_to_pydict(py: Python<'_>, params: &Params) -> PyResult<Py<PyDict>> {
101 let dict = PyDict::new(py);
102 for (key, value) in params {
103 let py_value = value_to_pyobject(py, value)?;
104 dict.set_item(key, py_value)?;
105 }
106 Ok(dict.into())
107}
108
109#[cfg(test)]
110mod tests {
111 use rstest::rstest;
112 use serde_json::json;
113
114 use super::*;
115
116 #[derive(Debug, Clone, Copy)]
117 enum ExpectedNumber {
118 I64(i64),
119 U64(u64),
120 F64(f64),
121 }
122
123 #[rstest]
124 #[case(json!(-100_i64), ExpectedNumber::I64(-100))]
125 #[case(json!(42_u64), ExpectedNumber::U64(42))]
126 #[case(json!(2.5_f64), ExpectedNumber::F64(2.5))]
127 fn test_value_to_pyobject_number_branches(
128 #[case] value: Value,
129 #[case] expected: ExpectedNumber,
130 ) {
131 Python::initialize();
132 Python::attach(|py| {
133 let py_obj = value_to_pyobject(py, &value).unwrap();
134
135 match expected {
136 ExpectedNumber::I64(expected) => {
137 assert_eq!(py_obj.extract::<i64>(py).unwrap(), expected);
138 }
139 ExpectedNumber::U64(expected) => {
140 assert_eq!(py_obj.extract::<u64>(py).unwrap(), expected);
141 }
142 ExpectedNumber::F64(expected) => {
143 let actual = py_obj.extract::<f64>(py).unwrap();
144 assert!((actual - expected).abs() < f64::EPSILON);
145 }
146 }
147 });
148 }
149}