coinex_swap_rest.rs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696
  1. use std::collections::BTreeMap;
  2. use std::error::Error;
  3. use std::time::{SystemTime, UNIX_EPOCH};
  4. use reqwest::header::{HeaderMap, HeaderValue};
  5. use hex;
  6. use reqwest::Client;
  7. use rust_decimal::Decimal;
  8. use rust_decimal_macros::dec;
  9. use serde_json::Value;
  10. use crate::http_tool::RestTool;
  11. use crate::response_base::ResponseData;
  12. use sha2::{Digest, Sha256};
  13. use tracing::{error};
  14. #[derive(Clone)]
  15. pub struct CoinexSwapRest {
  16. label: String,
  17. base_url: String,
  18. client: Client,
  19. /*******参数*/
  20. //登陆所需参数
  21. login_param: BTreeMap<String, String>,
  22. delays: Vec<i64>,
  23. max_delay: i64,
  24. avg_delay: Decimal,
  25. }
  26. impl CoinexSwapRest {
  27. /*******************************************************************************************************/
  28. /*****************************************获取一个对象****************************************************/
  29. /*******************************************************************************************************/
  30. pub fn new(login_param: BTreeMap<String, String>) -> CoinexSwapRest
  31. {
  32. return CoinexSwapRest::new_label("default-CoinexSwapRest".to_string(), login_param);
  33. }
  34. pub fn new_label(label: String, login_param: BTreeMap<String, String>) -> CoinexSwapRest
  35. {
  36. let base_url: String = String::from("https://api.coinex.com");
  37. /*****返回结构体*******/
  38. CoinexSwapRest {
  39. label,
  40. base_url,
  41. client: Client::new(),
  42. login_param,
  43. delays: vec![],
  44. max_delay: 0,
  45. avg_delay: dec!(0.0),
  46. }
  47. }
  48. /*******************************************************************************************************/
  49. /*****************************************rest请求函数********************************************************/
  50. /*******************************************************************************************************/
  51. //获取服务器当前时间
  52. pub async fn get_server_time(&mut self) -> ResponseData {
  53. let data = self.request("GET".to_string(),
  54. "/v2".to_string(),
  55. "/time".to_string(),
  56. false,
  57. None,
  58. None,
  59. ).await;
  60. data
  61. }
  62. //查询个人交易费率
  63. pub async fn wallet_fee(&mut self) -> ResponseData {
  64. let data = self.request("GET".to_string(),
  65. "/v2".to_string(),
  66. "/account/trade-fee-rate".to_string(),
  67. true,
  68. None,
  69. None,
  70. ).await;
  71. data
  72. }
  73. //查询合约账户
  74. pub async fn get_account(&mut self) -> ResponseData {
  75. let data = self.request("GET".to_string(),
  76. "/v2".to_string(),
  77. "/assets/futures/balance".to_string(),
  78. true,
  79. None,
  80. None,
  81. ).await;
  82. data
  83. }
  84. //查询现货账户
  85. pub async fn get_spot_account(&mut self) -> ResponseData {
  86. let data = self.request("GET".to_string(),
  87. "/v2".to_string(),
  88. "/assets/spot/balance".to_string(),
  89. true,
  90. None,
  91. None,
  92. ).await;
  93. data
  94. }
  95. //指定币对仓位列表
  96. pub async fn get_position(&mut self, market: String) -> ResponseData {
  97. let params = serde_json::json!({
  98. "market": market,
  99. "market_type": "FUTURES"
  100. });
  101. let data = self.request("GET".to_string(),
  102. "/v2".to_string(),
  103. "/futures/pending-position".to_string(),
  104. true,
  105. Some(params.to_string()),
  106. None,
  107. ).await;
  108. data
  109. }
  110. //用户仓位列表
  111. pub async fn get_user_position(&mut self) -> ResponseData {
  112. let params = serde_json::json!({
  113. "market_type": "FUTURES"
  114. });
  115. let data = self.request("GET".to_string(),
  116. "/v2".to_string(),
  117. "/futures/pending-position".to_string(),
  118. true,
  119. Some(params.to_string()),
  120. None,
  121. ).await;
  122. data
  123. }
  124. //获取所有合约交易行情统计 market 市场名列表,多个市场名之间使用英文","分隔,空字符串或不传表示查询全部市场,限制最多10个市场
  125. pub async fn get_ticker(&mut self, market: String) -> ResponseData {
  126. let params = serde_json::json!({
  127. "market": market
  128. });
  129. let data = self.request("GET".to_string(),
  130. "/v2".to_string(),
  131. "/futures/ticker".to_string(),
  132. false,
  133. Some(params.to_string()),
  134. None,
  135. ).await;
  136. data
  137. }
  138. //查询所有的合约信息
  139. pub async fn get_market_details(&mut self, market: String) -> ResponseData {
  140. let params = serde_json::json!({
  141. "market": market
  142. });
  143. let data = self.request("GET".to_string(),
  144. "/v2".to_string(),
  145. "/futures/market".to_string(),
  146. false,
  147. Some(params.to_string()),
  148. None,
  149. ).await;
  150. data
  151. }
  152. //查询单个订单详情 /spot/order-status?market=CETUSDT&order_id=13400
  153. pub async fn get_order_details(&mut self, order_id: String, market: String) -> ResponseData {
  154. let params = serde_json::json!({
  155. "market": market,
  156. "order_id": order_id
  157. });
  158. let data = self.request("GET".to_string(),
  159. "/v2".to_string(),
  160. "/futures/order-status".to_string(),
  161. true,
  162. Some(params.to_string()),
  163. None,
  164. ).await;
  165. data
  166. }
  167. //查询未完成合约订单 /futures/pending-order?market=CETUSDT&market_type=FUTURES&side=buy&page=1&limit=10
  168. pub async fn get_pending_order(&mut self, client_id: String) -> ResponseData {
  169. let params = serde_json::json!({
  170. "market_type": "FUTURES",
  171. "client_id": client_id,
  172. "limit": 10
  173. });
  174. let data = self.request("GET".to_string(),
  175. "/v2".to_string(),
  176. "/futures/pending-order".to_string(),
  177. true,
  178. Some(params.to_string()),
  179. None,
  180. ).await;
  181. data
  182. }
  183. pub async fn get_pending_orders(&mut self) -> ResponseData {
  184. let params = serde_json::json!({
  185. "market_type": "FUTURES",
  186. "limit": 100
  187. });
  188. let data = self.request("GET".to_string(),
  189. "/v2".to_string(),
  190. "/futures/pending-order".to_string(),
  191. true,
  192. Some(params.to_string()),
  193. None,
  194. ).await;
  195. data
  196. }
  197. pub async fn get_finished_orders(&mut self) -> ResponseData {
  198. let params = serde_json::json!({
  199. "market_type": "FUTURES",
  200. "limit": 100
  201. });
  202. let data = self.request("GET".to_string(),
  203. "/v2".to_string(),
  204. "/futures/finished-order".to_string(),
  205. true,
  206. Some(params.to_string()),
  207. None,
  208. ).await;
  209. data
  210. }
  211. //下单
  212. // coinex swap 平仓需考虑最小下单量 只能通过close_position和position_id来平仓
  213. pub async fn order(&mut self,
  214. market: String,
  215. pos_side: String,
  216. side: String,
  217. size: Decimal,
  218. price: Decimal,
  219. client_id: String,
  220. ) -> ResponseData
  221. {
  222. // 默认为限价单
  223. let mut type_y = "limit".to_string();
  224. // 0为市价单,
  225. if price == Decimal::ZERO {
  226. type_y = "market".to_string();
  227. }
  228. let data;
  229. match format!("{}_{}", pos_side, side).as_str() {
  230. "long_buy" => {//开多
  231. data = self.swap_order(market, side, type_y, size, price, client_id, false).await;
  232. }
  233. "long_sell" => {//平多
  234. data = self.close_position(market, type_y, price, client_id, false).await;
  235. }
  236. "short_buy" => {//平空
  237. data = self.close_position(market, type_y, price, client_id, false).await;
  238. }
  239. "short_sell" => {//开空
  240. data = self.swap_order(market, side, type_y, size, price, client_id, false).await;
  241. }
  242. _ => {
  243. // 处理未知请求类型
  244. error!("下单失败,数量异常! size: {}", size);
  245. data = ResponseData::error(self.label.clone(), format!("下单失败, 下单参数: <market: {:?}, pos_side: {:?}, side: {:?}, size: {}, price: {:?}, client_id: {:?}>", market, pos_side, side, size, price, client_id));
  246. }
  247. };
  248. data
  249. }
  250. // 平仓下单
  251. pub async fn close_position(&mut self, market: String, type_y: String, price: Decimal, client_id: String, is_hide: bool) -> ResponseData {
  252. // 数量不传为全平
  253. let param = serde_json::json!({
  254. "market":market,
  255. "market_type": "FUTURES",
  256. "type": type_y,
  257. "price":price,
  258. "client_id":client_id,
  259. "is_hide": is_hide
  260. });
  261. let data = self.request("POST".to_string(),
  262. "/v2".to_string(),
  263. "/futures/close-position".to_string(),
  264. true,
  265. None,
  266. Some(param.to_string()),
  267. ).await;
  268. data
  269. }
  270. //合约交易开仓下单
  271. pub async fn swap_order(&mut self, market: String, side: String, type_y: String, amount: Decimal, price: Decimal, client_id: String, is_hide: bool) -> ResponseData {
  272. let param = serde_json::json!({
  273. "market":market,
  274. "market_type": "FUTURES",
  275. "side": side,
  276. "type": type_y,
  277. "amount":amount,
  278. "price":price,
  279. "client_id":client_id,
  280. "is_hide": is_hide
  281. });
  282. let data = self.request("POST".to_string(),
  283. "/v2".to_string(),
  284. "/futures/order".to_string(),
  285. true,
  286. None,
  287. Some(param.to_string()),
  288. ).await;
  289. data
  290. }
  291. //设置持仓模式
  292. pub async fn setting_dual_mode(&mut self) -> ResponseData {
  293. ResponseData::error(self.label.clone(), "设置双向持仓失败, coinex没有设置双向持仓".to_string())
  294. }
  295. //更新双仓模式下的杠杆
  296. pub async fn setting_dual_leverage(&mut self, market: String, leverage: i32) -> ResponseData {
  297. let params = serde_json::json!({
  298. "market": market,
  299. "market_type": "FUTURES",
  300. // cross: 全仓。全仓模式下,合约账户的全部可用余额都可用作当前全部仓位的共享保证金,系统会使用合约账户中的可用余额自动追加保证金,以避免仓位被强平
  301. //isolated: 逐仓。逐仓模式下,仓位保证金不会共享,单个仓位的保证金仅用于当前仓位,系统不会自动追加保证金,需要手动追加。
  302. "margin_mode": "cross",
  303. "leverage":leverage,
  304. });
  305. let data = self.request("POST".to_string(),
  306. "/v2".to_string(),
  307. "/futures/adjust-position-leverage".to_string(),
  308. true,
  309. None,
  310. Some(params.to_string()),
  311. ).await;
  312. data
  313. }
  314. //撤销单个订单
  315. pub async fn cancel_order(&mut self, market: String, order_id: &str, client_id: &str) -> ResponseData {
  316. if order_id != "" { // 如果真实订单id不为空,则用真实订单id取消订单
  317. let id = order_id.parse::<i64>().unwrap();
  318. let params = serde_json::json!({
  319. "market": market,
  320. "market_type": "FUTURES",
  321. "order_id": id
  322. });
  323. let data = self.request("POST".to_string(),
  324. "/v2".to_string(),
  325. "/futures/cancel-order".to_string(),
  326. true,
  327. None,
  328. Some(params.to_string()),
  329. ).await;
  330. data
  331. } else if client_id != "" { // 如果客户端id不为空,则用客户端id取消订单
  332. let params = serde_json::json!({
  333. "market": market,
  334. "market_type": "FUTURES",
  335. "client_id": client_id
  336. });
  337. let mut data = self.request("POST".to_string(),
  338. "/v2".to_string(),
  339. "/futures/cancel-order-by-client-id".to_string(),
  340. true,
  341. None,
  342. Some(params.to_string()),
  343. ).await;
  344. // 非空的
  345. if data.code == 200 && !data.data.is_null() {
  346. data.data = data.data.as_array().unwrap()[0]["data"].clone();
  347. }
  348. data
  349. } else {
  350. // 否则返回错误
  351. error!("取消订单失败失败,id异常");
  352. ResponseData::error(self.label.clone(), format!("取消订单失败失败, orderId:{:?}, clientId: {:?} ", order_id, client_id))
  353. }
  354. }
  355. // 撤销所有挂单
  356. pub async fn cancel_order_all(&mut self, market: String) -> ResponseData {
  357. let params = serde_json::json!({
  358. "market": market,
  359. "market_type": "FUTURES"
  360. });
  361. let data = self.request("POST".to_string(),
  362. "/v2".to_string(),
  363. "/futures/cancel-all-order".to_string(),
  364. true,
  365. None,
  366. Some(params.to_string()),
  367. ).await;
  368. data
  369. }
  370. //查询个人成交记录
  371. pub async fn my_trades(&mut self, market: String, limit: i64) -> ResponseData {
  372. let mut params = serde_json::json!({
  373. "market": market,
  374. "market_type": "FUTURES",
  375. "limit": 1000
  376. });
  377. if limit > 0 {
  378. params["limit"] = serde_json::json!(limit);
  379. }
  380. let data = self.request("GET".to_string(),
  381. "/v2".to_string(),
  382. "/futures/user-deals".to_string(),
  383. true,
  384. Some(params.to_string()), None).await;
  385. data
  386. }
  387. //查询合约账户变更历史
  388. pub async fn account_book(&mut self) -> ResponseData {
  389. error!("查询合约账户变更历史失败,无实现");
  390. ResponseData::error(self.label.clone(), "查询合约账户变更历史失败,接口没实现".to_string())
  391. }
  392. //查询子账号列表
  393. pub async fn account_get(&mut self) -> ResponseData {
  394. let params = serde_json::json!({
  395. "is_frozen":false,
  396. "page":1,
  397. "limit":100,
  398. });
  399. let data = self.request("GET".to_string(),
  400. "/v2".to_string(),
  401. "/account/subs".to_string(),
  402. true,
  403. Some(params.to_string()), None).await;
  404. data
  405. }
  406. //根据子账号,生成子账号 APIKEY
  407. pub async fn account_subs_api(&mut self, params: Value) -> ResponseData {
  408. let data = self.request("POST".to_string(),
  409. "/v2".to_string(),
  410. "/account/subs/api".to_string(),
  411. true,
  412. None,
  413. Some(params.to_string())).await;
  414. data
  415. }
  416. //获取子账号 APIKEY 列表
  417. pub async fn account_get_apikey(&mut self, params: Value) -> ResponseData {
  418. let data = self.request("GET".to_string(),
  419. "/v2".to_string(),
  420. "/account/subs/api".to_string(),
  421. true,
  422. Some(params.to_string()), None).await;
  423. data
  424. }
  425. //获取子账号 APIKEY 详情
  426. pub async fn account_get_detail(&mut self, params: Value) -> ResponseData {
  427. let data = self.request("GET".to_string(),
  428. "/v2".to_string(),
  429. "/account/subs/api-detail".to_string(),
  430. true,
  431. Some(params.to_string()), None).await;
  432. data
  433. }
  434. //编辑子账号 APIKEY
  435. pub async fn account_get_update(&mut self, params: Value) -> ResponseData {
  436. let data = self.request("POST".to_string(),
  437. "/v2".to_string(),
  438. "/account/subs/edit-api".to_string(),
  439. true,
  440. None, Some(params.to_string())).await;
  441. data
  442. }
  443. //删除子账号 APIKEY
  444. pub async fn account_del_apikey(&mut self, params: Value) -> ResponseData {
  445. let data = self.request("POST".to_string(),
  446. "/v2".to_string(),
  447. "/account/subs/delete-api".to_string(),
  448. true,
  449. None,
  450. Some(params.to_string())).await;
  451. data
  452. }
  453. /*******************************************************************************************************/
  454. /*****************************************工具函数********************************************************/
  455. /*******************************************************************************************************/
  456. pub fn get_delays(&self) -> Vec<i64> {
  457. self.delays.clone()
  458. }
  459. pub fn get_avg_delay(&self) -> Decimal {
  460. self.avg_delay.clone()
  461. }
  462. pub fn get_max_delay(&self) -> i64 {
  463. self.max_delay.clone()
  464. }
  465. // fn get_delay_info(&mut self) {
  466. // let last_100 = if self.delays.len() > 100 {
  467. // self.delays[self.delays.len() - 100..].to_vec()
  468. // } else {
  469. // self.delays.clone()
  470. // };
  471. //
  472. // let max_value = last_100.iter().max().unwrap();
  473. // if max_value.clone().to_owned() > self.max_delay {
  474. // self.max_delay = max_value.clone().to_owned();
  475. // }
  476. //
  477. // let sum: i64 = last_100.iter().sum();
  478. // let sum_v = Decimal::from_i64(sum).unwrap();
  479. // let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  480. // self.avg_delay = (sum_v / len_v).round_dp(1);
  481. // self.delays = last_100.clone().into_iter().collect();
  482. // }
  483. //调用请求
  484. async fn request(&mut self,
  485. request_type: String,
  486. prefix_url: String,
  487. request_url: String,
  488. is_login: bool,
  489. params: Option<String>,
  490. body: Option<String>) -> ResponseData
  491. {
  492. // trace!("login_param:{:?}", self.login_param);
  493. //解析账号信息
  494. let mut access_key = "".to_string();
  495. let mut secret_key = "".to_string();
  496. if self.login_param.contains_key("access_key") {
  497. access_key = self.login_param.get("access_key").unwrap().to_string();
  498. }
  499. if self.login_param.contains_key("secret_key") {
  500. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  501. }
  502. let mut is_login_param = true;
  503. if access_key == "" || secret_key == "" {
  504. is_login_param = false
  505. }
  506. let timestamp = SystemTime::now().duration_since(UNIX_EPOCH).unwrap().as_millis().to_string();
  507. // url
  508. let mut url_and_query = format!("{}{}", prefix_url.clone(), request_url.clone());
  509. let mut headers = HeaderMap::new();
  510. headers.insert("Content-Type", HeaderValue::from_static("application/json"));
  511. headers.insert(
  512. "X-COINEX-KEY",
  513. HeaderValue::from_str(&self.login_param.get("access_key").unwrap()).unwrap(),
  514. );
  515. headers.insert(
  516. "X-COINEX-TIMESTAMP",
  517. HeaderValue::from_str(&timestamp).unwrap(),
  518. );
  519. if let Some(params) = params {
  520. let query = RestTool::parse_params_to_str(params);
  521. url_and_query = format!("{}?{}", url_and_query, query);
  522. }
  523. let body_s = if let Some(body) = body {
  524. body
  525. } else {
  526. "".to_string()
  527. };
  528. //是否需要登陆-- 组装sing
  529. if is_login {
  530. if !is_login_param {
  531. let e = ResponseData::error(self.label.clone(), "登陆参数错误!".to_string());
  532. return e;
  533. } else {//需要登陆-且登陆参数齐全
  534. //组装sing
  535. let sing = Self::sign(
  536. &request_type,
  537. &url_and_query,
  538. &body_s,
  539. timestamp.clone(),
  540. &secret_key,
  541. );
  542. // trace!("sing:{}", sing);
  543. //组装header
  544. headers.insert("X-COINEX-SIGN", HeaderValue::from_str(&sing.unwrap()).unwrap());
  545. }
  546. }
  547. let start_time = chrono::Utc::now().timestamp_millis();
  548. let response = self.http_toll(
  549. url_and_query,
  550. request_type,
  551. body_s.clone(),
  552. headers,
  553. ).await;
  554. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  555. self.delays.push(time_array);
  556. // self.get_delay_info();
  557. response
  558. }
  559. fn sign(
  560. method: &String,
  561. path: &String,
  562. body: &String,
  563. timestamp: String,
  564. secret_key: &String,
  565. ) -> Result<String, Box<dyn Error>> {
  566. let prepared_str = format!(
  567. "{}{}{}{}{}",
  568. method, path, body, timestamp, secret_key
  569. );
  570. let hash = Sha256::digest(prepared_str.as_bytes());
  571. Ok(hex::encode(hash))
  572. }
  573. async fn http_toll(&mut self, request_path: String, request_type: String, body: String, headers: HeaderMap) -> ResponseData {
  574. /****请求接口与 地址*/
  575. let url = format!("{}{}", self.base_url.to_string(), request_path);
  576. let request_type = request_type.clone().to_uppercase();
  577. let request_builder = match request_type.as_str() {
  578. "GET" => self.client.get(&url).headers(headers),
  579. "POST" => self.client.post(&url).body(body.clone()).headers(headers),
  580. "DELETE" => self.client.delete(&url).headers(headers),
  581. // "PUT" => self.client.put(url.clone()).json(&params),
  582. _ => {
  583. panic!("{}", format!("错误的请求类型:{}", request_type.clone()))
  584. }
  585. };
  586. // 读取响应的内容
  587. let res = request_builder.send().await;
  588. match res {
  589. Ok(response) => {
  590. let is_success = response.status().is_success(); // 先检查状态码
  591. let text_result = response.text().await;
  592. match text_result {
  593. Ok(text) => {
  594. let data_json_str: Result<Value, serde_json::Error> = serde_json::from_str(text.as_str());
  595. match data_json_str {
  596. Ok(data_json) => {
  597. return if is_success && data_json["code"].to_string() == "0" {
  598. self.on_success_data(data_json["data"].clone())
  599. } else {
  600. self.on_error_data(&text, &url, &body)
  601. };
  602. }
  603. Err(e) => {
  604. error!("{} 请求完成,解析响应内容JSON失败 {} {}", url, text.as_str(), e);
  605. self.on_error_data(&e.to_string(), &url, &body)
  606. }
  607. }
  608. }
  609. Err(e) => {
  610. error!("{} 请求完成,解析响应内容失败 {}", url, e);
  611. self.on_error_data(&e.to_string(), &url, &body)
  612. }
  613. }
  614. }
  615. Err(e) => {
  616. // 异常情况
  617. error!("{} 请求失败,网络错误 {}", url, e);
  618. self.on_error_data(&e.to_string(), &url, &body)
  619. }
  620. }
  621. }
  622. pub fn on_success_data(&mut self, text: Value) -> ResponseData {
  623. ResponseData::new(self.label.clone(),
  624. 200,
  625. "success".to_string(),
  626. text)
  627. }
  628. pub fn on_error_data(&mut self, text: &String, base_url: &String, params: &String) -> ResponseData {
  629. let json_value = serde_json::from_str::<Value>(&text);
  630. match json_value {
  631. Ok(data) => {
  632. let message;
  633. if !data["message"].is_null() {
  634. message = format!("{}:{}", data["code"].to_string(), data["message"].as_str().unwrap());
  635. } else {
  636. message = data["code"].to_string();
  637. }
  638. let mut error = ResponseData::error(self.label.clone(), message);
  639. error.message = format!("请求地址:{}, 请求参数:{}, 报错内容:{}。", base_url, params, error.message);
  640. error
  641. }
  642. Err(e) => {
  643. error!("解析错误:{:?}", e);
  644. let error = ResponseData::error("".to_string(), format!("json 解析失败:{},相关参数:{}", e, text));
  645. error
  646. }
  647. }
  648. }
  649. }