uv/crates/puffin-client/src/api.rs
2023-10-05 12:45:38 -04:00

165 lines
4.4 KiB
Rust

use std::fmt::Debug;
use reqwest::StatusCode;
use serde::{Deserialize, Serialize};
use tracing::trace;
use url::Url;
use puffin_requirements::metadata::Metadata21;
use puffin_requirements::package_name::PackageName;
use crate::client::PypiClient;
use crate::error::PypiClientError;
impl PypiClient {
pub async fn simple(
&self,
package_name: impl AsRef<str>,
) -> Result<SimpleJson, PypiClientError> {
let start = std::time::Instant::now();
// Format the URL for PyPI.
let mut url = self.registry.join("simple")?;
url.path_segments_mut()
.unwrap()
.push(PackageName::normalize(&package_name).as_ref());
url.path_segments_mut().unwrap().push("");
url.set_query(Some("format=application/vnd.pypi.simple.v1+json"));
trace!(
"fetching metadata for {} from {}",
package_name.as_ref(),
url
);
// Fetch from the registry.
let text = self.simple_impl(&package_name, &url).await?;
let payload = serde_json::from_str(&text)
.map_err(move |e| PypiClientError::from_json_err(e, String::new()));
trace!("fetched metadata for {} in {:?}", url, start.elapsed());
payload
}
async fn simple_impl(
&self,
package_name: impl AsRef<str>,
url: &Url,
) -> Result<String, PypiClientError> {
Ok(self
.client
.get(url.clone())
.header("Accept-Encoding", "gzip")
.send()
.await?
.error_for_status()
.map_err(|err| {
if err.status() == Some(StatusCode::NOT_FOUND) {
PypiClientError::PackageNotFound(
(*self.registry).clone(),
package_name.as_ref().to_string(),
)
} else {
PypiClientError::RequestError(err)
}
})?
.text()
.await?)
}
pub async fn file(&self, file: File) -> Result<Metadata21, PypiClientError> {
let start = std::time::Instant::now();
// Send to the proxy.
let url = self.proxy.join(
file.url
.strip_prefix("https://files.pythonhosted.org/")
.unwrap(),
)?;
trace!("fetching file {} from {}", file.filename, url);
// Fetch from the registry.
let text = self.file_impl(&file.filename, &url).await?;
let payload = Metadata21::parse(text.as_bytes()).map_err(std::convert::Into::into);
trace!("fetched file {} in {:?}", url, start.elapsed());
payload
}
async fn file_impl(
&self,
filename: impl AsRef<str>,
url: &Url,
) -> Result<String, PypiClientError> {
Ok(self
.client
.get(url.clone())
.send()
.await?
.error_for_status()
.map_err(|err| {
if err.status() == Some(StatusCode::NOT_FOUND) {
PypiClientError::FileNotFound(
(*self.registry).clone(),
filename.as_ref().to_string(),
)
} else {
PypiClientError::RequestError(err)
}
})?
.text()
.await?)
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SimpleJson {
pub files: Vec<File>,
pub meta: Meta,
pub name: String,
pub versions: Vec<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "kebab-case")]
pub struct File {
pub core_metadata: Metadata,
pub data_dist_info_metadata: Metadata,
pub filename: String,
pub hashes: Hashes,
pub requires_python: Option<String>,
pub size: i64,
pub upload_time: String,
pub url: String,
pub yanked: Yanked,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Metadata {
Bool(bool),
Hashes(Hashes),
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Yanked {
Bool(bool),
Reason(String),
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct Hashes {
pub sha256: String,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "kebab-case")]
pub struct Meta {
#[serde(rename = "_last-serial")]
pub last_serial: i64,
pub api_version: String,
}