extended_rest_client.rs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. use reqwest::Client;
  2. use reqwest::header::HeaderMap;
  3. use serde_json::{json, Value};
  4. use tracing::{error, info};
  5. use crate::exchange::extended_account::ExtendedAccount;
  6. use crate::utils::response::Response;
  7. use crate::utils::rest_utils::RestUtils;
  8. pub struct ExtendedRestClient {
  9. pub tag: String,
  10. // 一些私有变量
  11. base_url: String,
  12. client: Client,
  13. account: Option<ExtendedAccount>,
  14. // 延迟统计工具
  15. delays: Vec<i64>,
  16. max_delay: i64,
  17. avg_delay: i64,
  18. }
  19. impl ExtendedRestClient {
  20. pub fn new(tag: String, account: Option<ExtendedAccount>) -> Self {
  21. ExtendedRestClient {
  22. tag,
  23. base_url: "https://api.starknet.extended.exchange".to_string(),
  24. // base_url: "https://api.starknet.sepolia.extended.exchange".to_string(),
  25. client: Client::new(),
  26. account,
  27. delays: vec![],
  28. max_delay: 0,
  29. avg_delay: 0,
  30. }
  31. }
  32. // =================================== 公共方法区 ====================================
  33. // =================================== 私有方法区,这边仅需要携带header ====================================
  34. pub async fn cancel_order(&mut self, external_id: String) -> Response {
  35. let params = json!({
  36. "external_id": external_id,
  37. });
  38. self.request("DELETE".to_string(),
  39. "/api/v1/user".to_string(),
  40. "/order".to_string(),
  41. true,
  42. false,
  43. params,
  44. ).await
  45. }
  46. // =================================== 签名方法区,这边需要签名=========================
  47. // =================================== 网络层基础 ====================================
  48. // 发送请求
  49. pub async fn request(&mut self,
  50. method: String,
  51. prefix_url: String,
  52. request_url: String,
  53. is_private: bool,
  54. _is_sign: bool,
  55. mut params: Value) -> Response
  56. {
  57. // ----------------- 每个接口都有的公共参数 ------------------
  58. // let timestamp = Utc::now().timestamp_millis();
  59. // let recv_window = 3000;
  60. // params["timestamp"] = serde_json::json!(timestamp);
  61. // params["recvWindow"] = serde_json::json!(recv_window);
  62. // ------------------- 请求头填充 --------------------------
  63. let mut headers = HeaderMap::new();
  64. headers.insert("User-Agent", "RustClient/1.0".parse().unwrap());
  65. // ---------------- 请求类型不同,可能请求头body不同 ----------
  66. let mut body = "{}".to_string();
  67. if method == "POST" {
  68. headers.insert("Content-Type", "application/json".parse().unwrap());
  69. body = params.to_string();
  70. }
  71. // ---------------- 签名操作等等 ----------------------------
  72. if is_private {
  73. if self.account.is_none() {
  74. let e = Response::error(self.tag.clone(), "需要补齐登录参数".to_string());
  75. return e;
  76. } else {
  77. // 进行签名等操作
  78. headers.insert("X-Api-Key", self.account.clone().unwrap().api_key.parse().unwrap());
  79. // //组装sing
  80. // let sing = Self::sign(secret_key.clone(),
  81. // method.clone(),
  82. // prefix_url.clone(),
  83. // request_url.clone(),
  84. // params.clone(),
  85. // body.clone(),
  86. // timestamp.clone(),
  87. // );
  88. // //组装header
  89. // headers.extend(Self::headers(sing, timestamp, passphrase, access_key));
  90. }
  91. }
  92. // --------------------- 最终发送 ---------------------------
  93. let start_time = chrono::Utc::now().timestamp_millis();
  94. let response = self.do_request(
  95. format!("{}{}", prefix_url.clone(), request_url.clone()),
  96. method.to_string(),
  97. params.to_string(),
  98. body,
  99. headers,
  100. ).await;
  101. let time_array = chrono::Utc::now().timestamp_millis() - start_time;
  102. self.delays.push(time_array);
  103. self.get_delay_info();
  104. response
  105. }
  106. async fn do_request(&mut self, request_path: String,
  107. request_type: String,
  108. params: String,
  109. body: String,
  110. headers: HeaderMap) -> Response {
  111. let url = format!("{}{}", self.base_url.to_string(), request_path);
  112. let request_type = request_type.clone().to_uppercase();
  113. let params_str = RestUtils::parse_params_to_str(params.clone());
  114. let addrs_url: String = if params_str == "" {
  115. url.clone()
  116. } else {
  117. format!("{}?{}", url.clone(), params_str)
  118. };
  119. let request_builder = match request_type.as_str() {
  120. "GET" => self.client.get(addrs_url.clone()).headers(headers),
  121. "POST" => self.client.post(url.clone()).body(body).headers(headers),
  122. "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
  123. "PUT" => self.client.put(url.clone()).json(&params),
  124. _ => {
  125. panic!("{}", format!("错误的请求类型:{}", request_type.clone()))
  126. }
  127. };
  128. // 读取响应的内容
  129. let response = request_builder.send().await.unwrap();
  130. // 先检查状态码
  131. let is_success = response.status().is_success();
  132. let text = response.text().await.unwrap();
  133. info!(text);
  134. if is_success {
  135. self.on_success_data(&text)
  136. } else {
  137. self.on_error_data(&text, &addrs_url, &params)
  138. }
  139. }
  140. pub fn on_success_data(&mut self, text: &String) -> Response {
  141. let json_value = serde_json::from_str::<Value>(&text).unwrap();
  142. Response::new(self.tag.clone(), 200, "success".to_string(), json_value)
  143. }
  144. pub fn on_error_data(&mut self, text: &String, base_url: &String, params: &String) -> Response {
  145. let json_value = serde_json::from_str::<Value>(&text);
  146. match json_value {
  147. Ok(data) => {
  148. let message;
  149. if !data["message"].is_null() {
  150. message = format!("{}:{}", data["tag"].as_str().unwrap(), data["message"].as_str().unwrap());
  151. } else {
  152. message = data["tag"].to_string();
  153. }
  154. let mut error = Response::error(self.tag.clone(), message);
  155. error.message = format!("请求地址:{}, 请求参数:{}, 报错内容:{}。", base_url, params, error.message);
  156. error
  157. }
  158. Err(e) => {
  159. error!("解析错误:{:?}", e);
  160. let error = Response::error("".to_string(), text.clone());
  161. error
  162. }
  163. }
  164. }
  165. // =================================== 延迟统计相关 ==================================
  166. pub fn get_delays(&self) -> Vec<i64> {
  167. self.delays.clone()
  168. }
  169. pub fn get_avg_delay(&self) -> i64 {
  170. self.avg_delay.clone()
  171. }
  172. pub fn get_max_delay(&self) -> i64 {
  173. self.max_delay.clone()
  174. }
  175. fn get_delay_info(&mut self) {
  176. let last_100 = if self.delays.len() > 100 {
  177. self.delays[self.delays.len() - 100..].to_vec()
  178. } else {
  179. self.delays.clone()
  180. };
  181. let max_value = last_100.iter().max().unwrap();
  182. if max_value.clone().to_owned() > self.max_delay {
  183. self.max_delay = max_value.clone().to_owned();
  184. }
  185. let sum: i64 = last_100.iter().sum();
  186. let len = last_100.len() as i64;
  187. self.avg_delay = sum / len;
  188. self.delays = last_100.clone().into_iter().collect();
  189. }
  190. }
  191. #[cfg(test)]
  192. mod tests {
  193. use tracing::info;
  194. use crate::exchange::extended_account::ExtendedAccount;
  195. use crate::exchange::extended_rest_client::ExtendedRestClient;
  196. use crate::utils::log_setup::setup_logging;
  197. #[tokio::test]
  198. async fn test_cancel_order() {
  199. let _guard = setup_logging().unwrap();
  200. let account = ExtendedAccount::new("a7b197d06d35de11387b8b71f34c87e4".to_string());
  201. let mut client = ExtendedRestClient::new("Extended".to_string(), Some(account));
  202. let response = client.cancel_order("123456".to_string()).await;
  203. info!("{:?}", response);
  204. info!("{}", serde_json::to_string_pretty(&response.data).unwrap());
  205. }
  206. }