bybit_swap_rest_utils.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. use std::collections::BTreeMap;
  2. use reqwest::Client;
  3. use reqwest::header::HeaderMap;
  4. use ring::hmac;
  5. use rust_decimal::Decimal;
  6. use rust_decimal::prelude::FromPrimitive;
  7. use rust_decimal_macros::dec;
  8. use toml::Value;
  9. use tracing::{info, trace};
  10. use crate::http::request::Response;
  11. use crate::utils::utils::parse_params_to_str;
  12. #[derive(Clone, Debug)]
  13. pub struct BybitSwapRest {
  14. pub label: String,
  15. base_url: String,
  16. client: reqwest::Client,
  17. /*******参数*/
  18. //登陆所需参数
  19. login_param: BTreeMap<String, String>,
  20. delays: Vec<i64>,
  21. max_delay: i64,
  22. avg_delay: Decimal,
  23. }
  24. impl BybitSwapRest {
  25. /*******************************************************************************************************/
  26. /*****************************************获取一个对象****************************************************/
  27. /*******************************************************************************************************/
  28. pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> BybitSwapRest
  29. {
  30. return BybitSwapRest::new_label("default-BybitSwapRest".to_string(), is_colo, login_param);
  31. }
  32. pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> BybitSwapRest {
  33. let base_url = if is_colo {
  34. "https://api.bytick.com".to_string()
  35. } else {
  36. "https://api.bytick.com".to_string()
  37. };
  38. if is_colo {
  39. info!("开启高速(未配置,走普通:{})通道",base_url);
  40. } else {
  41. info!("走普通通道:{}",base_url);
  42. }
  43. /*****返回结构体*******/
  44. BybitSwapRest {
  45. label,
  46. base_url,
  47. client: Client::new(),
  48. login_param,
  49. delays: vec![],
  50. max_delay: 0,
  51. avg_delay: dec!(0.0),
  52. }
  53. }
  54. /*******************************************************************************************************/
  55. /*****************************************rest请求函数********************************************************/
  56. /*******************************************************************************************************/
  57. //服務器時間
  58. pub async fn get_server_time(&mut self) -> Response {
  59. let params = serde_json::json!({
  60. });
  61. let data = self.request("GET".to_string(),
  62. "/v5".to_string(),
  63. "/market/time".to_string(),
  64. false,
  65. params.to_string(),
  66. ).await;
  67. data
  68. }
  69. //查詢可交易產品的規格信息
  70. pub async fn get_instruments_info(&mut self, symbol: String) -> Response {
  71. let params = serde_json::json!({
  72. "category":"linear",
  73. "symbol":symbol
  74. });
  75. let data = self.request("GET".to_string(),
  76. "/v5".to_string(),
  77. "/market/instruments-info".to_string(),
  78. false,
  79. params.to_string(),
  80. ).await;
  81. data
  82. }
  83. //賬戶的交易日誌。
  84. pub async fn get_account_transaction_log(&mut self) -> Response {
  85. let mut params = serde_json::json!({
  86. "accountType":"UNIFIED",
  87. "category":"linear",
  88. "currency":"USDT",
  89. });
  90. params["startTime"] = serde_json::json!(1703124000000 as i64);
  91. params["endTime"] = serde_json::json!(1703664000000 as i64);
  92. let data = self.request("GET".to_string(),
  93. "/v5".to_string(),
  94. "/account/transaction-log".to_string(),
  95. true,
  96. params.to_string(),
  97. ).await;
  98. data
  99. }
  100. //查詢平倉盈虧
  101. pub async fn get_position_closed_pnl(&mut self, symbol: String) -> Response {
  102. let mut params = serde_json::json!({
  103. "symbol":symbol,
  104. "limit":"200"
  105. });
  106. params["startTime"] = serde_json::json!(1703124000000 as i64);
  107. params["endTime"] = serde_json::json!(1703664000000 as i64);
  108. let data = self.request("GET".to_string(),
  109. "/contract/v3".to_string(),
  110. "/private/position/closed-pnl".to_string(),
  111. true,
  112. params.to_string(),
  113. ).await;
  114. data
  115. }
  116. //查詢錢包資金紀錄
  117. pub async fn get_account_wallet_fund_records(&mut self, start_time: i64, end_time: i64) -> Response {
  118. let mut params = serde_json::json!({
  119. "coin":"USDT",
  120. "walletFundType":"RealisedPNL",
  121. "limit":"50",
  122. });
  123. // if start_time > 0 {
  124. params["startTime"] = serde_json::json!(1703124000000 as i64);
  125. // }
  126. // if end_time > 0 {
  127. params["endTime"] = serde_json::json!(1703664000000 as i64);
  128. // }
  129. let data = self.request("GET".to_string(),
  130. "/contract/v3".to_string(),
  131. "/private/account/wallet/fund-records".to_string(),
  132. true,
  133. params.to_string(),
  134. ).await;
  135. data
  136. }
  137. /*******************************************************************************************************/
  138. /*****************************************工具函数********************************************************/
  139. /*******************************************************************************************************/
  140. pub fn get_delays(&self) -> Vec<i64> {
  141. self.delays.clone()
  142. }
  143. pub fn get_avg_delay(&self) -> Decimal {
  144. self.avg_delay.clone()
  145. }
  146. pub fn get_max_delay(&self) -> i64 {
  147. self.max_delay.clone()
  148. }
  149. fn get_delay_info(&mut self) {
  150. let last_100 = if self.delays.len() > 100 {
  151. self.delays[self.delays.len() - 100..].to_vec()
  152. } else {
  153. self.delays.clone()
  154. };
  155. let max_value = last_100.iter().max().unwrap();
  156. if max_value.clone().to_owned() > self.max_delay {
  157. self.max_delay = max_value.clone().to_owned();
  158. }
  159. let sum: i64 = last_100.iter().sum();
  160. let sum_v = Decimal::from_i64(sum).unwrap();
  161. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  162. self.avg_delay = (sum_v / len_v).round_dp(1);
  163. self.delays = last_100.clone().into_iter().collect();
  164. }
  165. //调用请求
  166. pub async fn request(&mut self,
  167. method: String,
  168. prefix_url: String,
  169. request_url: String,
  170. is_login: bool,
  171. params: String) -> Response
  172. {
  173. trace!("login_param:{:?}", self.login_param);
  174. //解析账号信息
  175. let mut access_key = "".to_string();
  176. let mut secret_key = "".to_string();
  177. // let mut passphrase = "".to_string();
  178. if self.login_param.contains_key("access_key") {
  179. access_key = self.login_param.get("access_key").unwrap().to_string();
  180. }
  181. if self.login_param.contains_key("secret_key") {
  182. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  183. }
  184. // if self.login_param.contains_key("pass_key") {
  185. // passphrase = self.login_param.get("pass_key").unwrap().to_string();
  186. // }
  187. let mut is_login_param = true;
  188. if access_key == "" || secret_key == "" {
  189. is_login_param = false
  190. }
  191. //请求头配置-如果需要登陆则存在额外配置
  192. let mut body = "".to_string();
  193. let timestamp = Self::get_timestamp();
  194. let mut headers = HeaderMap::new();
  195. headers.insert("Content-Type", "application/json; charset=utf-8".parse().unwrap());
  196. headers.insert("X-BAPI-RECV-WINDOW", "5000".parse().unwrap());
  197. if method == "POST" {
  198. body = params.clone();
  199. }
  200. //是否需要登陆-- 组装sing
  201. if is_login {
  202. if !is_login_param {
  203. let mut res = Response::new();
  204. res.code = "-1".to_string();
  205. res.msg = "登陆参数错误".to_string();
  206. return res;
  207. } else {
  208. //需要登陆-且登陆参数齐全
  209. trace!("param:{}", params);
  210. trace!("body:{}", body);
  211. //组装sing
  212. let sing = Self::sign(
  213. access_key.clone(),
  214. secret_key.clone(),
  215. method.clone(),
  216. params.clone(),
  217. timestamp.clone(),
  218. );
  219. //组装header
  220. headers.extend(Self::headers(sing, timestamp, access_key));
  221. }
  222. }
  223. // trace!("headers:{:?}", headers);
  224. let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
  225. let start_time = chrono::Utc::now().timestamp_millis();
  226. let get_response = self.http_toll(
  227. format!("{}{}", prefix_url.clone(), request_url.clone()),
  228. method.to_string(),
  229. params.clone(),
  230. headers,
  231. ).await;
  232. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  233. self.delays.push(time_array);
  234. self.get_delay_info();
  235. let res_data = Self::res_data_analysis(get_response, base_url, params);
  236. res_data
  237. }
  238. pub fn headers(sign: String, timestamp: String, access_key: String) -> HeaderMap {
  239. let mut headers = HeaderMap::new();
  240. headers.insert("X-BAPI-SIGN-TYPE", "2".parse().unwrap());
  241. headers.insert("X-BAPI-API-KEY", access_key.parse().unwrap());
  242. headers.insert("X-BAPI-TIMESTAMP", timestamp.parse().unwrap());
  243. headers.insert("X-BAPI-SIGN", sign.parse().unwrap());
  244. // headers.insert("X-Referer", passphrase.parse().unwrap());
  245. headers
  246. }
  247. pub fn sign(access_key: String,
  248. secret_key: String,
  249. method: String,
  250. params: String, timestamp: String) -> String
  251. {
  252. /*签名生成*/
  253. let url_param_str = parse_params_to_str(params.clone());
  254. let parameters = if method == "GET" {
  255. url_param_str
  256. } else {
  257. params
  258. };
  259. let message = format!("{}{}5000{}", timestamp, access_key, parameters);
  260. trace!("message:{}",message);
  261. // 做签名
  262. let hmac_key = ring::hmac::Key::new(hmac::HMAC_SHA256, secret_key.as_bytes());
  263. let result = ring::hmac::sign(&hmac_key, &message.as_bytes());
  264. let sign = hex::encode(result.as_ref());
  265. sign
  266. }
  267. async fn http_toll(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<Response, reqwest::Error> {
  268. /****请求接口与 地址*/
  269. let url = format!("{}{}", self.base_url.to_string(), request_path);
  270. let request_type = request_type.clone().to_uppercase();
  271. let addrs_url = format!("{}?{}", url.clone(), parse_params_to_str(params.clone()));
  272. // let addrs_url: String = if parse_params_to_str(params.clone()) == "" {
  273. // url.clone()
  274. // } else {
  275. // format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()))
  276. // };
  277. trace!("url:{}", url);
  278. trace!("addrs_url:{}", addrs_url);
  279. let params_json: serde_json::Value = serde_json::from_str(&params).unwrap();
  280. trace!("params_json:{}",params_json);
  281. trace!("headers:{:?}",headers);
  282. let req = match request_type.as_str() {
  283. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  284. "POST" => self.client.post(url.clone()).body(params).headers(headers),
  285. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  286. // "PUT" => self.client.put(url.clone()).json(&params),
  287. // _ => return Ok(ResponseData::error(self.label.clone(), format!("错误的请求类型:{}", request_type.clone()))), // 处理未知请求类型
  288. _ => {
  289. let mut res = Response::new();
  290. res.code = "-1".to_string();
  291. res.msg = "请求类型错误".to_string();
  292. res.data = "".to_string();
  293. return Ok(res);
  294. }
  295. };
  296. let response = req.send().await?;
  297. let mut res = Response::new();
  298. if response.status().is_success() {
  299. // 读取响应的内容
  300. let body = response.text().await?;
  301. trace!("ok-----{}", body);
  302. // res_data = ResponseData::new(self.label.clone(), "200".to_string(), "success".to_string(), body);
  303. res.code = "200".to_string();
  304. res.msg = "success".to_string();
  305. res.data = body;
  306. } else {
  307. let body = response.text().await?;
  308. trace!("error-----{}", body);
  309. // res_data = ResponseData::error(self.label.clone(), body.to_string())
  310. res.code = "-1".to_string();
  311. res.msg = body;
  312. res.data = "".to_string();
  313. }
  314. Ok(res)
  315. }
  316. //res_data 解析
  317. pub fn res_data_analysis(result: Result<Response, reqwest::Error>, base_url: String, params: String) -> Response {
  318. let res = Response::new();
  319. trace!("原始数据:{:?}",result);
  320. match result {
  321. Ok(mut res_data) => {
  322. if res_data.code != "200" {
  323. // res_data
  324. trace!("不等于200");
  325. let msg = res_data.msg;
  326. let body: String = res_data.data.as_str().parse().unwrap();
  327. let json_value = serde_json::from_str::<Value>(&body);
  328. match json_value {
  329. Ok(_data) => {
  330. let mut error = Response::new();
  331. error.code = "-1".to_string();
  332. error.msg = format!("请求错误{:?}", res.msg);
  333. error.data = format!("请求地址:{},请求参数:{}", base_url, params);
  334. error
  335. }
  336. Err(e) => {
  337. let mut error = Response::new();
  338. error.code = "-1".to_string();
  339. error.msg = format!("json 解析失败:{},相关参数:{}", e, msg);
  340. error.data = "".to_string();
  341. error
  342. }
  343. }
  344. } else {
  345. res_data
  346. }
  347. }
  348. Err(err) => {
  349. let mut error = Response::new();
  350. error.code = "-1".to_string();
  351. error.msg = format!("json 解析失败:{},相关参数:{}", err, params);
  352. error.data = "".to_string();
  353. error
  354. }
  355. }
  356. }
  357. fn get_timestamp() -> String {
  358. chrono::Utc::now().timestamp_millis()
  359. .to_string()
  360. }
  361. }