gate_swap_rest.rs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615
  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, last_id: String) -> ResponseData {
  361. let mut params = serde_json::json!({
  362. "contract":contract,
  363. "last_id": last_id,
  364. "limit":1000
  365. });
  366. if limit > 0 {
  367. params["limit"] = serde_json::json!(limit);
  368. }
  369. let data = self.request("GET".to_string(),
  370. "/api/v4".to_string(),
  371. format!("/futures/{}/my_trades", settle),
  372. true,
  373. params.to_string(),
  374. ).await;
  375. data
  376. }
  377. //查询合约账户变更历史
  378. pub async fn account_book(&mut self, settle: String) -> ResponseData {
  379. let params = serde_json::json!({
  380. "limit":200
  381. });
  382. let data = self.request("GET".to_string(),
  383. "/api/v4".to_string(),
  384. format!("/futures/{}/account_book", settle),
  385. true,
  386. params.to_string(),
  387. ).await;
  388. data
  389. }
  390. /*******************************************************************************************************/
  391. /*****************************************工具函数********************************************************/
  392. /*******************************************************************************************************/
  393. pub fn get_delays(&self) -> Vec<i64> {
  394. self.delays.clone()
  395. }
  396. pub fn get_avg_delay(&self) -> Decimal {
  397. self.avg_delay.clone()
  398. }
  399. pub fn get_max_delay(&self) -> i64 {
  400. self.max_delay.clone()
  401. }
  402. fn get_delay_info(&mut self) {
  403. let last_100 = if self.delays.len() > 100 {
  404. self.delays[self.delays.len() - 100..].to_vec()
  405. } else {
  406. self.delays.clone()
  407. };
  408. let max_value = last_100.iter().max().unwrap();
  409. if max_value.clone().to_owned() > self.max_delay {
  410. self.max_delay = max_value.clone().to_owned();
  411. }
  412. let sum: i64 = last_100.iter().sum();
  413. let sum_v = Decimal::from_i64(sum).unwrap();
  414. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  415. self.avg_delay = (sum_v / len_v).round_dp(1);
  416. self.delays = last_100.clone().into_iter().collect();
  417. }
  418. //调用请求
  419. async fn request(&mut self,
  420. requesst_type: String,
  421. prefix_url: String,
  422. request_url: String,
  423. is_login: bool,
  424. params: String) -> ResponseData
  425. {
  426. // trace!("login_param:{:?}", self.login_param);
  427. //解析账号信息
  428. let mut access_key = "".to_string();
  429. let mut secret_key = "".to_string();
  430. if self.login_param.contains_key("access_key") {
  431. access_key = self.login_param.get("access_key").unwrap().to_string();
  432. }
  433. if self.login_param.contains_key("secret_key") {
  434. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  435. }
  436. let mut is_login_param = true;
  437. if access_key == "" || secret_key == "" {
  438. is_login_param = false
  439. }
  440. //请求头配置-如果需要登录则存在额外配置
  441. let mut body = "".to_string();
  442. let timestamp = chrono::Utc::now().timestamp().to_string();
  443. let mut headers = HeaderMap::new();
  444. if requesst_type == "GET" {
  445. headers.insert("Content-type", "application/x-www-form-urlencoded".parse().unwrap());
  446. headers.insert("User-Agent", "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) ".parse().unwrap());
  447. } else {
  448. headers.insert("Accept", "application/json".parse().unwrap());
  449. headers.insert("Content-Type", "application/json".parse().unwrap());
  450. }
  451. if requesst_type == "POST" {
  452. body = params.clone();
  453. }
  454. //是否需要登录-- 组装sing
  455. if is_login {
  456. if !is_login_param {
  457. let e = ResponseData::error(self.label.clone(), "登录参数错误!".to_string());
  458. return e;
  459. } else {//需要登录-且登录参数齐全
  460. //组装sing
  461. let sing = Self::sign(secret_key.clone(),
  462. requesst_type.clone(),
  463. prefix_url.clone(),
  464. request_url.clone(),
  465. params.clone(),
  466. body.clone(),
  467. timestamp.clone(),
  468. );
  469. // trace!("sing:{}", sing);
  470. //组装header
  471. headers.extend(Self::headers(access_key, timestamp, sing));
  472. }
  473. }
  474. // trace!("headers:{:?}", headers);
  475. let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
  476. let start_time = chrono::Utc::now().timestamp_millis();
  477. let response = self.http_tool(
  478. base_url.clone(),
  479. requesst_type.to_string(),
  480. params.clone(),
  481. headers,
  482. ).await;
  483. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  484. self.delays.push(time_array);
  485. self.get_delay_info();
  486. response
  487. }
  488. pub fn headers(access_key: String, timestamp: String, sign: String) -> HeaderMap {
  489. let mut headers = HeaderMap::new();
  490. headers.insert("KEY", access_key.clone().parse().unwrap());
  491. headers.insert("Timestamp", timestamp.clone().parse().unwrap());
  492. headers.insert("SIGN", sign.clone().parse().unwrap());
  493. headers
  494. }
  495. pub fn sign(secret_key: String,
  496. requesst_type: String, prefix_url: String, request_url: String,
  497. params: String, body_data: String, timestamp: String) -> String
  498. {
  499. let url = format!("{}{}", prefix_url, request_url);
  500. let params_str = RestTool::parse_params_to_str(params);
  501. let body = Some(body_data);
  502. let hashed_payload = if let Some(body) = body {
  503. let mut m = digest::Context::new(&digest::SHA512);
  504. m.update(body.as_bytes());
  505. hex::encode(m.finish().as_ref())
  506. } else {
  507. String::new()
  508. };
  509. // trace!("hashed_payload:{}", hashed_payload);
  510. let message = format!("{}\n{}\n{}\n{}\n{}",
  511. requesst_type,
  512. url,
  513. params_str,
  514. hashed_payload,
  515. timestamp);
  516. // trace!("**********", );
  517. // trace!("组装数据:{}", message);
  518. // trace!("**********", );
  519. let mut mac = Hmac::<Sha512>::new_varkey(secret_key.as_bytes()).expect("Failed to create HMAC");
  520. mac.update(message.as_bytes());
  521. let result = mac.finalize().into_bytes();
  522. let sign = hex::encode(result);
  523. sign
  524. }
  525. async fn http_tool(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> ResponseData {
  526. /****请求接口与 地址*/
  527. let url = format!("{}{}", self.base_url.to_string(), request_path);
  528. let request_type = request_type.clone().to_uppercase();
  529. let addrs_url = format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()));
  530. let request_builder = match request_type.as_str() {
  531. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  532. "POST" => self.client.post(addrs_url.clone()).body(params.clone()).headers(headers),
  533. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  534. // "PUT" => self.client.put(url.clone()).json(&params),
  535. _ => {
  536. panic!("{}", format!("错误的请求类型:{}", request_type.clone()))
  537. }
  538. };
  539. // 读取响应的内容
  540. let response = request_builder.send().await.unwrap();
  541. let is_success = response.status().is_success(); // 先检查状态码
  542. let text = response.text().await.unwrap();
  543. return if is_success {
  544. self.on_success_data(&text)
  545. } else {
  546. self.on_error_data(&text, &addrs_url, &params)
  547. }
  548. }
  549. pub fn on_success_data(&mut self, text: &String) -> ResponseData {
  550. let data = serde_json::from_str(text.as_str()).unwrap();
  551. ResponseData::new(self.label.clone(), 200, "success".to_string(), data)
  552. }
  553. pub fn on_error_data(&mut self, text: &String, base_url: &String, params: &String) -> ResponseData {
  554. let json_value = serde_json::from_str::<Value>(&text);
  555. match json_value {
  556. Ok(data) => {
  557. let message;
  558. if !data["message"].is_null() {
  559. message = format!("{}:{}", data["label"].as_str().unwrap(), data["message"].as_str().unwrap());
  560. } else {
  561. message = data["label"].to_string();
  562. }
  563. let mut error = ResponseData::error(self.label.clone(), message);
  564. error.message = format!("请求地址:{}, 请求参数:{}, 报错内容:{}。", base_url, params, error.message);
  565. error
  566. }
  567. Err(e) => {
  568. error!("解析错误:{:?}", e);
  569. let error = ResponseData::error("".to_string(),
  570. format!("json 解析失败:{},相关参数:{}", e, text));
  571. error
  572. }
  573. }
  574. }
  575. }