无法将 JSON 转换为 Polars 数据框,’从非数组 JSON 读取数组’。

huangapple go评论90阅读模式
英文:

unable to convert json to polars dataframe,'read an Array from a non-Array JSON''

问题

以下是您要翻译的内容:

  1. 我在将从API提供的数据反序列化后的JSON转换为Polars DataFrame方面遇到了困难
  2. 以下是我的结构和定义的函数:
  3. // 用于期权价格查询的结构
  4. #[derive(Serialize, Deserialize, Debug)]
  5. pub struct OptionResponse {
  6. pub tables: Vec<OptionTable>,
  7. }
  8. #[derive(Serialize, Deserialize, Debug)]
  9. pub struct OptionTable {
  10. pub table: OptionDataTable,
  11. }
  12. #[derive(Serialize, Deserialize, Debug)]
  13. pub struct OptionDataTable {
  14. pub volume: Vec<f64>,
  15. pub open: Vec<f64>,
  16. pub close: Vec<f64>,
  17. pub high: Vec<f64>,
  18. pub low: Vec<f64>,
  19. }
  20. pub async fn ifind_option_read(option_code: &str, access_token: &str, start_time_str: &str, end_time_str: &str) -> Result<(), Box<dyn std::error::Error>> {
  21. let url = "https://quantapi.51ifind.com/api/v1/high_frequency";
  22. // 创建高频价格请求
  23. let request_data = maplit::hashmap! {
  24. "codes" => option_code,
  25. "indicators" => "volume,open,close,high,low",
  26. "starttime" => start_time_str,
  27. "endtime" => end_time_str,
  28. };
  29. let request_json = serde_json::to_string(&request_data)?;
  30. // 发送HTTP POST请求
  31. let client = reqwest::Client::new();
  32. let response = client
  33. .post(url)
  34. .header(reqwest::header::CONTENT_TYPE, "application/json")
  35. .header("access_token", access_token)
  36. .body(request_json)
  37. .send()
  38. .await?;
  39. let response_json: OptionResponse = response.json().await?;
  40. let response_body = &response_json.tables[0].table;
  41. println!("{:?}", &response_body);
  42. // 将response_json转换为字节数组
  43. let json_bytes = serde_json::to_vec(&response_body)?;
  44. // 通过JsonReader的方法将响应读入df
  45. let df = JsonReader::new(Cursor::new(json_bytes))
  46. .with_batch_size(1024)
  47. .infer_schema_len(None)
  48. .finish().unwrap();
  49. println!("{:?}", df);
  50. Ok(())
  51. }

返回的response_body看起来对我来说非常正常,类似于:

  1. OptionDataTable { volume: [0.0, 0.0, 0.0], open: [1263.0, 1263.0, 1263.0], close: [1263.0, 1263.0, 1263.0], high: [1263.0, 1263.0, 1263.0], low: [1263.0, 1263.0, 1263.0] }

