|
@@ -1,449 +0,0 @@
|
|
|
-use std::collections::BTreeMap;
|
|
|
|
|
-
|
|
|
|
|
-use chrono::Utc;
|
|
|
|
|
-use percent_encoding::{AsciiSet, CONTROLS, utf8_percent_encode};
|
|
|
|
|
-use reqwest::Client;
|
|
|
|
|
-use reqwest::header::HeaderMap;
|
|
|
|
|
-use ring::hmac;
|
|
|
|
|
-use rust_decimal::Decimal;
|
|
|
|
|
-use rust_decimal::prelude::FromPrimitive;
|
|
|
|
|
-use rust_decimal_macros::dec;
|
|
|
|
|
-use serde_json::{json, Value};
|
|
|
|
|
-use tracing::{error, info, trace};
|
|
|
|
|
-use crate::http_tool::RestTool;
|
|
|
|
|
-
|
|
|
|
|
-use crate::response_base::ResponseData;
|
|
|
|
|
-
|
|
|
|
|
-// 定义用于 URL 编码的字符集
|
|
|
|
|
-pub const FRAGMENT: &AsciiSet = &CONTROLS
|
|
|
|
|
- .add(b' ')
|
|
|
|
|
- .add(b':')
|
|
|
|
|
- .add(b'=')
|
|
|
|
|
- .add(b'+')
|
|
|
|
|
- .add(b'/').add(b'?').add(b'#')
|
|
|
|
|
- .add(b'[').add(b']').add(b'@').add(b'!').add(b'$').add(b'&')
|
|
|
|
|
- .add(b'\'').add(b'(').add(b')').add(b'*').add(b',')
|
|
|
|
|
- .add(b';').add(b'"').add(b'%')
|
|
|
|
|
-;
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-#[derive(Clone)]
|
|
|
|
|
-pub struct BitMartSwapRest {
|
|
|
|
|
- label: String,
|
|
|
|
|
- base_url: String,
|
|
|
|
|
- client: reqwest::Client,
|
|
|
|
|
- /*******参数*/
|
|
|
|
|
- //登录所需参数
|
|
|
|
|
- login_param: BTreeMap<String, String>,
|
|
|
|
|
- delays: Vec<i64>,
|
|
|
|
|
- max_delay: i64,
|
|
|
|
|
- avg_delay: Decimal,
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-impl BitMartSwapRest {
|
|
|
|
|
- /*******************************************************************************************************/
|
|
|
|
|
- /*****************************************获取一个对象****************************************************/
|
|
|
|
|
- /*******************************************************************************************************/
|
|
|
|
|
- pub fn new(is_colo: bool, login_param: BTreeMap<String, String>) -> BitMartSwapRest
|
|
|
|
|
- {
|
|
|
|
|
- return BitMartSwapRest::new_label("default-BitMartSwapRest".to_string(), is_colo, login_param);
|
|
|
|
|
- }
|
|
|
|
|
- pub fn new_label(label: String, is_colo: bool, login_param: BTreeMap<String, String>) -> BitMartSwapRest
|
|
|
|
|
- {
|
|
|
|
|
- let base_url: String = String::from("https://api-cloud.bitmart.com");
|
|
|
|
|
- info!("走普通通道:{}",base_url);
|
|
|
|
|
-
|
|
|
|
|
- if is_colo {} else {}
|
|
|
|
|
- /*****返回结构体*******/
|
|
|
|
|
- BitMartSwapRest {
|
|
|
|
|
- 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_depth(&mut self, symbol: String) -> ResponseData {
|
|
|
|
|
- let params = json!({
|
|
|
|
|
- "symbol":symbol
|
|
|
|
|
- });
|
|
|
|
|
-
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/public/depth"),
|
|
|
|
|
- false,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //获取所有合约详情
|
|
|
|
|
- pub async fn get_market(&mut self, symbol: String) -> ResponseData {
|
|
|
|
|
- let params = json!({
|
|
|
|
|
- "symbol":symbol
|
|
|
|
|
- });
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/public/details"),
|
|
|
|
|
- false,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
- //查询合约账户 (查询合约资产明细 )
|
|
|
|
|
- pub async fn get_account(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/assets-detail"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //用户仓位列表(查询仓位详情 (KEYED))
|
|
|
|
|
- pub async fn get_user_position(&mut self, symbol: String) -> ResponseData {
|
|
|
|
|
- let params = json!({
|
|
|
|
|
- "symbol":symbol
|
|
|
|
|
- });
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/position"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //查询合约订单列表 (查询合约历史订单 (KEYED))
|
|
|
|
|
- pub async fn get_orders(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/order-history"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
- //查询单个订单详情 (【全仓】获取订单明细信息)
|
|
|
|
|
- pub async fn get_order_details(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("GET".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/order"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //合约交易下单 (合约下单 (SIGNED))
|
|
|
|
|
- pub async fn swap_order(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("POST".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/submit-order"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- // 全部撤单(批量撤销合约订单 (SIGNED))
|
|
|
|
|
- pub async fn cancel_price_order(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- self.request("POST".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/cancel-orders"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //撤销单个订单 (取消单个合约订单 (SIGNED))
|
|
|
|
|
- pub async fn cancel_order(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("POST".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/cancel-order"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).await;
|
|
|
|
|
- data
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- // //设置持仓模式 (【全仓】切换持仓模式)
|
|
|
|
|
- // pub async fn setting_dual_mode(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- // let data = self.request("POST".to_string(),
|
|
|
|
|
- // "/linear-swap-api/v1".to_string(),
|
|
|
|
|
- // format!("/swap_cross_switch_position_mode"),
|
|
|
|
|
- // true,
|
|
|
|
|
- // params,
|
|
|
|
|
- // ).await;
|
|
|
|
|
- // data
|
|
|
|
|
- // }
|
|
|
|
|
-
|
|
|
|
|
- //设置杠杆(合约杠杆调整 (SIGNED))
|
|
|
|
|
- pub async fn setting_dual_leverage(&mut self, params: Value) -> ResponseData {
|
|
|
|
|
- let data = self.request("POST".to_string(),
|
|
|
|
|
- "/contract".to_string(),
|
|
|
|
|
- format!("/private/submit-leverage"),
|
|
|
|
|
- true,
|
|
|
|
|
- params,
|
|
|
|
|
- ).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: Value) -> ResponseData
|
|
|
|
|
- {
|
|
|
|
|
- // trace!("login_param:{:?}", self.login_param);
|
|
|
|
|
- //解析账号信息
|
|
|
|
|
- let mut access_key = "".to_string();
|
|
|
|
|
- let mut secret_key = "".to_string();
|
|
|
|
|
- let mut api_memo = "".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();
|
|
|
|
|
- }
|
|
|
|
|
- if self.login_param.contains_key("secret_key") {
|
|
|
|
|
- api_memo = self.login_param.get("api_memo").unwrap().to_string();
|
|
|
|
|
- }
|
|
|
|
|
- let mut is_login_param = true;
|
|
|
|
|
- if access_key == "" || secret_key == "" {
|
|
|
|
|
- is_login_param = false
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- //请求头配置-如果需要登录则存在额外配置
|
|
|
|
|
- let mut body = "".to_string();
|
|
|
|
|
- let mut headers = HeaderMap::new();
|
|
|
|
|
- headers.insert("Content-Type", "application/json".parse().unwrap());
|
|
|
|
|
-
|
|
|
|
|
- // let timestamp = Utc::now().timestamp_millis().to_string();
|
|
|
|
|
- let timestamp = "1589793796145".to_string();
|
|
|
|
|
-
|
|
|
|
|
- // let mut params_str = params.to_string();
|
|
|
|
|
- let mut params_str = json!({
|
|
|
|
|
- "count":"100","price":"8600","symbol":"BTC_USDT"
|
|
|
|
|
- }).to_string();
|
|
|
|
|
- if requesst_type == "GET" {
|
|
|
|
|
- body = "{}".to_string();
|
|
|
|
|
- } else {
|
|
|
|
|
- body = params_str.clone();
|
|
|
|
|
- }
|
|
|
|
|
- //是否需要登录-- 组装sing
|
|
|
|
|
- let mut sing = "".to_string();
|
|
|
|
|
- if is_login {
|
|
|
|
|
- if !is_login_param {
|
|
|
|
|
- let e = ResponseData::error(self.label.clone(), "登录参数错误!".to_string());
|
|
|
|
|
- return e;
|
|
|
|
|
- } else {//需要登录-且登录参数齐全
|
|
|
|
|
- //组装sing
|
|
|
|
|
- sing = Self::sign(secret_key.clone(),
|
|
|
|
|
- api_memo.clone(),
|
|
|
|
|
- params_str.clone(),
|
|
|
|
|
- timestamp.clone(),
|
|
|
|
|
- );
|
|
|
|
|
- headers.extend(Self::headers(access_key.clone(), timestamp.clone(), sing.clone()));
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
- // trace!("headers:{:?}", headers);
|
|
|
|
|
- let base_url = format!("{}{}", prefix_url.clone(), request_url.clone());
|
|
|
|
|
- let start_time = chrono::Utc::now().timestamp_millis();
|
|
|
|
|
- let response = self.http_tool(
|
|
|
|
|
- base_url.clone(),
|
|
|
|
|
- requesst_type.to_string(),
|
|
|
|
|
- params_str.clone(),
|
|
|
|
|
- body.clone(),
|
|
|
|
|
- headers,
|
|
|
|
|
- ).await;
|
|
|
|
|
-
|
|
|
|
|
- let time_array = chrono::Utc::now().timestamp_millis() - start_time;
|
|
|
|
|
- self.delays.push(time_array);
|
|
|
|
|
- self.get_delay_info();
|
|
|
|
|
-
|
|
|
|
|
- response
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- pub fn headers(access_key: String, timestamp: String, sign: String) -> HeaderMap {
|
|
|
|
|
- let mut headers = HeaderMap::new();
|
|
|
|
|
- headers.insert("X-BM-KEY", access_key.clone().parse().unwrap());
|
|
|
|
|
- headers.insert("X-BM-SIGN", sign.clone().parse().unwrap());
|
|
|
|
|
- headers.insert("X-BM-TIMESTAMP", timestamp.clone().parse().unwrap());
|
|
|
|
|
- headers
|
|
|
|
|
- }
|
|
|
|
|
- pub fn sign(secret_key: String, api_memo: String, param_str: String, timestamp: String) -> String
|
|
|
|
|
- {
|
|
|
|
|
-
|
|
|
|
|
- // X-BM-SIGN= hmac_sha256(Your_api_secret_key, X-BM-TIMESTAMP + '#' +Your_api_memo + '#' + '{"symbol":"BTC_USDT","price":"8600","count":"100"}')
|
|
|
|
|
- let message = format!("{}#{}#{}", timestamp, api_memo, param_str);
|
|
|
|
|
- trace!("1组装数据:\n{}", message);
|
|
|
|
|
- let signed_key = hmac::Key::new(hmac::HMAC_SHA256, secret_key.as_ref());
|
|
|
|
|
- let sign = hex::encode(hmac::sign(&signed_key, message.as_bytes()).as_ref());
|
|
|
|
|
- sign
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
- async fn http_tool(&mut self, request_path: String,
|
|
|
|
|
- request_type: String,
|
|
|
|
|
- params: String,
|
|
|
|
|
- body: String, headers: HeaderMap) -> ResponseData {
|
|
|
|
|
- /****请求接口与 地址*/
|
|
|
|
|
- let url = format!("{}{}", self.base_url.to_string(), request_path);
|
|
|
|
|
- let request_type = request_type.clone().to_uppercase();
|
|
|
|
|
- let params_str = RestTool::parse_params_to_str(params.clone());
|
|
|
|
|
- let addrs_url = if params_str.len() > 0 {
|
|
|
|
|
- format!("{}?{}", url.clone(), params_str)
|
|
|
|
|
- } else {
|
|
|
|
|
- format!("{}", url.clone())
|
|
|
|
|
- };
|
|
|
|
|
-
|
|
|
|
|
- trace!("url-----:???{}",url.clone());
|
|
|
|
|
- trace!("addrs_url-----:???{}",addrs_url.clone());
|
|
|
|
|
- trace!("param-----:???{}",params.clone());
|
|
|
|
|
- trace!("param_str-----:???{}",params_str.clone());
|
|
|
|
|
- trace!("body-----:???{}",body.clone());
|
|
|
|
|
- trace!("headers-----:???{:?}",headers.clone());
|
|
|
|
|
-
|
|
|
|
|
- let request_builder = match request_type.as_str() {
|
|
|
|
|
- "GET" => self.client.get(addrs_url.clone()).headers(headers),
|
|
|
|
|
- "POST" => self.client.post(url.clone()).body(body).headers(headers),
|
|
|
|
|
- // "DELETE" => self.client.delete(addrs_url.clone()).headers(headers),
|
|
|
|
|
- // "PUT" => self.client.put(url.clone()).json(¶ms),
|
|
|
|
|
- _ => {
|
|
|
|
|
- panic!("{}", format!("错误的请求类型:{}", request_type.clone()))
|
|
|
|
|
- }
|
|
|
|
|
- };
|
|
|
|
|
-
|
|
|
|
|
- // 读取响应的内容
|
|
|
|
|
- let response = request_builder.send().await.unwrap();
|
|
|
|
|
- let is_success = response.status().is_success(); // 先检查状态码
|
|
|
|
|
- let text = response.text().await.unwrap();
|
|
|
|
|
- trace!("text:???{:?}",text);
|
|
|
|
|
- return if is_success {
|
|
|
|
|
- self.on_success_data(&text)
|
|
|
|
|
- } else {
|
|
|
|
|
- self.on_error_data(&text, &addrs_url, ¶ms)
|
|
|
|
|
- };
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- pub fn on_success_data(&mut self, text: &String) -> ResponseData {
|
|
|
|
|
- let json_value = serde_json::from_str::<Value>(&text).unwrap();
|
|
|
|
|
- // return ResponseData::new(self.label.clone(), 200, "success".to_string(), json_value);
|
|
|
|
|
-
|
|
|
|
|
- let code = json_value["code"].as_i64();
|
|
|
|
|
- let message = json_value["message"].as_str();
|
|
|
|
|
- let trace = json_value["trace"].as_str();
|
|
|
|
|
- match code {
|
|
|
|
|
- Some(c) => {
|
|
|
|
|
- if c == 1000 {
|
|
|
|
|
- let data = json_value.get("data").unwrap();
|
|
|
|
|
- ResponseData::new(self.label.clone(), 200, "success".to_string(), data.clone())
|
|
|
|
|
- } else {
|
|
|
|
|
- ResponseData::new(self.label.clone(), 400, "message".to_string(), json_value)
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- None => {
|
|
|
|
|
- ResponseData::new(self.label.clone(), 400, "message".to_string(), json_value)
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- pub fn on_error_data(&mut self, text: &String, base_url: &String, params: &String) -> ResponseData {
|
|
|
|
|
- let json_value = serde_json::from_str::<Value>(&text);
|
|
|
|
|
- match json_value {
|
|
|
|
|
- Ok(d) => {
|
|
|
|
|
- ResponseData::new(self.label.clone(), 400, format!("错误:{}", d["message"].as_str().unwrap()), d)
|
|
|
|
|
- }
|
|
|
|
|
- Err(_) => {
|
|
|
|
|
- ResponseData::new(self.label.clone(), 400, format!("未知错误"), text.parse().unwrap())
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-// 合并两个 JSON 对象的函数
|
|
|
|
|
-fn merge_json(a: &mut Value, b: &Value) {
|
|
|
|
|
- match (a, b) {
|
|
|
|
|
- (Value::Object(ref mut a_map), Value::Object(b_map)) => {
|
|
|
|
|
- for (k, v) in b_map {
|
|
|
|
|
- merge_json(a_map.entry(k.clone()).or_insert(Value::Null), v);
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- (a, b) => {
|
|
|
|
|
- *a = b.clone();
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-
|
|
|
|
|
-// 函数:将 JSON 对象转换为排序后的查询字符串
|
|
|
|
|
-fn json_to_query_string(value: Value) -> String {
|
|
|
|
|
- let mut params = BTreeMap::new();
|
|
|
|
|
-
|
|
|
|
|
- if let Value::Object(obj) = value {
|
|
|
|
|
- for (k, v) in obj {
|
|
|
|
|
- // 确保只处理字符串值
|
|
|
|
|
- if let Value::String(v_str) = v {
|
|
|
|
|
- params.insert(k, v_str);
|
|
|
|
|
- } else {
|
|
|
|
|
- // 对于非字符串值,我们可以转换为字符串或执行其他处理
|
|
|
|
|
- params.insert(k, v.to_string());
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- // 拼接键值对为查询字符串
|
|
|
|
|
- params.iter()
|
|
|
|
|
- .map(|(k, v)| format!("{}={}", utf8_percent_encode(k, FRAGMENT), utf8_percent_encode(v, FRAGMENT)))
|
|
|
|
|
- .collect::<Vec<String>>()
|
|
|
|
|
- .join("&")
|
|
|
|
|
-}
|
|
|