phemex_swap_rest.rs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. use std::collections::BTreeMap;
  2. use chrono::Utc;
  3. use reqwest::{Client, Proxy};
  4. use reqwest::header::HeaderMap;
  5. use ring::hmac;
  6. use rust_decimal::Decimal;
  7. use rust_decimal::prelude::FromPrimitive;
  8. use rust_decimal_macros::dec;
  9. use serde_json::{json, Value};
  10. use tracing::{error, info, trace};
  11. use crate::http_tool::RestTool;
  12. use crate::proxy;
  13. use crate::response_base::ResponseData;
  14. #[derive(Clone, Debug)]
  15. pub struct PhemexSwapRest {
  16. pub tag: String,
  17. base_url: String,
  18. /*******参数*/
  19. //是否需要登录
  20. //登录所需参数
  21. login_param: BTreeMap<String, String>,
  22. delays: Vec<i64>,
  23. max_delay: i64,
  24. avg_delay: Decimal,
  25. }
  26. impl PhemexSwapRest {
  27. /*******************************************************************************************************/
  28. /*****************************************获取一个对象****************************************************/
  29. /*******************************************************************************************************/
  30. pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> PhemexSwapRest
  31. {
  32. return PhemexSwapRest::new_with_tag("default-PhemexSwapRest".to_string(), is_colo, login_param);
  33. }
  34. pub fn new_with_tag(tag: String, is_colo: bool, login_param: BTreeMap<String, String>) -> PhemexSwapRest {
  35. let base_url = if is_colo {
  36. let z = "https://api.phemex.com".to_string();
  37. info!("开启高速(未配置,走普通:{})通道",z);
  38. z
  39. } else {
  40. let z = "https://api.phemex.com".to_string();
  41. info!("走普通通道:{}",z);
  42. z
  43. };
  44. /*****返回结构体*******/
  45. PhemexSwapRest {
  46. tag,
  47. base_url,
  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(&mut self) -> ResponseData {
  59. let params = json!({});
  60. let data = self.request("GET".to_string(),
  61. "".to_string(),
  62. "/public/time".to_string(),
  63. false,
  64. params,
  65. ).await;
  66. data
  67. }
  68. //查詢合約基礎信息
  69. pub async fn get_market(&mut self, params: Value) -> ResponseData {
  70. let data = self.request("GET".to_string(),
  71. "".to_string(),
  72. "/public/products".to_string(),
  73. false,
  74. params,
  75. ).await;
  76. data
  77. }
  78. //查詢ticker(Query 24 ticker)
  79. pub async fn get_ticker(&mut self, params: Value) -> ResponseData {
  80. let data = self.request("GET".to_string(),
  81. "".to_string(),
  82. "/md/v3/ticker/24hr".to_string(),
  83. false,
  84. params,
  85. ).await;
  86. data
  87. }
  88. //持仓(查询交易账户和仓位)
  89. pub async fn get_account_and_positions(&mut self, params: Value) -> ResponseData {
  90. let data = self.request("GET".to_string(),
  91. "".to_string(),
  92. "/g-accounts/accountPositions".to_string(),
  93. true,
  94. params,
  95. ).await;
  96. data
  97. }
  98. //仓位设置(Switch Position Mode Synchronously)
  99. pub async fn set_target_pos_mode(&mut self, params: Value) -> ResponseData {
  100. let data = self.request("PUT".to_string(),
  101. "".to_string(),
  102. "/g-positions/switch-pos-mode-sync".to_string(),
  103. true,
  104. params,
  105. ).await;
  106. data
  107. }
  108. //设置杠杆(Set Leverage 设置杠杆)
  109. pub async fn set_leverage(&mut self, params: Value) -> ResponseData {
  110. let data = self.request("PUT".to_string(),
  111. "".to_string(),
  112. "/g-positions/leverage".to_string(),
  113. true,
  114. params,
  115. ).await;
  116. data
  117. }
  118. //下单
  119. pub async fn orders(&mut self, params: Value) -> ResponseData {
  120. let data = self.request("PUT".to_string(),
  121. "".to_string(),
  122. "/g-orders/create".to_string(),
  123. true,
  124. params,
  125. ).await;
  126. data
  127. }
  128. //撤单
  129. pub async fn cancel_order(&mut self, params: Value) -> ResponseData {
  130. let data = self.request("DELETE".to_string(),
  131. "".to_string(),
  132. "/g-orders/cancel".to_string(),
  133. true,
  134. params,
  135. ).await;
  136. data
  137. }
  138. //撤销所有
  139. pub async fn cancel_order_all(&mut self, params: Value) -> ResponseData {
  140. let data = self.request("DELETE".to_string(),
  141. "".to_string(),
  142. "/g-orders/all".to_string(),
  143. true,
  144. params,
  145. ).await;
  146. data
  147. }
  148. //订单列表
  149. pub async fn get_orders(&mut self, params: Value) -> ResponseData {
  150. let data = self.request("GET".to_string(),
  151. "".to_string(),
  152. "/api-data/g-futures/orders".to_string(),
  153. true,
  154. params,
  155. ).await;
  156. data
  157. }
  158. //根据id查询订单
  159. pub async fn get_orders_by_id(&mut self, params: Value) -> ResponseData {
  160. let data = self.request("GET".to_string(),
  161. "".to_string(),
  162. "/api-data/g-futures/orders/by-order-id".to_string(),
  163. true,
  164. params,
  165. ).await;
  166. data
  167. }
  168. /*******************************************************************************************************/
  169. /*****************************************工具函数********************************************************/
  170. /*******************************************************************************************************/
  171. pub fn get_delays(&self) -> Vec<i64> {
  172. self.delays.clone()
  173. }
  174. pub fn get_avg_delay(&self) -> Decimal {
  175. self.avg_delay.clone()
  176. }
  177. pub fn get_max_delay(&self) -> i64 {
  178. self.max_delay.clone()
  179. }
  180. fn get_delay_info(&mut self) {
  181. let last_100 = if self.delays.len() > 100 {
  182. self.delays[self.delays.len() - 100..].to_vec()
  183. } else {
  184. self.delays.clone()
  185. };
  186. let max_value = last_100.iter().max().unwrap();
  187. if max_value.clone().to_owned() > self.max_delay {
  188. self.max_delay = max_value.clone().to_owned();
  189. }
  190. let sum: i64 = last_100.iter().sum();
  191. let sum_v = Decimal::from_i64(sum).unwrap();
  192. let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
  193. self.avg_delay = (sum_v / len_v).round_dp(1);
  194. self.delays = last_100.clone().into_iter().collect();
  195. }
  196. //调用请求
  197. pub async fn request(&mut self,
  198. method: String,
  199. prefix_url: String,
  200. request_url: String,
  201. is_login: bool,
  202. params_json: Value) -> ResponseData
  203. {
  204. trace!("login_param:{:?}", self.login_param);
  205. //解析账号信息
  206. let mut access_key = "".to_string();
  207. let mut secret_key = "".to_string();
  208. if self.login_param.contains_key("access_key") {
  209. access_key = self.login_param.get("access_key").unwrap().to_string();
  210. }
  211. if self.login_param.contains_key("secret_key") {
  212. secret_key = self.login_param.get("secret_key").unwrap().to_string();
  213. }
  214. let mut is_login_param = true;
  215. if access_key == "" || secret_key == "" {
  216. is_login_param = false
  217. }
  218. //每个接口都有的参数
  219. let timestamp = (Utc::now().timestamp_millis() + (60 * 1000)) / 1000;
  220. //请求类型不同,可能请求头body 不同
  221. let mut body = "".to_string();
  222. let mut params = "".to_string();
  223. let mut headers = HeaderMap::new();
  224. if method == "POST" {
  225. body = params_json.to_string();
  226. }
  227. if method == "GET" || method == "PUT" || method == "DELETE" {
  228. let z = params_json.to_string();
  229. params = RestTool::parse_params_to_str(z);
  230. }
  231. //是否需要登录-- 组装sing
  232. if is_login {
  233. if !is_login_param {
  234. let e = ResponseData::error(self.tag.clone(), "登录参数错误!".to_string());
  235. return e;
  236. } else {
  237. //需要登录-且登录参数齐全
  238. trace!("Path:{}{}", prefix_url.clone(),request_url.clone());
  239. trace!("Query:{}", params);
  240. trace!("Body:{}", body);
  241. trace!("expire:{}", timestamp.to_string());
  242. //组装sing
  243. let sing = Self::sign(secret_key.clone(),
  244. prefix_url.clone(),
  245. request_url.clone(),
  246. params.clone(),
  247. body.clone(),
  248. timestamp.to_string(),
  249. );
  250. trace!("Signature:{}", sing);
  251. //组装header
  252. headers.extend(Self::headers(sing, timestamp.to_string(), access_key));
  253. }
  254. }
  255. let start_time = chrono::Utc::now().timestamp_millis();
  256. let response = self.http_tool(
  257. format!("{}{}", prefix_url.clone(), request_url.clone()),
  258. method.to_string(),
  259. params.clone(),
  260. body.clone(),
  261. headers,
  262. is_login,
  263. ).await;
  264. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  265. self.delays.push(time_array);
  266. self.get_delay_info();
  267. response
  268. }
  269. pub fn headers(sign: String, timestamp: String, access_key: String) -> HeaderMap {
  270. let mut headers = HeaderMap::new();
  271. headers.insert("x-phemex-access-token", access_key.parse().unwrap());// 这是 Phemex 网站的 API-KEY(id 字段)
  272. headers.insert("x-phemex-request-expiry", timestamp.parse().unwrap());// 描述请求过期的 Unix EPoch 秒数,通常应为 (Now() + 1 分钟)
  273. headers.insert("x-phemex-request-signature", sign.parse().unwrap());// 这是 http 请求的 HMAC SHA256 签名。Secret 是 API Secret,其公式为:HMacSha256(URL Path + QueryString + Expiry + body)
  274. // let tracing = format!("4l_{:?}", Utc::now().timestamp_millis().to_string());
  275. // headers.insert("x-phemex-request-tracing", tracing.parse().unwrap());
  276. headers
  277. }
  278. pub fn sign(secret_key: String,
  279. prefix_url: String, request_url: String,
  280. url_param_str: String, body: String, timestamp: String) -> String
  281. {
  282. /*签名生成*/
  283. let base_url = format!("{}{}", prefix_url, request_url);
  284. // HMacSha256(URL Path + QueryString + Expiry + body)
  285. // 时间戳 + 请求类型+ 请求参数字符串
  286. let message = format!("{}{}{}{}", base_url, url_param_str, timestamp, body);
  287. trace!("message:{}",message);
  288. // 做签名
  289. let hmac_key = ring::hmac::Key::new(hmac::HMAC_SHA256, secret_key.as_bytes());
  290. let sign = hex::encode(hmac::sign(&hmac_key, message.as_bytes()).as_ref());
  291. sign
  292. }
  293. // async fn http_tool(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<ResponseData, reqwest::Error> {
  294. async fn http_tool(&mut self, request_path: String,
  295. request_type: String,
  296. params: String,
  297. body: String,
  298. headers: HeaderMap,
  299. is_login: bool) -> ResponseData {
  300. /****请求接口与 地址*/
  301. let url = format!("{}{}", self.base_url.to_string(), request_path);
  302. let request_type = request_type.clone().to_uppercase();
  303. let addrs_url: String = if params == "" {
  304. url.clone()
  305. } else {
  306. format!("{}?{}", url.clone(), params)
  307. };
  308. trace!("url-----:{}",url.clone());
  309. trace!("addrs_url-----:{}",addrs_url.clone());
  310. trace!("params-----:{}",params.clone());
  311. trace!("body-----:{}",body.clone());
  312. trace!("headers-----:{:?}",headers.clone());
  313. trace!("request_type-----:{:?}",request_type.clone());
  314. let client = if is_login {
  315. let params = proxy::ParsingDetail::http_enable_proxy(Some("phemex"));
  316. let client_re;
  317. if params {
  318. let proxy_address = "socks5://127.0.0.1:17890"; // 替换为你的 SOCKS5 代理地址
  319. let proxy = Proxy::all(proxy_address).unwrap();
  320. client_re = Client::builder().proxy(proxy).build().unwrap();
  321. } else {
  322. client_re = Client::new()
  323. }
  324. client_re
  325. } else {
  326. proxy::ParsingDetail::http_enable_proxy(None);
  327. Client::new()
  328. };
  329. let request_builder = match request_type.as_str() {
  330. "GET" => client.get(addrs_url.clone()).headers(headers),
  331. "POST" => client.post(url.clone()).body(body).headers(headers),
  332. "DELETE" => client.delete(addrs_url.clone()).headers(headers),
  333. "PUT" => client.put(addrs_url.clone()).headers(headers),
  334. _ => {
  335. panic!("{}", format!("错误的请求类型:{}", request_type.clone()))
  336. }
  337. };
  338. // 读取响应的内容
  339. let response = request_builder.send().await.unwrap();
  340. let is_success = response.status().is_success(); // 先检查状态码
  341. let text = response.text().await.unwrap();
  342. // trace!("text:{:?}",text);
  343. return if is_success {
  344. self.on_success_data(&text)
  345. } else {
  346. self.on_error_data(&text, &addrs_url, &params)
  347. };
  348. }
  349. pub fn on_success_data(&mut self, text: &String) -> ResponseData {
  350. let json_value = serde_json::from_str::<Value>(&text).unwrap();
  351. // return ResponseData::new(self.tag.clone(), 200, "success".to_string(), json_value);
  352. let id = json_value["id"].as_i64();
  353. match id {
  354. None => {}
  355. Some(v) => {
  356. match v {
  357. 0 => {
  358. let result = json_value.get("result").unwrap();
  359. return ResponseData::new(self.tag.clone(), 200, "success".to_string(), result.clone());
  360. }
  361. _ => {}
  362. }
  363. }
  364. }
  365. let code = json_value["code"].as_i64();
  366. match code {
  367. None => {}
  368. Some(v) => {
  369. match v {
  370. 0 => {
  371. //判断是否有code ,没有表示特殊接口,直接返回
  372. if json_value.get("data").is_some() {
  373. let data = json_value.get("data").unwrap();
  374. return ResponseData::new(self.tag.clone(), 200, "success".to_string(), data.clone());
  375. } else {
  376. return ResponseData::new(self.tag.clone(), 200, "success".to_string(), json_value);
  377. }
  378. }
  379. _ => {
  380. if json_value.get("msg").is_some() {
  381. return ResponseData::new(self.tag.clone(), 400, format!("{:?}", json_value["msg"].as_str()), json_value);
  382. } else {
  383. return ResponseData::new(self.tag.clone(), 400, "error".to_string(), json_value);
  384. }
  385. }
  386. }
  387. }
  388. }
  389. return ResponseData::new(self.tag.clone(), 400, "error".to_string(), json_value);
  390. }
  391. pub fn on_error_data(&mut self, text: &String, _base_url: &String, _params: &String) -> ResponseData {
  392. let json_value = serde_json::from_str::<Value>(&text);
  393. match json_value {
  394. Ok(data) => {
  395. let code = data["code"].as_i64();
  396. match code {
  397. None => {}
  398. Some(v) => {
  399. match v {
  400. _ => {
  401. if data.get("msg").is_some() {
  402. return ResponseData::new(self.tag.clone(), 400, format!("{:?}", data["msg"].as_str().unwrap()), data);
  403. } else {
  404. return ResponseData::new(self.tag.clone(), 400, "error".to_string(), data);
  405. }
  406. }
  407. }
  408. }
  409. }
  410. }
  411. Err(e) => {
  412. error!("解析错误:{:?}", e);
  413. return ResponseData::error("".to_string(),
  414. format!("json 解析失败:{},相关参数:{}", e, text));
  415. }
  416. }
  417. return ResponseData::error("".to_string(), format!("请求失败:{:?}", text));
  418. }
  419. }