尽管如此,JsonReader无法将其读取为DataFrame。这让我感到困惑。这是JSON格式的问题吗?

  1. <details>
  2. <summary>英文:</summary>
  3. i am struggling with conversion from JSON(deserialised from API-provicded data) to polars dataframe.
  4. below are my structs and defined function:
  5. ```rust
  6. // structs for option price query
  7. #[derive(Serialize, Deserialize, Debug)]
  8. pub struct OptionResponse {
  9. pub tables: Vec&lt;OptionTable&gt;,
  10. }
  11. #[derive(Serialize, Deserialize, Debug)]
  12. pub struct OptionTable {
  13. pub table: OptionDataTable,
  14. }
  15. #[derive(Serialize, Deserialize, Debug)]
  16. pub struct OptionDataTable {
  17. pub volume: Vec&lt;f64&gt;,
  18. pub open: Vec&lt;f64&gt;,
  19. pub close: Vec&lt;f64&gt;,
  20. pub high: Vec&lt;f64&gt;,
  21. pub low: Vec&lt;f64&gt;,
  22. }
  23. pub async fn ifind_option_read(option_code: &amp;str, access_token: &amp;str, start_time_str: &amp;str, end_time_str: &amp;str) -&gt; Result&lt;(), Box&lt;dyn std::error::Error&gt;&gt; {
  24. let url = &quot;https://quantapi.51ifind.com/api/v1/high_frequency&quot;;
  25. // create the high-frequency prices request
  26. let request_data = maplit::hashmap! {
  27. &quot;codes&quot; =&gt; option_code,
  28. &quot;indicators&quot; =&gt; &quot;volume,open,close,high,low&quot;,
  29. &quot;starttime&quot; =&gt; start_time_str,
  30. &quot;endtime&quot; =&gt; end_time_str,
  31. };
  32. let request_json = serde_json::to_string(&amp;request_data)?;
  33. // send the HTTP POST request
  34. let client = reqwest::Client::new();
  35. let response = client
  36. .post(url)
  37. .header(reqwest::header::CONTENT_TYPE, &quot;application/json&quot;)
  38. .header(&quot;access_token&quot;, access_token)
  39. .body(request_json)
  40. .send()
  41. .await?;
  42. let response_json: OptionResponse = response.json().await?;
  43. let response_body = &amp;response_json.tables[0].table;
  44. println!(&quot;{:?}&quot;, &amp;response_body);
  45. // Convert response_json to byte array
  46. let json_bytes = serde_json::to_vec(&amp;response_body)?;
  47. // read the response into df, per method of JsonReader
  48. let df = JsonReader::new(Cursor::new(json_bytes))
  49. .with_batch_size(1024)
  50. .infer_schema_len(None)
  51. .finish().unwrap();
  52. println!(&quot;{:?}&quot;, df);
  53. Ok(())
  54. }

the response_body returned looks very normal to me and looks like

  1. OptionDataTable { volume: [0.0, 0.0, 0.0], open: [1263.0, 1263.0, 1263.0], close: [1263.0, 1263.0, 1263.0], high: [1263.0, 1263.0, 1263.0], low: [1263.0, 1263.0, 1263.0] }

neverthless the JsonReader cant read it as a df.
this is confusing. is this a json format problem?

答案1

得分: 0

以前在polars/polars-json/src/json/deserialize.rs中的deserialize函数使用了todo!宏,导致了这个错误。

  1. pub fn deserialize(json: &BorrowedValue, data_type: DataType) -> Result<Box<dyn Array>, Error> {
  2. match json {
  3. BorrowedValue::Array(rows) => match data_type {
  4. DataType::LargeList(inner) => Ok(_deserialize(rows, inner.data_type)),
  5. _ => todo!("从非数组数据类型中读取数组"),
  6. },
  7. _ => todo!("从非数组JSON中读取数组"),
  8. }
  9. }

现在用Ok(_deserialize(&[json], data_type))替代了。

通过提交#207e0b9,现在一切都应该正常了。

英文:

previously the deserialize function in
polars/polars-json/src/json/deserialize.rs has a todo! macro causing this error.

  1. pub fn deserialize(json: &amp;BorrowedValue, data_type: DataType) -&gt; Result&lt;Box&lt;dyn Array&gt;, Error&gt; {
  2. match json {
  3. BorrowedValue::Array(rows) =&gt; match data_type {
  4. DataType::LargeList(inner) =&gt; Ok(_deserialize(rows, inner.data_type)),
  5. _ =&gt; todo!(&quot;read an Array from a non-Array data type&quot;),
  6. },
  7. _ =&gt; todo!(&quot;read an Array from a non-Array JSON&quot;),
  8. }
  9. }

now it is replaced with Ok(_deserialize(&amp;[json], data_type))

with commit #207e0b9 everything should be fine now.

huangapple
  • 本文由 发表于 2023年7月3日 18:36:36
  • 转载请务必保留本文链接:https://go.coder-hub.com/76603937.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定