Запрос Reqwest не получает заголовки по умолчанию для моего reqwest :: клиента

#api #rest #rust #reqwest #todoist

#API #rest #Ржавчина #reqwest #todoist

Вопрос:

Я пытаюсь написать библиотеку, которая взаимодействует с REST API Todoist. Идея заключается в том, что библиотека предоставляет структуру TodoistAPI, которая содержит reqwest::Client и base_url . Существует new() функция, которая возвращает созданную структуру TodoistAPI с клиентом, у которого есть токен-носитель (предоставленный программой, использующей мою библиотеку) в заголовках по умолчанию.

Однако я сталкиваюсь с проблемой, когда, когда приходит время фактически использовать клиент для выполнения запроса API, заголовки по умолчанию вообще не устанавливаются.

TodoistAPI Структура, new метод и get_projects метод.

 #[derive(Debug)]
pub struct TodoistAPI{
    base_url: Url,
    client: reqwest::Client
}

impl TodoistAPI {
    #[allow(dead_code)]
    pub fn new(token: amp;str) -> Result<TodoistAPI, TodoistAPIError> {
        let mut headers = header::HeaderMap::new();
        let header_token_value = header::HeaderValue::from_str(token).map_err(TodoistAPIError::InvalidHeaderValue)?;
        headers.insert(header::HeaderName::from_bytes(b"Bearer").map_err(TodoistAPIError::InvalidHeaderName)?, header_token_value);
        let client = reqwest::Client::builder()
            .default_headers(headers)
            .build().map_err(TodoistAPIError::Error)?;
        println!("{:#?}", client);
        let base_url = Url::parse(BASE_URL).map_err(TodoistAPIError::UrlParseError)?;
        return Ok(TodoistAPI{ base_url, client })
    }

    #[allow(dead_code)]
    pub async fn get_projects(amp;self) -> Result<Vec<Project>, TodoistAPIError> {
        let url = self.base_url.join("projects").map_err(TodoistAPIError::UrlParseError)?;
        let request_builder = self.client.request(reqwest::Method::GET, url);
        println!("{:#?}", request_builder);
        let request = request_builder.build().map_err(TodoistAPIError::Error)?;
        println!("{:#?}", request);
        let response = self.client.execute(request).await.map_err(TodoistAPIError::Error)?;
        println!("Status: {}", response.status());
        println!("STatus: {:#?}", response.text().await.map_err(TodoistAPIError::Error)?);
        let url = self.base_url.join("projects").map_err(TodoistAPIError::UrlParseError)?;
        let projects = self.client.get(url)
            .send()
            .await.map_err(TodoistAPIError::Error)?
            .json::<Vec<Project>>()
            .await.map_err(TodoistAPIError::Error)?;
        return Ok(projects);
    }
}
 

Небольшая программа CLI, которая получает токен из переменной среды и вызывает get_projects метод.

 use structopt::StructOpt;
use oxidoist_api::TodoistAPI;
use oxidoist_api::Project;
use oxidoist_api::TodoistAPIError;

use std::env;

#[derive(StructOpt, Debug)]
struct Cli {
    verb: String, //get, add, complete, etc.
    datatype: String, //project, task, section, etc.
}

#[tokio::main]
async fn main() -> Result<(), TodoistAPIError> {
    let args = Cli::from_args();
    let token = env::var("TODOIST_API_KEY").unwrap();
    let todoist_api_object = TodoistAPI::new(token.as_str()).unwrap();
    if args.verb == "get" {
        if args.datatype == "projects" {
            let projects: Vec<Project> = todoist_api_object.get_projects().await?;
            println!("{:?}", projects);
        }
    }
    
    Ok(())
}
 

В println! результате выполнения инструкций получается следующий вывод (с некоторой явно отредактированной личной информацией).

 Client {
    accepts: Accepts,
    proxies: [
        Proxy(
            System(
                {},
            ),
            None,
        ),
    ],
    referer: true,
    default_headers: {
        "accept": "*/*",
        "bearer": "REDACTED",
    },
}
RequestBuilder {
    method: GET,
    url: Url {
        scheme: "https",
        host: Some(
            Domain(
                "api.todoist.com",
            ),
        ),
        port: None,
        path: "/rest/v1/projects",
        query: None,
        fragment: None,
    },
    headers: {},
}
Request {
    method: GET,
    url: Url {
        scheme: "https",
        host: Some(
            Domain(
                "api.todoist.com",
            ),
        ),
        port: None,
        path: "/rest/v1/projects",
        query: None,
        fragment: None,
    },
    headers: {},
}
Status: 400 Bad Request
STatus: "Empty tokenn"
Error: Error(reqwest::Error { kind: Decode, source: Error("expected value", line: 1, column: 1) })
 

Я действительно в тупике. Все, что я читаю, говорит о том, что я все делаю правильно, но заголовки по умолчанию определенно НЕ добавляются к запросам, создаваемым клиентом.

Ответ №1:

Так что, оказывается, я неправильно понял пару вещей. Во-первых, заголовки. Оказывается, я просто вводил их неправильно. Имя заголовка должно быть b"Authorization" , а значение должно быть "Bearer <token>" . Я изменил код следующим образом:

 let mut token: String = "Bearer ".to_string();
token.push_str(amp;self.token);
let header_token_value = header::HeaderValue::from_str(amp;token).map_err(TodoistAPIError::InvalidHeaderValue)?;
 

Во-вторых, даже после правильной работы headers: {} запись в Request RequestBuilder распечатках и по-прежнему оставалась пустой, поэтому я предполагаю, что reqwest объединяет их с Client ‘s default_headers , когда он фактически отправляет запрос.

Могут быть лучшие способы обработки токена на предъявителя, это выглядит немного неуклюже. Если кто-нибудь найдет это, знает о каких-либо, не стесняйтесь оставлять лучший ответ.