bybit_swap_rest.rs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510
  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 tracing::{info, trace};
  9. use crate::http_tool::RestTool;
  10. use crate::response_base::ResponseData;
  11. #[derive(Clone, Debug)]
  12. pub struct BybitSwapRest {
  13. pub label: String,
  14. base_url: String,
  15. client: reqwest::Client,
  16. /*******参数*/
  17. //登陆所需参数
  18. login_param: BTreeMap<String, String>,
  19. delays: Vec<i64>,
  20. max_delay: i64,
  21. avg_delay: Decimal,
  22. }
  23. impl BybitSwapRest {
  24. /*******************************************************************************************************/
  25. /*****************************************获取一个对象****************************************************/
  26. /*******************************************************************************************************/
  27. pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> BybitSwapRest
  28. {
  29. return BybitSwapRest::new_label("default-BybitSwapRest".to_string(), is_colo, login_param);
  30. }
  31. pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> BybitSwapRest {
  32. let base_url = if is_colo {
  33. "https://api.bytick.com".to_string()
  34. } else {
  35. "https://api.bytick.com".to_string()
  36. };
  37. if is_colo {
  38. info!("开启高速(未配置,走普通:{})通道",base_url);
  39. } else {
  40. info!("走普通通道:{}",base_url);
  41. }
  42. /*****返回结构体*******/
  43. BybitSwapRest {
  44. label,
  45. base_url,
  46. client: Client::new(),
  47. login_param,
  48. delays: vec![],
  49. max_delay: 0,
  50. avg_delay: dec!(0.0),
  51. }
  52. }
  53. /*******************************************************************************************************/
  54. /*****************************************rest请求函数********************************************************/
  55. /*******************************************************************************************************/
  56. //服務器時間
  57. pub async fn get_server_time(&mut self) -> ResponseData {
  58. let params = serde_json::json!({
  59. });
  60. let data = self.request("GET".to_string(),
  61. "/v5".to_string(),
  62. "/market/time".to_string(),
  63. false,
  64. params.to_string(),
  65. ).await;
  66. data
  67. }
  68. //查詢最新行情信息
  69. pub async fn get_tickers(&mut self, symbol: String) -> ResponseData {
  70. let params = serde_json::json!({
  71. "category":"linear",
  72. "symbol":symbol
  73. });
  74. let data = self.request("GET".to_string(),
  75. "/v5".to_string(),
  76. "/market/tickers".to_string(),
  77. false,
  78. params.to_string(),
  79. ).await;
  80. data
  81. }
  82. //查詢市場價格K線數據
  83. pub async fn get_kline(&mut self, symbol: String) -> ResponseData {
  84. let params = serde_json::json!({
  85. "category":"linear",
  86. "symbol":symbol,
  87. "interval":"1",
  88. "limit":"200"
  89. });
  90. let data = self.request("GET".to_string(),
  91. "/v5".to_string(),
  92. "/market/kline".to_string(),
  93. false,
  94. params.to_string(),
  95. ).await;
  96. data
  97. }
  98. //查詢公告
  99. pub async fn get_announcements(&mut self) -> ResponseData {
  100. let params = serde_json::json!({
  101. "locale":"zh-TW"
  102. });
  103. let data = self.request("GET".to_string(),
  104. "/v5".to_string(),
  105. "/announcements/index".to_string(),
  106. false,
  107. params.to_string(),
  108. ).await;
  109. data
  110. }
  111. //查詢可交易產品的規格信息
  112. pub async fn get_instruments_info(&mut self, symbol: String) -> ResponseData {
  113. let params = serde_json::json!({
  114. "category":"linear",
  115. "symbol":symbol
  116. });
  117. let data = self.request("GET".to_string(),
  118. "/v5".to_string(),
  119. "/market/instruments-info".to_string(),
  120. false,
  121. params.to_string(),
  122. ).await;
  123. data
  124. }
  125. //查看持仓信息
  126. pub async fn get_positions(&mut self, symbol: String) -> ResponseData {
  127. let mut params = serde_json::json!({
  128. "category":"linear",
  129. });
  130. if symbol.len() > 0 {
  131. params.as_object_mut().unwrap().insert("symbol".parse().unwrap(), serde_json::Value::from(symbol));
  132. }
  133. let data = self.request("GET".to_string(),
  134. "/v5".to_string(),
  135. "/position/list".to_string(),
  136. true,
  137. params.to_string(),
  138. ).await;
  139. data
  140. }
  141. //设置持仓模式
  142. pub async fn set_position_mode(&mut self, symbol: String, mode: i64) -> ResponseData {
  143. let params = serde_json::json!({
  144. "category": "linear",
  145. "symbol": symbol,
  146. "mode": mode,
  147. });
  148. let data = self.request("POST".to_string(),
  149. "/v5".to_string(),
  150. "/position/switch-mode".to_string(),
  151. true,
  152. params.to_string(),
  153. ).await;
  154. data
  155. }
  156. //設置槓桿
  157. pub async fn set_leverage(&mut self, symbol: String,
  158. lever: String) -> ResponseData {
  159. let params = serde_json::json!({
  160. "category": "linear",
  161. "symbol": symbol,
  162. "buyLeverage": lever,
  163. "sellLeverage": lever,
  164. });
  165. let data = self.request("POST".to_string(),
  166. "/v5".to_string(),
  167. "/position/set-leverage".to_string(),
  168. true,
  169. params.to_string(),
  170. ).await;
  171. data
  172. }
  173. //查詢錢包餘額
  174. pub async fn get_account_balance(&mut self, symbol: String) -> ResponseData {
  175. let params = serde_json::json!({
  176. "accountType":"UNIFIED",
  177. "coin":symbol
  178. });
  179. let data = self.request("GET".to_string(),
  180. "/v5".to_string(),
  181. "/account/wallet-balance".to_string(),
  182. true,
  183. params.to_string(),
  184. ).await;
  185. data
  186. }
  187. //創建委託單
  188. pub async fn swap_order(&mut self, params: serde_json::Value) -> ResponseData {
  189. let data = self.request("POST".to_string(),
  190. "/v5".to_string(),
  191. "/order/create".to_string(),
  192. true,
  193. params.to_string(),
  194. ).await;
  195. data
  196. }
  197. //查詢實時委託單
  198. pub async fn get_order(&mut self, symbol: String, order_id: String, order_link_id: String) -> ResponseData {
  199. let mut params = serde_json::json!({
  200. "category":"linear",
  201. "symbol":symbol,
  202. });
  203. if order_id.len() > 0 {
  204. params.as_object_mut().unwrap().insert("orderId".parse().unwrap(), serde_json::Value::from(order_id));
  205. }
  206. if order_link_id.len() > 0 {
  207. params.as_object_mut().unwrap().insert("orderLinkId".parse().unwrap(), serde_json::Value::from(order_link_id));
  208. }
  209. let data = self.request("GET".to_string(),
  210. "/v5".to_string(),
  211. "/order/realtime".to_string(),
  212. true,
  213. params.to_string(),
  214. ).await;
  215. data
  216. }
  217. //撤单
  218. pub async fn cancel_order(&mut self, symbol: String, order_id: String, order_link_id: String) -> ResponseData {
  219. let mut params = serde_json::json!({
  220. "category": "linear",
  221. "symbol": symbol,
  222. });
  223. if order_id.len() > 0 {
  224. params.as_object_mut().unwrap().insert("orderId".parse().unwrap(), serde_json::Value::from(order_id));
  225. }
  226. if order_link_id.len() > 0 {
  227. params.as_object_mut().unwrap().insert("orderLinkId".parse().unwrap(), serde_json::Value::from(order_link_id));
  228. }
  229. let data = self.request("POST".to_string(),
  230. "/v5".to_string(),
  231. "/order/cancel".to_string(),
  232. true,
  233. params.to_string(),
  234. ).await;
  235. data
  236. }
  237. //撤銷所有訂單
  238. pub async fn cancel_orders(&mut self, symbol: String) -> ResponseData {
  239. let params = serde_json::json!({
  240. "category": "linear",
  241. "symbol": symbol,
  242. });
  243. let data = self.request("POST".to_string(),
  244. "/v5".to_string(),
  245. "/order/cancel-all".to_string(),
  246. true,
  247. params.to_string(),
  248. ).await;
  249. data
  250. }
  251. /*******************************************************************************************************/
  252. /*****************************************工具函数********************************************************/
  253. /*******************************************************************************************************/
  254. pub fn get_delays(&self) -> Vec<i64> {
  255. self.delays.clone()
  256. }
  257. pub fn get_avg_delay(&self) -> Decimal {
  258. self.avg_delay.clone()
  259. }
  260. pub fn get_max_delay(&self) -> i64 {
  261. self.max_delay.clone()
  262. }
  263. fn get_delay_info(&mut self) {
  264. let last_100 = if self.delays.len() > 100 {
  265. self.delays[self.delays.len() - 100..].to_vec()
  266. } else {
  267. self.delays.clone()
  268. };
  269. let max_value = last_100.iter().max().unwrap();
  270. if max_value.clone().to_owned() > self.max_delay {
  271. self.max_delay = max_value.clone().to_owned();
  272. }
  273. let sum: i64 = last_100.iter().sum();
  274. let sum_v = Decimal::from_i64(sum).unwrap();
  275. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  276. self.avg_delay = (sum_v / len_v).round_dp(1);
  277. self.delays = last_100.clone().into_iter().collect();
  278. }
  279. //调用请求
  280. pub async fn request(&mut self,
  281. method: String,
  282. prefix_url: String,
  283. request_url: String,
  284. is_login: bool,
  285. params: String) -> ResponseData
  286. {
  287. trace!("login_param:{:?}", self.login_param);
  288. //解析账号信息
  289. let mut access_key = "".to_string();
  290. let mut secret_key = "".to_string();
  291. // let mut passphrase = "".to_string();
  292. if self.login_param.contains_key("access_key") {
  293. access_key = self.login_param.get("access_key").unwrap().to_string();
  294. }
  295. if self.login_param.contains_key("secret_key") {
  296. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  297. }
  298. // if self.login_param.contains_key("pass_key") {
  299. // passphrase = self.login_param.get("pass_key").unwrap().to_string();
  300. // }
  301. let mut is_login_param = true;
  302. if access_key == "" || secret_key == "" {
  303. is_login_param = false
  304. }
  305. //请求头配置-如果需要登陆则存在额外配置
  306. let mut body = "".to_string();
  307. let timestamp = Self::get_timestamp();
  308. let mut headers = HeaderMap::new();
  309. headers.insert("Content-Type", "application/json; charset=utf-8".parse().unwrap());
  310. headers.insert("X-BAPI-RECV-WINDOW", "5000".parse().unwrap());
  311. if method == "POST" {
  312. body = params.clone();
  313. }
  314. //是否需要登陆-- 组装sing
  315. if is_login {
  316. if !is_login_param {
  317. let e = ResponseData::error(self.label.clone(), "登陆参数错误!".to_string());
  318. return e;
  319. } else {
  320. //需要登陆-且登陆参数齐全
  321. trace!("param:{}", params);
  322. trace!("body:{}", body);
  323. //组装sing
  324. let sing = Self::sign(
  325. access_key.clone(),
  326. secret_key.clone(),
  327. method.clone(),
  328. params.clone(),
  329. timestamp.clone(),
  330. );
  331. //组装header
  332. headers.extend(Self::headers(sing, timestamp, access_key));
  333. }
  334. }
  335. // trace!("headers:{:?}", headers);
  336. let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
  337. let start_time = chrono::Utc::now().timestamp_millis();
  338. let get_response = self.http_toll(
  339. format!("{}{}", prefix_url.clone(), request_url.clone()),
  340. method.to_string(),
  341. params.clone(),
  342. headers,
  343. ).await;
  344. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  345. self.delays.push(time_array);
  346. self.get_delay_info();
  347. let res_data = Self::res_data_analysis(get_response, base_url, params);
  348. res_data
  349. }
  350. pub fn headers(sign: String, timestamp: String, access_key: String) -> HeaderMap {
  351. let mut headers = HeaderMap::new();
  352. headers.insert("X-BAPI-SIGN-TYPE", "2".parse().unwrap());
  353. headers.insert("X-BAPI-API-KEY", access_key.parse().unwrap());
  354. headers.insert("X-BAPI-TIMESTAMP", timestamp.parse().unwrap());
  355. headers.insert("X-BAPI-SIGN", sign.parse().unwrap());
  356. // headers.insert("X-Referer", passphrase.parse().unwrap());
  357. headers
  358. }
  359. pub fn sign(access_key: String,
  360. secret_key: String,
  361. method: String,
  362. params: String, timestamp: String) -> String
  363. {
  364. /*签名生成*/
  365. let url_param_str = RestTool::parse_params_to_str(params.clone());
  366. let parameters = if method == "GET" {
  367. url_param_str
  368. } else {
  369. params
  370. };
  371. let message = format!("{}{}5000{}", timestamp, access_key, parameters);
  372. trace!("message:{}",message);
  373. // 做签名
  374. let hmac_key = ring::hmac::Key::new(hmac::HMAC_SHA256, secret_key.as_bytes());
  375. let result = ring::hmac::sign(&hmac_key, &message.as_bytes());
  376. let sign = hex::encode(result.as_ref());
  377. sign
  378. }
  379. async fn http_toll(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<ResponseData, reqwest::Error> {
  380. let res_data: ResponseData;
  381. /****请求接口与 地址*/
  382. let url = format!("{}{}", self.base_url.to_string(), request_path);
  383. let request_type = request_type.clone().to_uppercase();
  384. let addrs_url: String = if RestTool::parse_params_to_str(params.clone()) == "" {
  385. url.clone()
  386. } else {
  387. format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()))
  388. };
  389. trace!("url:{}", url);
  390. trace!("addrs_url:{}", addrs_url);
  391. let params_json: serde_json::Value = serde_json::from_str(&params).unwrap();
  392. trace!("params_json:{}",params_json);
  393. trace!("headers:{:?}",headers);
  394. let req = match request_type.as_str() {
  395. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  396. "POST" => self.client.post(url.clone()).body(params).headers(headers),
  397. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  398. // "PUT" => self.client.put(url.clone()).json(&params),
  399. _ => return Ok(ResponseData::error(self.label.clone(), format!("错误的请求类型:{}", request_type.clone()))), // 处理未知请求类型
  400. };
  401. let response = req.send().await?;
  402. if response.status().is_success() {
  403. // 读取响应的内容
  404. let body = response.text().await?;
  405. // trace!("ok-----{}", body);
  406. res_data = ResponseData::new(self.label.clone(), "200".to_string(), "success".to_string(), body);
  407. } else {
  408. let body = response.text().await?;
  409. // trace!("error-----{}", body);
  410. res_data = ResponseData::error(self.label.clone(), body.to_string())
  411. }
  412. Ok(res_data)
  413. }
  414. //res_data 解析
  415. pub fn res_data_analysis(result: Result<ResponseData, reqwest::Error>, base_url: String, params: String) -> ResponseData {
  416. // trace!("原始数据:{:?}",result);
  417. match result {
  418. Ok(res_data) => {
  419. if res_data.code != "200" {
  420. // trace!("不等于200");
  421. let message: String = res_data.message;
  422. let json_value: serde_json::Value = serde_json::from_str(&message).unwrap();
  423. let code = json_value["code"].as_str().unwrap();
  424. let msg = json_value["msg"].as_str().unwrap();
  425. let error = ResponseData::new("".to_string(),
  426. format!("{}", code),
  427. format!("{}", msg),
  428. format!("请求地址:{},请求参数:{}", base_url, params));
  429. error
  430. } else {
  431. let body: String = res_data.data.as_str().parse().unwrap();
  432. let json_value: serde_json::Value = serde_json::from_str(&body).unwrap();
  433. // trace!("json_value:{:?}",json_value.to_string().as_str());
  434. let code: i64 = if json_value["retCode"].as_i64().is_some() {
  435. json_value["retCode"].as_i64().unwrap()
  436. } else if json_value["ret_code"].as_i64().is_some() {
  437. json_value["ret_code"].as_i64().unwrap()
  438. } else {
  439. -1
  440. };
  441. let msg: &str = if json_value["retMsg"].as_str().is_some() {
  442. json_value["retMsg"].as_str().unwrap()
  443. } else if json_value["ret_msg"].as_str().is_some() {
  444. json_value["ret_msg"].as_str().unwrap()
  445. } else {
  446. ""
  447. };
  448. if code == 0 {
  449. let data = serde_json::to_string(&json_value["result"]).unwrap();
  450. let mut success = ResponseData::new("".to_string(), "200".to_string(), "success".to_string(), data.parse().unwrap());
  451. success.time = json_value["time"].as_i64().unwrap();
  452. success
  453. } else {
  454. let error = ResponseData::new("".to_string(),
  455. format!("{}", code),
  456. format!("{}", msg),
  457. format!("请求地址:{},请求参数:{}", base_url, params));
  458. error
  459. }
  460. }
  461. }
  462. Err(err) => {
  463. let error = ResponseData::error("".to_string(), format!("json 解析失败:{}", err));
  464. error
  465. }
  466. }
  467. }
  468. fn get_timestamp() -> String {
  469. chrono::Utc::now().timestamp_millis()
  470. .to_string()
  471. }
  472. }