From 27754bad4a938035c511d501d67de3d06d94346d Mon Sep 17 00:00:00 2001 From: asamaayako <2433047519@qq.com> Date: Tue, 13 Aug 2024 17:45:51 +0800 Subject: [PATCH] fix: try fix #816 json encodeSupport all json types. --- src/types/request.rs | 73 ++++++++++++++++++++++++++++++++------------ 1 file changed, 53 insertions(+), 20 deletions(-) diff --git a/src/types/request.rs b/src/types/request.rs index d5c4d9c57..8430de325 100644 --- a/src/types/request.rs +++ b/src/types/request.rs @@ -5,7 +5,7 @@ use actix_web::{ }; use futures_util::StreamExt as _; use log::debug; -use pyo3::types::{PyBytes, PyDict, PyString}; +use pyo3::types::{PyBytes, PyDict, PyList, PyString}; use pyo3::{exceptions::PyValueError, prelude::*}; use serde_json::Value; use std::collections::HashMap; @@ -140,8 +140,8 @@ impl Request { let body: Vec = if headers.contains(String::from("content-type")) && headers - .get(String::from("content-type")) - .is_some_and(|val| val.contains("multipart/form-data")) + .get(String::from("content-type")) + .is_some_and(|val| val.contains("multipart/form-data")) { let h = headers.get(String::from("content-type")).unwrap(); debug!("Content-Type: {:?}", h); @@ -256,25 +256,58 @@ impl PyRequest { pub fn json(&self, py: Python) -> PyResult { match self.body.as_ref(py).downcast::() { - Ok(python_string) => match serde_json::from_str(python_string.extract()?) { - Ok(Value::Object(map)) => { - let dict = PyDict::new(py); - - for (key, value) in map.iter() { - let py_key = key.to_string().into_py(py); - let py_value = match value { - Value::String(s) => s.as_str().into_py(py), - _ => value.to_string().into_py(py), - }; - - dict.set_item(py_key, py_value)?; - } + Ok(python_string) => json_string_to_py_object(py, python_string.extract()?), + Err(e) => Err(e.into()) + } + } +} - Ok(dict.into_py(py)) - } - _ => Err(PyValueError::new_err("Invalid JSON object")), +///serde_json not impl IntoPy but orphan principle. so use this function +fn json_string_to_pyobject(py: Python, json_string: &str) -> PyResult { + /// Converts a serde_json::Number to a Python object. + fn number_to_pyobject(num: serde_json::Number, py: Python) -> PyResult { + match num.as_u64() { + Some(u) => Ok(u.into_py(py)), + None => match num.as_i64() { + Some(i) => Ok(i.into_py(py)), + None => match num.as_f64() { + Some(f) => Ok(f.into_py(py)), + None => Err(PyErr::new::("Invalid number format")), + }, }, - Err(e) => Err(e.into()), } } + + /// Converts a serde_json::Value to a Python object. + fn value_to_pyobject(py: Python, value: serde_json::Value) -> PyResult { + match value { + Value::Null => Ok(().into_py(py)), + Value::Bool(b) => Ok(b.into_py(py)), + Value::Number(num) => number_to_pyobject(num, py), + Value::String(s) => { + Ok(s.into_py(py)) + } + Value::Array(array) => { + array.into_iter() + .map(|v| value_to_pyobject(py, v)) + .collect::>>() + .map(|list|PyList::new(py, list).into()) + } + Value::Object(map) => { + let dict = PyDict::new(py); + for (key, value) in map.into_iter() { + let py_key = key.to_string().into_py(py); + let py_value = value_to_pyobject(py, value)?; + dict.set_item(py_key, py_value)?; + } + Ok(dict.into_py(py)) + } + } + } + match serde_json::from_str(json_string) { + Ok(value) => value_to_pyobject(py, value), + _ => Err(PyValueError::new_err("json decode error")), + } } + +