gate_swap_rest.rs 23 KB

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