bitget_swap_rest.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. use std::collections::BTreeMap;
  2. use std::str::FromStr;
  3. use reqwest::Client;
  4. use reqwest::header::HeaderMap;
  5. use rust_decimal::Decimal;
  6. use tracing::info;
  7. use crate::http_tool::RestTool;
  8. use crate::response_base::ResponseData;
  9. use ring::hmac;
  10. use rust_decimal::prelude::FromPrimitive;
  11. use serde_json::Value;
  12. #[derive(Clone, Debug)]
  13. pub struct BitgetSwapRest {
  14. pub label: String,
  15. base_url: String,
  16. client: Client,
  17. login_param: BTreeMap<String, String>, // 登录参数
  18. delays: Vec<i64>,
  19. max_delay: i64,
  20. avg_delay: Decimal,
  21. }
  22. impl BitgetSwapRest {
  23. pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> BitgetSwapRest {
  24. return BitgetSwapRest::new_label("default-BitgetSwapRest".to_string(), is_colo, login_param)
  25. }
  26. // 构造Bitget,可以自定义label
  27. pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> BitgetSwapRest {
  28. let base_url = if is_colo {
  29. "https://api.bitget.com".to_string()
  30. } else {
  31. "https://api.bitget.com".to_string()
  32. };
  33. if is_colo {
  34. info!("开启高速(未配置,走普通:{})通道", base_url);
  35. } else {
  36. info!("走普通通道:{}", base_url);
  37. }
  38. BitgetSwapRest {
  39. label,
  40. base_url,
  41. client: Client::new(),
  42. login_param,
  43. delays: vec![],
  44. max_delay: 0,
  45. avg_delay: Decimal::ZERO,
  46. }
  47. }
  48. //获取行情信息
  49. pub async fn get_contracts(&mut self, symbol: String) -> ResponseData {
  50. let params = serde_json::json!({
  51. "symbol": symbol,
  52. "productType": "USDT-FUTURES"
  53. });
  54. let data = self.request("GET".to_string(),
  55. "/api/v2".to_string(),
  56. "/mix/market/contracts".to_string(),
  57. true,
  58. params.to_string(),
  59. ).await;
  60. data
  61. }
  62. //获取行情信息
  63. pub async fn get_tickers(&mut self, symbol: String) -> ResponseData {
  64. let params = serde_json::json!({
  65. "symbol": symbol,
  66. "productType": "USDT-FUTURES"
  67. });
  68. let data = self.request("GET".to_string(),
  69. "/api/v2".to_string(),
  70. "/mix/market/ticker".to_string(),
  71. true,
  72. params.to_string(),
  73. ).await;
  74. data
  75. }
  76. // 获取服务器时间
  77. pub async fn get_server_time(&mut self) -> ResponseData {
  78. let params = serde_json::json!({});
  79. self.request("GET".to_string(),
  80. "/api/v2".to_string(),
  81. "/public/time".to_string(),
  82. false,
  83. params.to_string(),
  84. ).await
  85. }
  86. // 获取账户信息
  87. pub async fn get_account_info(&mut self) -> ResponseData {
  88. let params = serde_json::json!({
  89. "productType": "USDT-FUTURES"
  90. });
  91. self.request("GET".to_string(),
  92. "/api/v2".to_string(),
  93. "/mix/account/accounts".to_string(),
  94. true,
  95. params.to_string(),
  96. ).await
  97. }
  98. // 获取仓位信息(单个)
  99. pub async fn get_single_position(&mut self, params: Value) -> ResponseData {
  100. self.request("GET".to_string(),
  101. "/api/v2".to_string(),
  102. "/mix/position/single-position".to_string(),
  103. true,
  104. params.to_string(),
  105. ).await
  106. }
  107. // 获取所有仓位
  108. pub async fn get_all_position(&mut self, params: Value) -> ResponseData {
  109. self.request("GET".to_string(),
  110. "/api/v2".to_string(),
  111. "/mix/position/all-position".to_string(),
  112. true,
  113. params.to_string(),
  114. ).await
  115. }
  116. // 获取K线
  117. pub async fn get_market_candles(&mut self, params: Value) -> ResponseData {
  118. self.request("GET".to_string(),
  119. "/api/v2".to_string(),
  120. "/mix/market/candles".to_string(),
  121. true,
  122. params.to_string(),
  123. ).await
  124. }
  125. // 下单
  126. pub async fn swap_order(&mut self, params: Value) -> ResponseData {
  127. self.request("POST".to_string(),
  128. "/api/v2".to_string(),
  129. "/mix/order/place-order".to_string(),
  130. true,
  131. params.to_string(),
  132. ).await
  133. }
  134. // 撤单
  135. pub async fn cancel_order(&mut self, params: Value) -> ResponseData {
  136. self.request("POST".to_string(),
  137. "/api/v2".to_string(),
  138. "/mix/order/cancel-order".to_string(),
  139. true,
  140. params.to_string(),
  141. ).await
  142. }
  143. // 获取订单详情
  144. pub async fn get_order(&mut self, params: Value) -> ResponseData {
  145. self.request("GET".to_string(),
  146. "/api/v2".to_string(),
  147. "/mix/order/detail".to_string(),
  148. true,
  149. params.to_string(),
  150. ).await
  151. }
  152. // 获取当前未成交订单
  153. pub async fn get_pending_orders(&mut self) -> ResponseData {
  154. let params = serde_json::json!({
  155. "productType": "USDT-FUTURES"
  156. });
  157. self.request("GET".to_string(),
  158. "/api/v2".to_string(),
  159. "/mix/order/orders-pending".to_string(),
  160. true,
  161. params.to_string(),
  162. ).await
  163. }
  164. // 调整杠杆
  165. pub async fn set_leverage(&mut self, params: Value) -> ResponseData {
  166. self.request("POST".to_string(),
  167. "/api/v2".to_string(),
  168. "/mix/account/set-leverage".to_string(),
  169. true,
  170. params.to_string(),
  171. ).await
  172. }
  173. // 调整持仓模式(单向、双向)
  174. pub async fn set_position_mode(&mut self, params: Value) -> ResponseData {
  175. self.request("POST".to_string(),
  176. "/api/v2".to_string(),
  177. "/mix/account/set-position-mode".to_string(),
  178. true,
  179. params.to_string(),
  180. ).await
  181. }
  182. // 发送请求
  183. pub async fn request(&mut self,
  184. method: String,
  185. prefix_url: String,
  186. request_url: String,
  187. is_login: bool,
  188. params: String) -> ResponseData
  189. {
  190. // trace!("login_param:{:?}", self.login_param);
  191. //解析账号信息
  192. let mut access_key = "".to_string();
  193. let mut secret_key = "".to_string();
  194. let mut passphrase = "".to_string();
  195. if self.login_param.contains_key("access_key") {
  196. access_key = self.login_param.get("access_key").unwrap().to_string();
  197. }
  198. if self.login_param.contains_key("secret_key") {
  199. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  200. }
  201. if self.login_param.contains_key("pass_key") {
  202. passphrase = self.login_param.get("pass_key").unwrap().to_string();
  203. }
  204. let mut is_login_param = true;
  205. if access_key == "" || secret_key == "" || passphrase == "" {
  206. is_login_param = false
  207. }
  208. //请求头配置-如果需要登陆则存在额外配置
  209. let mut body = "".to_string();
  210. let timestamp = chrono::Utc::now().timestamp_millis().to_string();
  211. let mut headers = HeaderMap::new();
  212. headers.insert("Content-Type", "application/json".parse().unwrap());
  213. headers.insert("locale", "en-US".parse().unwrap());
  214. if method == "POST" {
  215. body = params.clone();
  216. }
  217. //是否需要登陆-- 组装sing
  218. if is_login {
  219. if !is_login_param {
  220. let e = ResponseData::error(self.label.clone(), "登录参数错误!".to_string());
  221. return e;
  222. } else {
  223. //需要登陆-且登陆参数齐全
  224. // trace!("param:{}", params);
  225. // trace!("body:{}", body);
  226. //组装sing
  227. let sing = Self::sign(secret_key.clone(),
  228. method.clone(),
  229. prefix_url.clone(),
  230. request_url.clone(),
  231. params.clone(),
  232. body.clone(),
  233. timestamp.clone(),
  234. );
  235. //组装header
  236. headers.extend(Self::headers(sing, timestamp, passphrase, access_key));
  237. }
  238. }
  239. // trace!("headers:{:?}", headers);
  240. let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
  241. let start_time = chrono::Utc::now().timestamp_millis();
  242. let get_response = self.http_tool(
  243. format!("{}{}", prefix_url.clone(), request_url.clone()),
  244. method.to_string(),
  245. params.clone(),
  246. headers,
  247. ).await;
  248. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  249. self.delays.push(time_array);
  250. self.get_delay_info();
  251. let res_data = Self::res_data_analysis(get_response, base_url, params);
  252. res_data
  253. }
  254. pub fn res_data_analysis(result: Result<ResponseData, reqwest::Error>, base_url: String, params: String) -> ResponseData {
  255. match result {
  256. Ok(res_data) => {
  257. if res_data.code != 200 {
  258. let json_value = res_data.data;
  259. let error = ResponseData::new("".to_string(),
  260. res_data.code,
  261. format!("请求地址:{}, 请求参数:{}, 响应结果:{}", base_url, params, res_data.message),
  262. json_value);
  263. error
  264. } else {
  265. let json_value = res_data.data;
  266. let code = json_value["code"].as_str().unwrap();
  267. if code == "00000" {
  268. let data = serde_json::to_string(&json_value["data"]).unwrap();
  269. let success = ResponseData::new("".to_string(), 200, "success".to_string(), data.parse().unwrap());
  270. success
  271. } else {
  272. let code_num = i16::from_str(json_value["code"].as_str().unwrap()).unwrap();
  273. let error = ResponseData::new("".to_string(), code_num,
  274. format!("请求地址:{}, 请求参数:{}, 响应结果:{}", base_url, params, json_value.as_str().unwrap()),
  275. json_value);
  276. error
  277. }
  278. }
  279. }
  280. Err(err) => {
  281. let error = ResponseData::error("".to_string(), format!("json 解析失败:{}", err));
  282. error
  283. }
  284. }
  285. }
  286. fn get_delay_info(&mut self) {
  287. let last_100 = if self.delays.len() > 100 {
  288. self.delays[self.delays.len() - 100..].to_vec()
  289. } else {
  290. self.delays.clone()
  291. };
  292. let max_value = last_100.iter().max().unwrap();
  293. if max_value.clone().to_owned() > self.max_delay {
  294. self.max_delay = max_value.clone().to_owned();
  295. }
  296. let sum: i64 = last_100.iter().sum();
  297. let sum_v = Decimal::from_i64(sum).unwrap();
  298. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  299. self.avg_delay = (sum_v / len_v).round_dp(1);
  300. self.delays = last_100.clone().into_iter().collect();
  301. }
  302. // 封装请求头
  303. pub fn headers(sign: String, timestamp: String, passphrase: String, access_key: String) -> HeaderMap {
  304. let mut headers = HeaderMap::new();
  305. headers.insert("ACCESS-KEY", access_key.parse().unwrap());
  306. headers.insert("ACCESS-SIGN", sign.parse().unwrap());
  307. headers.insert("ACCESS-TIMESTAMP", timestamp.parse().unwrap());
  308. headers.insert("ACCESS-PASSPHRASE", passphrase.parse().unwrap());
  309. headers
  310. }
  311. async fn http_tool(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<ResponseData, reqwest::Error> {
  312. let res_data: ResponseData;
  313. /****请求接口与 地址*/
  314. let url = format!("{}{}", self.base_url.to_string(), request_path);
  315. let request_type = request_type.clone().to_uppercase();
  316. let addrs_url: String = if RestTool::parse_params_to_str(params.clone()) == "" {
  317. url.clone()
  318. } else {
  319. format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()))
  320. };
  321. // trace!("url:{}", url);
  322. // trace!("addrs_url:{}", addrs_url);
  323. // let params_json: serde_json::Value = serde_json::from_str(&params).unwrap();
  324. // trace!("params_json:{}",params_json);
  325. // trace!("headers:{:?}",headers);
  326. let req = match request_type.as_str() {
  327. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  328. "POST" => self.client.post(url.clone()).body(params).headers(headers),
  329. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  330. // "PUT" => self.client.put(url.clone()).json(&params),
  331. _ => return Ok(ResponseData::error(self.label.clone(), format!("错误的请求类型:{}", request_type.clone()))), // 处理未知请求类型
  332. };
  333. let response = req.send().await?;
  334. if response.status().is_success() {
  335. // 读取响应的内容
  336. let body = response.text().await?;
  337. // trace!("ok-----{}", body);
  338. res_data = ResponseData::new(self.label.clone(), 200, "success".to_string(), body.parse().unwrap());
  339. } else {
  340. let body = response.text().await?;
  341. res_data = ResponseData::error(self.label.clone(), body.to_string())
  342. }
  343. Ok(res_data)
  344. }
  345. // 对请求进行签名处理
  346. pub fn sign(secret_key: String,
  347. method: String, prefix_url: String, request_url: String,
  348. params: String, body: String, timestamp: String) -> String {
  349. let url_param_str = RestTool::parse_params_to_str(params);
  350. let base_url = if method == "GET" && url_param_str.len() > 0 {
  351. format!("{}{}?{}", prefix_url, request_url, url_param_str)
  352. } else {
  353. format!("{}{}", prefix_url, request_url)
  354. };
  355. // 时间戳 + 请求类型+ 请求参数字符串
  356. let message = format!("{}{}{}{}", timestamp, method, base_url, body);
  357. // trace!("message:{}",message);
  358. // 做签名
  359. let hmac_key = hmac::Key::new(hmac::HMAC_SHA256, secret_key.as_bytes());
  360. let result = hmac::sign(&hmac_key, &message.as_bytes());
  361. let sign = base64::encode(result);
  362. sign
  363. }
  364. }