gate_swap_rest.rs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. use std::collections::BTreeMap;
  2. use reqwest::header::HeaderMap;
  3. use ring::{digest};
  4. use hex;
  5. use hmac::{Hmac, Mac, NewMac};
  6. use reqwest::Client;
  7. use rust_decimal::Decimal;
  8. use rust_decimal::prelude::FromPrimitive;
  9. use rust_decimal_macros::dec;
  10. use serde_json::Value;
  11. use crate::http_tool::RestTool;
  12. use crate::response_base::ResponseData;
  13. use sha2::Sha512;
  14. use tracing::{error, info, trace};
  15. #[derive(Clone)]
  16. pub struct GateSwapRest {
  17. label: String,
  18. base_url: String,
  19. client: reqwest::Client,
  20. /*******参数*/
  21. //登陆所需参数
  22. login_param: BTreeMap<String, String>,
  23. delays: Vec<i64>,
  24. max_delay: i64,
  25. avg_delay: Decimal,
  26. }
  27. impl GateSwapRest {
  28. /*******************************************************************************************************/
  29. /*****************************************获取一个对象****************************************************/
  30. /*******************************************************************************************************/
  31. pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> GateSwapRest
  32. {
  33. return GateSwapRest::new_label("default-GateSwapRest".to_string(), is_colo, login_param);
  34. }
  35. pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> GateSwapRest
  36. {
  37. let base_url = if is_colo {
  38. // "https://apiv4-private.gateapi.io".to_string()
  39. "https://api.gateio.ws".to_string()
  40. } else {
  41. "https://api.gateio.ws".to_string()
  42. };
  43. if is_colo {
  44. info!("开启高速(未配置,走普通:{})通道",base_url);
  45. } else {
  46. info!("走普通通道:{}",base_url);
  47. }
  48. /*****返回结构体*******/
  49. GateSwapRest {
  50. label,
  51. base_url: base_url.to_string(),
  52. client: Client::new(),
  53. login_param,
  54. delays: vec![],
  55. max_delay: 0,
  56. avg_delay: dec!(0.0),
  57. }
  58. }
  59. /*******************************************************************************************************/
  60. /*****************************************rest请求函数********************************************************/
  61. /*******************************************************************************************************/
  62. //获取服务器当前时间
  63. pub async fn get_server_time(&mut self) -> ResponseData {
  64. let params = serde_json::json!({
  65. });
  66. let data = self.request("GET".to_string(),
  67. "/api/v4".to_string(),
  68. format!("/spot/time"),
  69. true,
  70. params.to_string(),
  71. ).await;
  72. data
  73. }
  74. //查询个人交易费率
  75. pub async fn wallet_fee(&mut self) -> ResponseData {
  76. let params = serde_json::json!({
  77. });
  78. let data = self.request("GET".to_string(),
  79. "/api/v4".to_string(),
  80. format!("/wallet/fee"),
  81. true,
  82. params.to_string(),
  83. ).await;
  84. data
  85. }
  86. //查询合约账户
  87. pub async fn get_account(&mut self, settle: String) -> ResponseData {
  88. let params = serde_json::json!({
  89. });
  90. let data = self.request("GET".to_string(),
  91. "/api/v4".to_string(),
  92. format!("/futures/{}/accounts", settle),
  93. true,
  94. params.to_string(),
  95. ).await;
  96. data
  97. }
  98. //用户仓位列表
  99. pub async fn get_user_position(&mut self, settle: String) -> ResponseData {
  100. let params = serde_json::json!({
  101. });
  102. let data = self.request("GET".to_string(),
  103. "/api/v4".to_string(),
  104. format!("/futures/{}/positions", settle),
  105. true,
  106. params.to_string(),
  107. ).await;
  108. data
  109. }
  110. //双仓模式下的持仓信息
  111. pub async fn get_position(&mut self, settle: String, contract: String) -> ResponseData {
  112. let params = serde_json::json!({
  113. });
  114. let data = self.request("GET".to_string(),
  115. "/api/v4".to_string(),
  116. format!("/futures/{}/dual_comp/positions/{}", settle, contract),
  117. true,
  118. params.to_string(),
  119. ).await;
  120. data
  121. }
  122. //获取所有合约交易行情统计
  123. pub async fn get_ticker(&mut self, settle: String) -> ResponseData {
  124. let params = serde_json::json!({
  125. });
  126. let data = self.request("GET".to_string(),
  127. "/api/v4".to_string(),
  128. format!("/futures/{}/tickers", settle),
  129. true,
  130. params.to_string(),
  131. ).await;
  132. data
  133. }
  134. //查询所有的合约信息
  135. pub async fn get_market_details(&mut self, settle: String) -> ResponseData {
  136. let params = serde_json::json!({
  137. });
  138. let data = self.request("GET".to_string(),
  139. "/api/v4".to_string(),
  140. format!("/futures/{}/contracts", settle),
  141. true,
  142. params.to_string(),
  143. ).await;
  144. data
  145. }
  146. //查询单个订单详情
  147. pub async fn get_order_details(&mut self, settle: String, order_id: String) -> ResponseData {
  148. let params = serde_json::json!({
  149. });
  150. let data = self.request("GET".to_string(),
  151. "/api/v4".to_string(),
  152. format!("/futures/{}/orders/{}", settle, order_id),
  153. true,
  154. params.to_string(),
  155. ).await;
  156. data
  157. }
  158. //查询合约订单列表
  159. pub async fn get_orders(&mut self, settle: String, status: String) -> ResponseData {
  160. let params = serde_json::json!({
  161. "status":status
  162. });
  163. let data = self.request("GET".to_string(),
  164. "/api/v4".to_string(),
  165. format!("/futures/{}/orders", settle),
  166. true,
  167. params.to_string(),
  168. ).await;
  169. data
  170. }
  171. //下单:side-下单方向,pos_side-持仓方向
  172. pub async fn order(&mut self,
  173. settle: String,
  174. pos_side: String,
  175. side: String,
  176. contract: String,
  177. size: i64,
  178. price: String,
  179. text: String,
  180. ) -> ResponseData
  181. {
  182. if side != "buy" && side != "sell" {
  183. ResponseData::error(self.label.clone(), format!("未知下单方向!{}", side));
  184. }
  185. if pos_side != "long" && pos_side != "short" {
  186. ResponseData::error(self.label.clone(), format!("未知持仓方向!{}", side));
  187. }
  188. let mut param = serde_json::json!({
  189. "contract":contract, //合约标识
  190. "size":size,
  191. "price":price,
  192. "text":text,
  193. });
  194. if price == "0" {
  195. param.as_object_mut().unwrap().insert("tif".to_string(), serde_json::json!("ioc"));
  196. }
  197. if size == 0 { //数量为0则平仓
  198. param.as_object_mut().unwrap().insert("close".to_string(), serde_json::json!(true));
  199. }
  200. match format!("{}_{}", pos_side, side).as_str() {
  201. "long_buy" => {//开多
  202. param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(false));
  203. }
  204. "long_sell" => {//平多
  205. param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(true));
  206. }
  207. "short_buy" => {//平空
  208. param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(true));
  209. }
  210. "short_sell" => {//开空
  211. param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(false));
  212. }
  213. _ => {} // 处理未知请求类型
  214. };
  215. trace!("----param{}", param.to_string());
  216. let data = self.swap_order(settle, param).await;
  217. data
  218. }
  219. //合约交易下单
  220. pub async fn swap_order(&mut self, settle: String, params: serde_json::Value) -> ResponseData {
  221. let data = self.request("POST".to_string(),
  222. "/api/v4".to_string(),
  223. format!("/futures/{}/orders", settle),
  224. true,
  225. params.to_string(),
  226. ).await;
  227. data
  228. }
  229. //设置持仓模式
  230. pub async fn setting_dual_mode(&mut self, settle: String, dual_mode: bool) -> ResponseData {
  231. let params = serde_json::json!({
  232. "dual_mode":dual_mode,
  233. });
  234. let data = self.request("POST".to_string(),
  235. "/api/v4".to_string(),
  236. format!("/futures/{}/dual_mode", settle),
  237. true,
  238. params.to_string(),
  239. ).await;
  240. data
  241. }
  242. //更新双仓模式下的杠杆
  243. pub async fn setting_dual_leverage(&mut self, settle: String, symbol: String, leverage: String) -> ResponseData {
  244. let params = serde_json::json!({
  245. "leverage":leverage,
  246. });
  247. let data = self.request("POST".to_string(),
  248. "/api/v4".to_string(),
  249. format!("/futures/{}/dual_comp/positions/{}/leverage", settle, symbol),
  250. true,
  251. params.to_string(),
  252. ).await;
  253. data
  254. }
  255. //交易账户互转
  256. pub async fn wallet_transfers(&mut self, currency: String, from: String, to: String, amount: String, settle: String) -> ResponseData {
  257. let params = serde_json::json!({
  258. "currency":currency,
  259. "from":from,
  260. "to":to,
  261. "amount":amount,
  262. "settle":settle,
  263. });
  264. let data = self.request("POST".to_string(),
  265. "/api/v4".to_string(),
  266. format!("/wallet/transfers"),
  267. true,
  268. params.to_string(),
  269. ).await;
  270. data
  271. }
  272. //撤销单个订单
  273. pub async fn cancel_order(&mut self, settle: String, order_id: String) -> ResponseData {
  274. let params = serde_json::json!({
  275. });
  276. let data = self.request("DELETE".to_string(),
  277. "/api/v4".to_string(),
  278. format!("/futures/{}/orders/{}", settle, order_id),
  279. true,
  280. params.to_string(),
  281. ).await;
  282. data
  283. }
  284. //弃用
  285. pub async fn swap_bazaar_order(&mut self, text: String, origin_side: String, settle: String, contract: String, size: i64) -> ResponseData {
  286. let mut reduce_only = false;
  287. let mut param = serde_json::json!({
  288. "text":text,
  289. "contract":contract,
  290. "price":"0",
  291. "size":size,
  292. });
  293. let req = match origin_side.as_str() {
  294. "kd" => {
  295. reduce_only = false;
  296. true
  297. }
  298. "pd" => {
  299. reduce_only = true;
  300. true
  301. }
  302. "kk" => {
  303. reduce_only = false;
  304. true
  305. }
  306. "pk" => {
  307. reduce_only = true;
  308. true
  309. }
  310. _ => { false } // 处理未知请求类型
  311. };
  312. if req {
  313. param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(reduce_only));
  314. }
  315. let data = self.swap_order(settle, param).await;
  316. data
  317. }
  318. //批量取消状态为 open 的订单
  319. pub async fn cancel_orders(&mut self, settle: String, contract: String) -> ResponseData {
  320. let params = serde_json::json!({
  321. "contract":contract
  322. });
  323. let data = self.request("DELETE".to_string(),
  324. "/api/v4".to_string(),
  325. format!("/futures/{}/orders", settle),
  326. true,
  327. params.to_string(),
  328. ).await;
  329. data
  330. }
  331. //查询个人成交记录
  332. pub async fn my_trades(&mut self, settle: String) -> ResponseData {
  333. let params = serde_json::json!({
  334. });
  335. let data = self.request("GET".to_string(),
  336. "/api/v4".to_string(),
  337. format!("/futures/{}/my_trades", settle),
  338. true,
  339. params.to_string(),
  340. ).await;
  341. data
  342. }
  343. //查询合约账户变更历史
  344. pub async fn account_book(&mut self, settle: String) -> ResponseData {
  345. let params = serde_json::json!({
  346. "limit":200
  347. });
  348. let data = self.request("GET".to_string(),
  349. "/api/v4".to_string(),
  350. format!("/futures/{}/account_book", settle),
  351. true,
  352. params.to_string(),
  353. ).await;
  354. data
  355. }
  356. /*******************************************************************************************************/
  357. /*****************************************工具函数********************************************************/
  358. /*******************************************************************************************************/
  359. pub fn get_delays(&self) -> Vec<i64> {
  360. self.delays.clone()
  361. }
  362. pub fn get_avg_delay(&self) -> Decimal {
  363. self.avg_delay.clone()
  364. }
  365. pub fn get_max_delay(&self) -> i64 {
  366. self.max_delay.clone()
  367. }
  368. fn get_delay_info(&mut self) {
  369. let last_100 = if self.delays.len() > 100 {
  370. self.delays[self.delays.len() - 100..].to_vec()
  371. } else {
  372. self.delays.clone()
  373. };
  374. let max_value = last_100.iter().max().unwrap();
  375. if max_value.clone().to_owned() > self.max_delay {
  376. self.max_delay = max_value.clone().to_owned();
  377. }
  378. let sum: i64 = last_100.iter().sum();
  379. let sum_v = Decimal::from_i64(sum).unwrap();
  380. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  381. self.avg_delay = (sum_v / len_v).round_dp(1);
  382. self.delays = last_100.clone().into_iter().collect();
  383. }
  384. //调用请求
  385. async fn request(&mut self,
  386. requesst_type: String,
  387. prefix_url: String,
  388. request_url: String,
  389. is_login: bool,
  390. params: String) -> ResponseData
  391. {
  392. // trace!("login_param:{:?}", self.login_param);
  393. //解析账号信息
  394. let mut access_key = "".to_string();
  395. let mut secret_key = "".to_string();
  396. if self.login_param.contains_key("access_key") {
  397. access_key = self.login_param.get("access_key").unwrap().to_string();
  398. }
  399. if self.login_param.contains_key("secret_key") {
  400. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  401. }
  402. let mut is_login_param = true;
  403. if access_key == "" || secret_key == "" {
  404. is_login_param = false
  405. }
  406. //请求头配置-如果需要登陆则存在额外配置
  407. let mut body = "".to_string();
  408. let timestamp = chrono::Utc::now().timestamp().to_string();
  409. let mut headers = HeaderMap::new();
  410. if requesst_type == "GET" {
  411. headers.insert("Content-type", "application/x-www-form-urlencoded".parse().unwrap());
  412. headers.insert("User-Agent", "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) ".parse().unwrap());
  413. } else {
  414. headers.insert("Accept", "application/json".parse().unwrap());
  415. headers.insert("Content-Type", "application/json".parse().unwrap());
  416. }
  417. if requesst_type == "POST" {
  418. body = params.clone();
  419. }
  420. //是否需要登陆-- 组装sing
  421. if is_login {
  422. if !is_login_param {
  423. let e = ResponseData::error(self.label.clone(), "登陆参数错误!".to_string());
  424. return e;
  425. } else {//需要登陆-且登陆参数齐全
  426. //组装sing
  427. let sing = Self::sign(secret_key.clone(),
  428. requesst_type.clone(),
  429. prefix_url.clone(),
  430. request_url.clone(),
  431. params.clone(),
  432. body.clone(),
  433. timestamp.clone(),
  434. );
  435. // trace!("sing:{}", sing);
  436. //组装header
  437. headers.extend(Self::headers(access_key, timestamp, sing));
  438. }
  439. }
  440. // trace!("headers:{:?}", headers);
  441. let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
  442. let start_time = chrono::Utc::now().timestamp_millis();
  443. let get_response = self.http_toll(
  444. base_url.clone(),
  445. requesst_type.to_string(),
  446. params.clone(),
  447. headers,
  448. ).await;
  449. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  450. self.delays.push(time_array);
  451. self.get_delay_info();
  452. let res_data = Self::res_data_analysis(get_response, base_url, params);
  453. res_data
  454. }
  455. pub fn headers(access_key: String, timestamp: String, sign: String) -> HeaderMap {
  456. let mut headers = HeaderMap::new();
  457. headers.insert("KEY", access_key.clone().parse().unwrap());
  458. headers.insert("Timestamp", timestamp.clone().parse().unwrap());
  459. headers.insert("SIGN", sign.clone().parse().unwrap());
  460. headers
  461. }
  462. pub fn sign(secret_key: String,
  463. requesst_type: String, prefix_url: String, request_url: String,
  464. params: String, body_data: String, timestamp: String) -> String
  465. {
  466. let url = format!("{}{}", prefix_url, request_url);
  467. let params_str = RestTool::parse_params_to_str(params);
  468. let body = Some(body_data);
  469. let hashed_payload = if let Some(body) = body {
  470. let mut m = digest::Context::new(&digest::SHA512);
  471. m.update(body.as_bytes());
  472. hex::encode(m.finish().as_ref())
  473. } else {
  474. String::new()
  475. };
  476. // trace!("hashed_payload:{}", hashed_payload);
  477. let message = format!("{}\n{}\n{}\n{}\n{}",
  478. requesst_type,
  479. url,
  480. params_str,
  481. hashed_payload,
  482. timestamp);
  483. // trace!("**********", );
  484. // trace!("组装数据:{}", message);
  485. // trace!("**********", );
  486. let mut mac = Hmac::<Sha512>::new_varkey(secret_key.as_bytes()).expect("Failed to create HMAC");
  487. mac.update(message.as_bytes());
  488. let result = mac.finalize().into_bytes();
  489. let sign = hex::encode(result);
  490. sign
  491. }
  492. async fn http_toll(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<ResponseData, reqwest::Error> {
  493. let res_data: ResponseData;
  494. /****请求接口与 地址*/
  495. let url = format!("{}{}", self.base_url.to_string(), request_path);
  496. let request_type = request_type.clone().to_uppercase();
  497. let addrs_url = format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()));
  498. // let params_json: serde_json::Value = serde_json::from_str(&params).unwrap();
  499. // trace!("url:{}",url);
  500. // trace!("addrs_url:{}",url);
  501. // trace!("params_json:{}",params_json);
  502. // trace!("headers:{:?}",headers);
  503. let req = match request_type.as_str() {
  504. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  505. "POST" => self.client.post(addrs_url.clone()).body(params).headers(headers),
  506. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  507. // "PUT" => self.client.put(url.clone()).json(&params),
  508. _ => return Ok(ResponseData::error(self.label.clone(), format!("错误的请求类型:{}", request_type.clone()))), // 处理未知请求类型
  509. };
  510. let response = req.send().await?;
  511. if response.status().is_success() {
  512. // 读取响应的内容
  513. let body = response.text().await?;
  514. // trace!("ok-----{}", body);
  515. res_data = ResponseData::new(self.label.clone(), "200".to_string(), "success".to_string(), body);
  516. } else {
  517. let body = response.text().await?;
  518. // trace!("error-----{}", body);
  519. res_data = ResponseData::error(self.label.clone(), body.to_string())
  520. }
  521. Ok(res_data)
  522. }
  523. //res_data 解析
  524. pub fn res_data_analysis(result: Result<ResponseData, reqwest::Error>, base_url: String, params: String) -> ResponseData {
  525. match result {
  526. Ok(res_data) => {
  527. if res_data.code != "200" {
  528. let message = res_data.message;
  529. // let json_value: serde_json::Value = serde_json::from_str(&message).unwrap();//这种方式会触发 解析错误
  530. let json_value = serde_json::from_str::<Value>(&message);
  531. match json_value {
  532. Ok(data) => {
  533. let label = data["label"].as_str().unwrap();
  534. let mut error = ResponseData::error(res_data.label, label.parse().unwrap());
  535. error.data = format!("请求地址:{},请求参数:{}", base_url, params);
  536. error
  537. }
  538. Err(e) => {
  539. error!("解析错误:{:?}",e);
  540. let error = ResponseData::error("".to_string(), format!("json 解析失败:{},相关参数:{}", e, message));
  541. error
  542. }
  543. }
  544. } else {
  545. res_data
  546. }
  547. }
  548. Err(err) => {
  549. let error = ResponseData::error("".to_string(), format!("json 解析失败:{},相关参数:{}", err, params));
  550. error
  551. }
  552. }
  553. }
  554. }