| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584 |
- use std::collections::BTreeMap;
- use reqwest::header::HeaderMap;
- use ring::{digest};
- use hex;
- use hmac::{Hmac, Mac, NewMac};
- use reqwest::Client;
- use rust_decimal::Decimal;
- use rust_decimal::prelude::FromPrimitive;
- use rust_decimal_macros::dec;
- use serde_json::Value;
- use crate::http_tool::RestTool;
- use crate::response_base::ResponseData;
- use sha2::Sha512;
- use tracing::{error, info, trace};
- #[derive(Clone)]
- pub struct GateSwapRest {
- label: String,
- base_url: String,
- client: reqwest::Client,
- /*******参数*/
- //登陆所需参数
- login_param: BTreeMap<String, String>,
- delays: Vec<i64>,
- max_delay: i64,
- avg_delay: Decimal,
- }
- impl GateSwapRest {
- /*******************************************************************************************************/
- /*****************************************获取一个对象****************************************************/
- /*******************************************************************************************************/
- pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> GateSwapRest
- {
- return GateSwapRest::new_label("default-GateSwapRest".to_string(), is_colo, login_param);
- }
- pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> GateSwapRest
- {
- let base_url = if is_colo {
- // "https://apiv4-private.gateapi.io".to_string()
- "https://api.gateio.ws".to_string()
- } else {
- "https://api.gateio.ws".to_string()
- };
- if is_colo {
- info!("开启高速(未配置,走普通:{})通道",base_url);
- } else {
- info!("走普通通道:{}",base_url);
- }
- /*****返回结构体*******/
- GateSwapRest {
- label,
- base_url: base_url.to_string(),
- client: Client::new(),
- login_param,
- delays: vec![],
- max_delay: 0,
- avg_delay: dec!(0.0),
- }
- }
- /*******************************************************************************************************/
- /*****************************************rest请求函数********************************************************/
- /*******************************************************************************************************/
- //获取服务器当前时间
- pub async fn get_server_time(&mut self) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/spot/time"),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询个人交易费率
- pub async fn wallet_fee(&mut self) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/wallet/fee"),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询合约账户
- pub async fn get_account(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/accounts", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //用户仓位列表
- pub async fn get_user_position(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/positions", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //双仓模式下的持仓信息
- pub async fn get_position(&mut self, settle: String, contract: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/dual_comp/positions/{}", settle, contract),
- true,
- params.to_string(),
- ).await;
- data
- }
- //获取所有合约交易行情统计
- pub async fn get_ticker(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/tickers", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询所有的合约信息
- pub async fn get_market_details(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/contracts", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询单个订单详情
- pub async fn get_order_details(&mut self, settle: String, order_id: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/orders/{}", settle, order_id),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询合约订单列表
- pub async fn get_orders(&mut self, settle: String, status: String) -> ResponseData {
- let params = serde_json::json!({
- "status":status
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/orders", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //下单:side-下单方向,pos_side-持仓方向
- pub async fn order(&mut self,
- settle: String,
- pos_side: String,
- side: String,
- contract: String,
- size: i64,
- price: String,
- text: String,
- ) -> ResponseData
- {
- if side != "buy" && side != "sell" {
- ResponseData::error(self.label.clone(), format!("未知下单方向!{}", side));
- }
- if pos_side != "long" && pos_side != "short" {
- ResponseData::error(self.label.clone(), format!("未知持仓方向!{}", side));
- }
- let mut param = serde_json::json!({
- "contract":contract, //合约标识
- "size":size,
- "price":price,
- "text":text,
- });
- if price == "0" {
- param.as_object_mut().unwrap().insert("tif".to_string(), serde_json::json!("ioc"));
- }
- if size == 0 { //数量为0则平仓
- param.as_object_mut().unwrap().insert("close".to_string(), serde_json::json!(true));
- }
- match format!("{}_{}", pos_side, side).as_str() {
- "long_buy" => {//开多
- param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(false));
- }
- "long_sell" => {//平多
- param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(true));
- }
- "short_buy" => {//平空
- param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(true));
- }
- "short_sell" => {//开空
- param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(false));
- }
- _ => {} // 处理未知请求类型
- };
- trace!("----param{}", param.to_string());
- let data = self.swap_order(settle, param).await;
- data
- }
- //合约交易下单
- pub async fn swap_order(&mut self, settle: String, params: serde_json::Value) -> ResponseData {
- let data = self.request("POST".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/orders", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //设置持仓模式
- pub async fn setting_dual_mode(&mut self, settle: String, dual_mode: bool) -> ResponseData {
- let params = serde_json::json!({
- "dual_mode":dual_mode,
- });
- let data = self.request("POST".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/dual_mode", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //更新双仓模式下的杠杆
- pub async fn setting_dual_leverage(&mut self, settle: String, symbol: String, leverage: String) -> ResponseData {
- let params = serde_json::json!({
- "leverage":leverage,
- });
- let data = self.request("POST".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/dual_comp/positions/{}/leverage", settle, symbol),
- true,
- params.to_string(),
- ).await;
- data
- }
- //交易账户互转
- pub async fn wallet_transfers(&mut self, currency: String, from: String, to: String, amount: String, settle: String) -> ResponseData {
- let params = serde_json::json!({
- "currency":currency,
- "from":from,
- "to":to,
- "amount":amount,
- "settle":settle,
- });
- let data = self.request("POST".to_string(),
- "/api/v4".to_string(),
- format!("/wallet/transfers"),
- true,
- params.to_string(),
- ).await;
- data
- }
- //撤销单个订单
- pub async fn cancel_order(&mut self, settle: String, order_id: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("DELETE".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/orders/{}", settle, order_id),
- true,
- params.to_string(),
- ).await;
- data
- }
- //弃用
- pub async fn swap_bazaar_order(&mut self, text: String, origin_side: String, settle: String, contract: String, size: i64) -> ResponseData {
- let mut reduce_only = false;
- let mut param = serde_json::json!({
- "text":text,
- "contract":contract,
- "price":"0",
- "size":size,
- });
- let req = match origin_side.as_str() {
- "kd" => {
- reduce_only = false;
- true
- }
- "pd" => {
- reduce_only = true;
- true
- }
- "kk" => {
- reduce_only = false;
- true
- }
- "pk" => {
- reduce_only = true;
- true
- }
- _ => { false } // 处理未知请求类型
- };
- if req {
- param.as_object_mut().unwrap().insert("reduce_only".to_string(), serde_json::json!(reduce_only));
- }
- let data = self.swap_order(settle, param).await;
- data
- }
- //批量取消状态为 open 的订单
- pub async fn cancel_orders(&mut self, settle: String, contract: String) -> ResponseData {
- let params = serde_json::json!({
- "contract":contract
- });
- let data = self.request("DELETE".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/orders", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询个人成交记录
- pub async fn my_trades(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/my_trades", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- //查询合约账户变更历史
- pub async fn account_book(&mut self, settle: String) -> ResponseData {
- let params = serde_json::json!({
- "limit":200
- });
- let data = self.request("GET".to_string(),
- "/api/v4".to_string(),
- format!("/futures/{}/account_book", settle),
- true,
- params.to_string(),
- ).await;
- data
- }
- /*******************************************************************************************************/
- /*****************************************工具函数********************************************************/
- /*******************************************************************************************************/
- pub fn get_delays(&self) -> Vec<i64> {
- self.delays.clone()
- }
- pub fn get_avg_delay(&self) -> Decimal {
- self.avg_delay.clone()
- }
- pub fn get_max_delay(&self) -> i64 {
- self.max_delay.clone()
- }
- fn get_delay_info(&mut self) {
- let last_100 = if self.delays.len() > 100 {
- self.delays[self.delays.len() - 100..].to_vec()
- } else {
- self.delays.clone()
- };
- let max_value = last_100.iter().max().unwrap();
- if max_value.clone().to_owned() > self.max_delay {
- self.max_delay = max_value.clone().to_owned();
- }
- let sum: i64 = last_100.iter().sum();
- let sum_v = Decimal::from_i64(sum).unwrap();
- let len_v = Decimal::from_u64(last_100.len() as u64).unwrap();
- self.avg_delay = (sum_v / len_v).round_dp(1);
- self.delays = last_100.clone().into_iter().collect();
- }
- //调用请求
- async fn request(&mut self,
- requesst_type: String,
- prefix_url: String,
- request_url: String,
- is_login: bool,
- params: String) -> ResponseData
- {
- // trace!("login_param:{:?}", self.login_param);
- //解析账号信息
- let mut access_key = "".to_string();
- let mut secret_key = "".to_string();
- if self.login_param.contains_key("access_key") {
- access_key = self.login_param.get("access_key").unwrap().to_string();
- }
- if self.login_param.contains_key("secret_key") {
- secret_key = self.login_param.get("secret_key").unwrap().to_string();
- }
- let mut is_login_param = true;
- if access_key == "" || secret_key == "" {
- is_login_param = false
- }
- //请求头配置-如果需要登陆则存在额外配置
- let mut body = "".to_string();
- let timestamp = chrono::Utc::now().timestamp().to_string();
- let mut headers = HeaderMap::new();
- if requesst_type == "GET" {
- headers.insert("Content-type", "application/x-www-form-urlencoded".parse().unwrap());
- headers.insert("User-Agent", "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) ".parse().unwrap());
- } else {
- headers.insert("Accept", "application/json".parse().unwrap());
- headers.insert("Content-Type", "application/json".parse().unwrap());
- }
- if requesst_type == "POST" {
- body = params.clone();
- }
- //是否需要登陆-- 组装sing
- if is_login {
- if !is_login_param {
- let e = ResponseData::error(self.label.clone(), "登陆参数错误!".to_string());
- return e;
- } else {//需要登陆-且登陆参数齐全
- //组装sing
- let sing = Self::sign(secret_key.clone(),
- requesst_type.clone(),
- prefix_url.clone(),
- request_url.clone(),
- params.clone(),
- body.clone(),
- timestamp.clone(),
- );
- // trace!("sing:{}", sing);
- //组装header
- headers.extend(Self::headers(access_key, timestamp, sing));
- }
- }
- // trace!("headers:{:?}", headers);
- let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
- let start_time = chrono::Utc::now().timestamp_millis();
- let get_response = self.http_toll(
- base_url.clone(),
- requesst_type.to_string(),
- params.clone(),
- headers,
- ).await;
- let time_array = chrono::Utc::now().timestamp_millis() - start_time;
- self.delays.push(time_array);
- self.get_delay_info();
- let res_data = Self::res_data_analysis(get_response, base_url, params);
- res_data
- }
- pub fn headers(access_key: String, timestamp: String, sign: String) -> HeaderMap {
- let mut headers = HeaderMap::new();
- headers.insert("KEY", access_key.clone().parse().unwrap());
- headers.insert("Timestamp", timestamp.clone().parse().unwrap());
- headers.insert("SIGN", sign.clone().parse().unwrap());
- headers
- }
- pub fn sign(secret_key: String,
- requesst_type: String, prefix_url: String, request_url: String,
- params: String, body_data: String, timestamp: String) -> String
- {
- let url = format!("{}{}", prefix_url, request_url);
- let params_str = RestTool::parse_params_to_str(params);
- let body = Some(body_data);
- let hashed_payload = if let Some(body) = body {
- let mut m = digest::Context::new(&digest::SHA512);
- m.update(body.as_bytes());
- hex::encode(m.finish().as_ref())
- } else {
- String::new()
- };
- // trace!("hashed_payload:{}", hashed_payload);
- let message = format!("{}\n{}\n{}\n{}\n{}",
- requesst_type,
- url,
- params_str,
- hashed_payload,
- timestamp);
- // trace!("**********", );
- // trace!("组装数据:{}", message);
- // trace!("**********", );
- let mut mac = Hmac::<Sha512>::new_varkey(secret_key.as_bytes()).expect("Failed to create HMAC");
- mac.update(message.as_bytes());
- let result = mac.finalize().into_bytes();
- let sign = hex::encode(result);
- sign
- }
- async fn http_toll(&mut self, request_path: String, request_type: String, params: String, headers: HeaderMap) -> Result<ResponseData, reqwest::Error> {
- let res_data: ResponseData;
- /****请求接口与 地址*/
- let url = format!("{}{}", self.base_url.to_string(), request_path);
- let request_type = request_type.clone().to_uppercase();
- let addrs_url = format!("{}?{}", url.clone(), RestTool::parse_params_to_str(params.clone()));
- // let params_json: serde_json::Value = serde_json::from_str(¶ms).unwrap();
- // trace!("url:{}",url);
- // trace!("addrs_url:{}",url);
- // trace!("params_json:{}",params_json);
- // trace!("headers:{:?}",headers);
- let req = match request_type.as_str() {
- "GET" => self.client.get(addrs_url.clone()).headers(headers),
- "POST" => self.client.post(addrs_url.clone()).body(params).headers(headers),
- "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
- // "PUT" => self.client.put(url.clone()).json(¶ms),
- _ => return Ok(ResponseData::error(self.label.clone(), format!("错误的请求类型:{}", request_type.clone()))), // 处理未知请求类型
- };
- let response = req.send().await?;
- if response.status().is_success() {
- // 读取响应的内容
- let body = response.text().await?;
- // trace!("ok-----{}", body);
- res_data = ResponseData::new(self.label.clone(), "200".to_string(), "success".to_string(), body);
- } else {
- let body = response.text().await?;
- // trace!("error-----{}", body);
- res_data = ResponseData::error(self.label.clone(), body.to_string())
- }
- Ok(res_data)
- }
- //res_data 解析
- pub fn res_data_analysis(result: Result<ResponseData, reqwest::Error>, base_url: String, params: String) -> ResponseData {
- match result {
- Ok(res_data) => {
- if res_data.code != "200" {
- let message = res_data.message;
- // let json_value: serde_json::Value = serde_json::from_str(&message).unwrap();//这种方式会触发 解析错误
- let json_value = serde_json::from_str::<Value>(&message);
- match json_value {
- Ok(data) => {
- let label = data["label"].as_str().unwrap();
- let mut error = ResponseData::error(res_data.label, label.parse().unwrap());
- error.data = format!("请求地址:{},请求参数:{}", base_url, params);
- error
- }
- Err(e) => {
- error!("解析错误:{:?}",e);
- let error = ResponseData::error("".to_string(), format!("json 解析失败:{},相关参数:{}", e, message));
- error
- }
- }
- } else {
- res_data
- }
- }
- Err(err) => {
- let error = ResponseData::error("".to_string(), format!("json 解析失败:{},相关参数:{}", err, params));
- error
- }
- }
- }
- }
|