diff --git a/crates/uv-build-backend/src/fs_write_dispatcher.rs b/crates/uv-build-backend/src/fs_write_dispatcher.rs new file mode 100644 index 0000000000000..bd5ddb0293ec6 --- /dev/null +++ b/crates/uv-build-backend/src/fs_write_dispatcher.rs @@ -0,0 +1,263 @@ +//! Dispatcher between writing to a directory, writing to a zip, writing to a `.tar.gz` and +//! listing files. + +use crate::wheel::{write_hashed, RecordEntry}; +use crate::Error; +use flate2::write::GzEncoder; +use flate2::Compression; +use fs_err::File; +use sha2::{Digest, Sha256}; +use std::io::{BufReader, Cursor, Write}; +use std::path::{Path, PathBuf}; +use std::{io, mem}; +use tar::{EntryType, Header}; +use tracing::trace; +use uv_fs::Simplified; +use zip::{CompressionMethod, ZipWriter}; + +/// Dispatcher between writing to a directory, writing to a zip, writing to a `.tar.gz` and +/// listing files. +/// +/// All paths are string types instead of path types since wheels are portable between platforms. +/// +/// Contract: You must call close before dropping to obtain a valid output (dropping is fine in the +/// error case). +pub(crate) trait FsWriteDispatcher { + /// Add a file with the given content. + fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error>; + + /// Add a local file. + fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error>; + + /// Create a directory. + fn write_directory(&mut self, directory: &str) -> Result<(), Error>; + + /// Write the `RECORD` file and if applicable, the central directory. + fn close(self, dist_info_dir: &str) -> Result<(), Error>; +} + +/// Zip archive (wheel) writer. +pub(crate) struct ZipDirectoryWriter { + writer: ZipWriter, + compression: CompressionMethod, + /// The entries in the `RECORD` file. + record: Vec, +} + +impl ZipDirectoryWriter { + /// A wheel writer with deflate compression. + pub(crate) fn new_wheel(file: File) -> Self { + Self { + writer: ZipWriter::new(file), + compression: CompressionMethod::Deflated, + record: Vec::new(), + } + } + + /// A wheel writer with no (stored) compression. + /// + /// Since editables are temporary, we save time be skipping compression and decompression. + #[expect(dead_code)] + fn new_editable(file: File) -> Self { + Self { + writer: ZipWriter::new(file), + compression: CompressionMethod::Stored, + record: Vec::new(), + } + } + + /// Add a file with the given name and return a writer for it. + fn new_writer<'slf>(&'slf mut self, path: &str) -> Result, Error> { + // TODO(konsti): We need to preserve permissions, at least the executable bit. + self.writer.start_file( + path, + zip::write::FileOptions::default().compression_method(self.compression), + )?; + Ok(Box::new(&mut self.writer)) + } +} + +impl FsWriteDispatcher for ZipDirectoryWriter { + fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { + trace!("Adding {}", path); + let options = zip::write::FileOptions::default().compression_method(self.compression); + self.writer.start_file(path, options)?; + self.writer.write_all(bytes)?; + + let hash = format!("{:x}", Sha256::new().chain_update(bytes).finalize()); + self.record.push(RecordEntry { + path: path.to_string(), + hash, + size: bytes.len(), + }); + + Ok(()) + } + + fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { + trace!("Adding {} from {}", path, file.user_display()); + let mut reader = BufReader::new(File::open(file)?); + let mut writer = self.new_writer(path)?; + let record = write_hashed(path, &mut reader, &mut writer)?; + drop(writer); + self.record.push(record); + Ok(()) + } + + fn write_directory(&mut self, directory: &str) -> Result<(), Error> { + trace!("Adding directory {}", directory); + let options = zip::write::FileOptions::default().compression_method(self.compression); + Ok(self.writer.add_directory(directory, options)?) + } + + /// Write the `RECORD` file and the central directory. + fn close(mut self, dist_info_dir: &str) -> Result<(), Error> { + let record_path = format!("{dist_info_dir}/RECORD"); + trace!("Adding {record_path}"); + let record = mem::take(&mut self.record); + crate::wheel::write_record(&mut self.new_writer(&record_path)?, dist_info_dir, record)?; + + trace!("Adding central directory"); + self.writer.finish()?; + Ok(()) + } +} + +pub(crate) struct FilesystemWriter { + /// The virtualenv or metadata directory that add file paths are relative to. + root: PathBuf, + /// The entries in the `RECORD` file. + record: Vec, +} + +impl FilesystemWriter { + pub(crate) fn new(root: &Path) -> Self { + Self { + root: root.to_owned(), + record: Vec::new(), + } + } + + /// Add a file with the given name and return a writer for it. + fn new_writer<'slf>(&'slf mut self, path: &str) -> Result, Error> { + trace!("Adding {}", path); + Ok(Box::new(File::create(self.root.join(path))?)) + } +} + +/// File system writer. +impl FsWriteDispatcher for FilesystemWriter { + fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { + trace!("Adding {}", path); + let hash = format!("{:x}", Sha256::new().chain_update(bytes).finalize()); + self.record.push(RecordEntry { + path: path.to_string(), + hash, + size: bytes.len(), + }); + + Ok(fs_err::write(self.root.join(path), bytes)?) + } + fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { + trace!("Adding {} from {}", path, file.user_display()); + let mut reader = BufReader::new(File::open(file)?); + let mut writer = self.new_writer(path)?; + let record = write_hashed(path, &mut reader, &mut writer)?; + drop(writer); + self.record.push(record); + Ok(()) + } + + fn write_directory(&mut self, directory: &str) -> Result<(), Error> { + trace!("Adding directory {}", directory); + Ok(fs_err::create_dir(self.root.join(directory))?) + } + + /// Write the `RECORD` file. + fn close(mut self, dist_info_dir: &str) -> Result<(), Error> { + let record = mem::take(&mut self.record); + crate::wheel::write_record( + &mut self.new_writer(&format!("{dist_info_dir}/RECORD"))?, + dist_info_dir, + record, + )?; + + Ok(()) + } +} + +pub(crate) struct TarGzWriter { + path: PathBuf, + tar: tar::Builder>, +} + +impl TarGzWriter { + pub(crate) fn new(path: impl Into) -> Result { + let path = path.into(); + let file = File::create(&path)?; + let enc = GzEncoder::new(file, Compression::default()); + let tar = tar::Builder::new(enc); + Ok(Self { path, tar }) + } +} + +impl FsWriteDispatcher for TarGzWriter { + fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { + let mut header = Header::new_gnu(); + header.set_size(bytes.len() as u64); + // Reasonable default to avoid 0o000 permissions, the user's umask will be applied on + // unpacking. + header.set_mode(0o644); + header.set_cksum(); + self.tar + .append_data(&mut header, path, Cursor::new(bytes)) + .map_err(|err| Error::TarWrite(self.path.clone(), err))?; + Ok(()) + } + + fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { + let metadata = fs_err::metadata(file)?; + let mut header = Header::new_gnu(); + #[cfg(unix)] + { + // Preserve for example an executable bit. + header.set_mode(std::os::unix::fs::MetadataExt::mode(&metadata)); + } + #[cfg(not(unix))] + { + // Reasonable default to avoid 0o000 permissions, the user's umask will be applied on + // unpacking. + header.set_mode(0o644); + } + header.set_size(metadata.len()); + header.set_cksum(); + let reader = BufReader::new(File::open(file)?); + self.tar + .append_data(&mut header, path, reader) + .map_err(|err| Error::TarWrite(self.path.clone(), err))?; + Ok(()) + } + + fn write_directory(&mut self, directory: &str) -> Result<(), Error> { + let mut header = Header::new_gnu(); + // Directories are always executable, which means they can be listed. + header.set_mode(0o755); + header.set_entry_type(EntryType::Directory); + header + .set_path(directory) + .map_err(|err| Error::TarWrite(self.path.clone(), err))?; + header.set_size(0); + header.set_cksum(); + self.tar + .append(&header, io::empty()) + .map_err(|err| Error::TarWrite(self.path.clone(), err))?; + Ok(()) + } + + fn close(mut self, _dist_info_dir: &str) -> Result<(), Error> { + self.tar + .finish() + .map_err(|err| Error::TarWrite(self.path.clone(), err))?; + Ok(()) + } +} diff --git a/crates/uv-build-backend/src/lib.rs b/crates/uv-build-backend/src/lib.rs index ce0a4cef897ea..2b62bf50efd74 100644 --- a/crates/uv-build-backend/src/lib.rs +++ b/crates/uv-build-backend/src/lib.rs @@ -1,27 +1,20 @@ +mod fs_write_dispatcher; mod metadata; +mod source_dist; +mod wheel; pub use metadata::PyProjectToml; +pub use source_dist::build_source_dist; +pub use wheel::{build_editable, build_wheel, metadata}; -use crate::metadata::{BuildBackendSettings, ValidationError}; -use flate2::write::GzEncoder; -use flate2::Compression; -use fs_err::File; -use globset::{Glob, GlobSet, GlobSetBuilder}; -use itertools::Itertools; -use sha2::{Digest, Sha256}; +use crate::metadata::ValidationError; +use globset::{GlobSet, GlobSetBuilder}; use std::fs::FileType; -use std::io::{BufReader, Cursor, Read, Write}; +use std::io; use std::path::{Path, PathBuf, StripPrefixError}; -use std::{io, mem}; -use tar::{EntryType, Header}; use thiserror::Error; -use tracing::{debug, trace}; -use uv_distribution_filename::{SourceDistExtension, SourceDistFilename, WheelFilename}; use uv_fs::Simplified; -use uv_globfilter::{parse_portable_glob, GlobDirFilter, PortableGlobError}; -use uv_warnings::warn_user_once; -use walkdir::WalkDir; -use zip::{CompressionMethod, ZipWriter}; +use uv_globfilter::{parse_portable_glob, PortableGlobError}; #[derive(Debug, Error)] pub enum Error { @@ -77,803 +70,8 @@ pub enum Error { TarWrite(PathBuf, #[source] io::Error), } -/// Dispatcher between writing to a directory, writing to a zip, writing to a `.tar.gz` and -/// listing files. -/// -/// All paths are string types instead of path types since wheels are portable between platforms. -/// -/// Contract: You must call close before dropping to obtain a valid output (dropping is fine in the -/// error case). -trait DirectoryWriter { - /// Add a file with the given content. - fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error>; - - /// Add a local file. - fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error>; - - /// Create a directory. - fn write_directory(&mut self, directory: &str) -> Result<(), Error>; - - /// Write the `RECORD` file and if applicable, the central directory. - fn close(self, dist_info_dir: &str) -> Result<(), Error>; -} - -/// Zip archive (wheel) writer. -struct ZipDirectoryWriter { - writer: ZipWriter, - compression: CompressionMethod, - /// The entries in the `RECORD` file. - record: Vec, -} - -impl ZipDirectoryWriter { - /// A wheel writer with deflate compression. - fn new_wheel(file: File) -> Self { - Self { - writer: ZipWriter::new(file), - compression: CompressionMethod::Deflated, - record: Vec::new(), - } - } - - /// A wheel writer with no (stored) compression. - /// - /// Since editables are temporary, we save time be skipping compression and decompression. - #[expect(dead_code)] - fn new_editable(file: File) -> Self { - Self { - writer: ZipWriter::new(file), - compression: CompressionMethod::Stored, - record: Vec::new(), - } - } - - /// Add a file with the given name and return a writer for it. - fn new_writer<'slf>(&'slf mut self, path: &str) -> Result, Error> { - // TODO(konsti): We need to preserve permissions, at least the executable bit. - self.writer.start_file( - path, - zip::write::FileOptions::default().compression_method(self.compression), - )?; - Ok(Box::new(&mut self.writer)) - } -} - -impl DirectoryWriter for ZipDirectoryWriter { - fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { - trace!("Adding {}", path); - let options = zip::write::FileOptions::default().compression_method(self.compression); - self.writer.start_file(path, options)?; - self.writer.write_all(bytes)?; - - let hash = format!("{:x}", Sha256::new().chain_update(bytes).finalize()); - self.record.push(RecordEntry { - path: path.to_string(), - hash, - size: bytes.len(), - }); - - Ok(()) - } - - fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { - trace!("Adding {} from {}", path, file.user_display()); - let mut reader = BufReader::new(File::open(file)?); - let mut writer = self.new_writer(path)?; - let record = write_hashed(path, &mut reader, &mut writer)?; - drop(writer); - self.record.push(record); - Ok(()) - } - - fn write_directory(&mut self, directory: &str) -> Result<(), Error> { - trace!("Adding directory {}", directory); - let options = zip::write::FileOptions::default().compression_method(self.compression); - Ok(self.writer.add_directory(directory, options)?) - } - - /// Write the `RECORD` file and the central directory. - fn close(mut self, dist_info_dir: &str) -> Result<(), Error> { - let record_path = format!("{dist_info_dir}/RECORD"); - trace!("Adding {record_path}"); - let record = mem::take(&mut self.record); - write_record(&mut self.new_writer(&record_path)?, dist_info_dir, record)?; - - trace!("Adding central directory"); - self.writer.finish()?; - Ok(()) - } -} - -struct FilesystemWriter { - /// The virtualenv or metadata directory that add file paths are relative to. - root: PathBuf, - /// The entries in the `RECORD` file. - record: Vec, -} - -impl FilesystemWriter { - fn new(root: &Path) -> Self { - Self { - root: root.to_owned(), - record: Vec::new(), - } - } - - /// Add a file with the given name and return a writer for it. - fn new_writer<'slf>(&'slf mut self, path: &str) -> Result, Error> { - trace!("Adding {}", path); - Ok(Box::new(File::create(self.root.join(path))?)) - } -} - -/// File system writer. -impl DirectoryWriter for FilesystemWriter { - fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { - trace!("Adding {}", path); - let hash = format!("{:x}", Sha256::new().chain_update(bytes).finalize()); - self.record.push(RecordEntry { - path: path.to_string(), - hash, - size: bytes.len(), - }); - - Ok(fs_err::write(self.root.join(path), bytes)?) - } - fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { - trace!("Adding {} from {}", path, file.user_display()); - let mut reader = BufReader::new(File::open(file)?); - let mut writer = self.new_writer(path)?; - let record = write_hashed(path, &mut reader, &mut writer)?; - drop(writer); - self.record.push(record); - Ok(()) - } - - fn write_directory(&mut self, directory: &str) -> Result<(), Error> { - trace!("Adding directory {}", directory); - Ok(fs_err::create_dir(self.root.join(directory))?) - } - - /// Write the `RECORD` file. - fn close(mut self, dist_info_dir: &str) -> Result<(), Error> { - let record = mem::take(&mut self.record); - write_record( - &mut self.new_writer(&format!("{dist_info_dir}/RECORD"))?, - dist_info_dir, - record, - )?; - - Ok(()) - } -} - -struct TarGzWriter { - path: PathBuf, - tar: tar::Builder>, -} - -impl TarGzWriter { - fn new(path: impl Into) -> Result { - let path = path.into(); - let file = File::create(&path)?; - let enc = GzEncoder::new(file, Compression::default()); - let tar = tar::Builder::new(enc); - Ok(Self { path, tar }) - } -} - -impl DirectoryWriter for TarGzWriter { - fn write_bytes(&mut self, path: &str, bytes: &[u8]) -> Result<(), Error> { - let mut header = Header::new_gnu(); - header.set_size(bytes.len() as u64); - // Reasonable default to avoid 0o000 permissions, the user's umask will be applied on - // unpacking. - header.set_mode(0o644); - header.set_cksum(); - self.tar - .append_data(&mut header, path, Cursor::new(bytes)) - .map_err(|err| Error::TarWrite(self.path.clone(), err))?; - Ok(()) - } - - fn write_file(&mut self, path: &str, file: &Path) -> Result<(), Error> { - let metadata = fs_err::metadata(file)?; - let mut header = Header::new_gnu(); - #[cfg(unix)] - { - // Preserve for example an executable bit. - header.set_mode(std::os::unix::fs::MetadataExt::mode(&metadata)); - } - #[cfg(not(unix))] - { - // Reasonable default to avoid 0o000 permissions, the user's umask will be applied on - // unpacking. - header.set_mode(0o644); - } - header.set_size(metadata.len()); - header.set_cksum(); - let reader = BufReader::new(File::open(file)?); - self.tar - .append_data(&mut header, path, reader) - .map_err(|err| Error::TarWrite(self.path.clone(), err))?; - Ok(()) - } - - fn write_directory(&mut self, directory: &str) -> Result<(), Error> { - let mut header = Header::new_gnu(); - // Directories are always executable, which means they can be listed. - header.set_mode(0o755); - header.set_entry_type(EntryType::Directory); - header - .set_path(directory) - .map_err(|err| Error::TarWrite(self.path.clone(), err))?; - header.set_size(0); - header.set_cksum(); - self.tar - .append(&header, io::empty()) - .map_err(|err| Error::TarWrite(self.path.clone(), err))?; - Ok(()) - } - - fn close(mut self, _dist_info_dir: &str) -> Result<(), Error> { - self.tar - .finish() - .map_err(|err| Error::TarWrite(self.path.clone(), err))?; - Ok(()) - } -} - -/// An entry in the `RECORD` file. -/// -/// -struct RecordEntry { - /// The path to the file relative to the package root. - /// - /// While the spec would allow backslashes, we always use portable paths with forward slashes. - path: String, - /// The SHA256 of the files. - hash: String, - /// The size of the file in bytes. - size: usize, -} - -/// Read the input file and write it both to the hasher and the target file. -/// -/// We're implementing this tee-ing manually since there is no sync `InspectReader` or std tee -/// function. -fn write_hashed( - path: &str, - reader: &mut dyn Read, - writer: &mut dyn Write, -) -> Result { - let mut hasher = Sha256::new(); - let mut size = 0; - // 8KB is the default defined in `std::sys_common::io`. - let mut buffer = vec![0; 8 * 1024]; - loop { - let read = match reader.read(&mut buffer) { - Ok(read) => read, - Err(err) if err.kind() == io::ErrorKind::Interrupted => continue, - Err(err) => return Err(err), - }; - if read == 0 { - // End of file - break; - } - hasher.update(&buffer[..read]); - writer.write_all(&buffer[..read])?; - size += read; - } - Ok(RecordEntry { - path: path.to_string(), - hash: format!("{:x}", hasher.finalize()), - size, - }) -} - -/// Build a wheel from the source tree and place it in the output directory. -pub fn build_wheel( - source_tree: &Path, - wheel_dir: &Path, - metadata_directory: Option<&Path>, - uv_version: &str, -) -> Result { - let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; - let pyproject_toml = PyProjectToml::parse(&contents)?; - for warning in pyproject_toml.check_build_system(uv_version) { - warn_user_once!("{warning}"); - } - let settings = pyproject_toml - .settings() - .cloned() - .unwrap_or_else(BuildBackendSettings::default); - - check_metadata_directory(source_tree, metadata_directory, &pyproject_toml)?; - - let filename = WheelFilename { - name: pyproject_toml.name().clone(), - version: pyproject_toml.version().clone(), - build_tag: None, - python_tag: vec!["py3".to_string()], - abi_tag: vec!["none".to_string()], - platform_tag: vec!["any".to_string()], - }; - - let wheel_path = wheel_dir.join(filename.to_string()); - debug!("Writing wheel at {}", wheel_path.user_display()); - let mut wheel_writer = ZipDirectoryWriter::new_wheel(File::create(&wheel_path)?); - - // Wheel excludes - let mut excludes: Vec = settings.wheel_exclude; - // The wheel must not include any files excluded by the source distribution (at least until we - // have files generated in the source dist -> wheel build step). - for exclude in settings.source_exclude { - // Avoid duplicate entries. - if !excludes.contains(&exclude) { - excludes.push(exclude); - } - } - debug!("Wheel excludes: {:?}", excludes); - let exclude_matcher = build_exclude_matcher(excludes)?; - - debug!("Adding content files to {}", wheel_path.user_display()); - if settings.module_root.is_absolute() { - return Err(Error::AbsoluteModuleRoot(settings.module_root.clone())); - } - let strip_root = source_tree.join(settings.module_root); - let module_root = strip_root.join(pyproject_toml.name().as_dist_info_name().as_ref()); - if !module_root.join("__init__.py").is_file() { - return Err(Error::MissingModule(module_root)); - } - let mut files_visited = 0; - for entry in WalkDir::new(module_root) - .into_iter() - .filter_entry(|entry| !exclude_matcher.is_match(entry.path())) - { - let entry = entry.map_err(|err| Error::WalkDir { - root: source_tree.to_path_buf(), - err, - })?; - - files_visited += 1; - if files_visited > 10000 { - warn_user_once!( - "Visited more than 10,000 files for wheel build. \ - Consider using more constrained includes or more excludes." - ); - } - - // We only want to take the module root, but since excludes start at the source tree root, - // we strip higher than we iterate. - let match_path = entry - .path() - .strip_prefix(source_tree) - .expect("walkdir starts with root"); - let wheel_path = entry - .path() - .strip_prefix(&strip_root) - .expect("walkdir starts with root"); - if exclude_matcher.is_match(match_path) { - trace!("Excluding from module: `{}`", match_path.user_display()); - continue; - } - let wheel_path = wheel_path.portable_display().to_string(); - - debug!("Adding to wheel: `{wheel_path}`"); - - if entry.file_type().is_dir() { - wheel_writer.write_directory(&wheel_path)?; - } else if entry.file_type().is_file() { - wheel_writer.write_file(&wheel_path, entry.path())?; - } else { - // TODO(konsti): We may want to support symlinks, there is support for installing them. - return Err(Error::UnsupportedFileType( - entry.path().to_path_buf(), - entry.file_type(), - )); - } - } - debug!("Visited {files_visited} files for wheel build"); - - // Add the license files - if let Some(license_files) = &pyproject_toml.license_files() { - debug!("Adding license files"); - let license_dir = format!( - "{}-{}.dist-info/licenses/", - pyproject_toml.name().as_dist_info_name(), - pyproject_toml.version() - ); - - wheel_subdir_from_globs( - source_tree, - &license_dir, - license_files, - &mut wheel_writer, - "project.license-files", - )?; - } - - // Add the data files - for (name, directory) in settings.data.iter() { - debug!("Adding {name} data files from: `{directory}`"); - let data_dir = format!( - "{}-{}.data/{}/", - pyproject_toml.name().as_dist_info_name(), - pyproject_toml.version(), - name - ); - - wheel_subdir_from_globs( - &source_tree.join(directory), - &data_dir, - &["**".to_string()], - &mut wheel_writer, - &format!("tool.uv.build-backend.data.{name}"), - )?; - } - - debug!("Adding metadata files to: `{}`", wheel_path.user_display()); - let dist_info_dir = write_dist_info( - &mut wheel_writer, - &pyproject_toml, - &filename, - source_tree, - uv_version, - )?; - wheel_writer.close(&dist_info_dir)?; - - Ok(filename) -} - -/// Build a wheel from the source tree and place it in the output directory. -pub fn build_editable( - source_tree: &Path, - wheel_dir: &Path, - metadata_directory: Option<&Path>, - uv_version: &str, -) -> Result { - let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; - let pyproject_toml = PyProjectToml::parse(&contents)?; - for warning in pyproject_toml.check_build_system(uv_version) { - warn_user_once!("{warning}"); - } - let settings = pyproject_toml - .settings() - .cloned() - .unwrap_or_else(BuildBackendSettings::default); - - check_metadata_directory(source_tree, metadata_directory, &pyproject_toml)?; - - let filename = WheelFilename { - name: pyproject_toml.name().clone(), - version: pyproject_toml.version().clone(), - build_tag: None, - python_tag: vec!["py3".to_string()], - abi_tag: vec!["none".to_string()], - platform_tag: vec!["any".to_string()], - }; - - let wheel_path = wheel_dir.join(filename.to_string()); - debug!("Writing wheel at {}", wheel_path.user_display()); - let mut wheel_writer = ZipDirectoryWriter::new_wheel(File::create(&wheel_path)?); - - debug!("Adding pth file to {}", wheel_path.user_display()); - if settings.module_root.is_absolute() { - return Err(Error::AbsoluteModuleRoot(settings.module_root.clone())); - } - let src_root = source_tree.join(settings.module_root); - let module_root = src_root.join(pyproject_toml.name().as_dist_info_name().as_ref()); - if !module_root.join("__init__.py").is_file() { - return Err(Error::MissingModule(module_root)); - } - wheel_writer.write_bytes( - &format!("{}.pth", pyproject_toml.name().as_dist_info_name()), - src_root.as_os_str().as_encoded_bytes(), - )?; - - debug!("Adding metadata files to: `{}`", wheel_path.user_display()); - let dist_info_dir = write_dist_info( - &mut wheel_writer, - &pyproject_toml, - &filename, - source_tree, - uv_version, - )?; - wheel_writer.close(&dist_info_dir)?; - - Ok(filename) -} - -/// Add the files and directories matching from the source tree matching any of the globs in the -/// wheel subdirectory. -fn wheel_subdir_from_globs( - src: &Path, - target: &str, - globs: &[String], - wheel_writer: &mut ZipDirectoryWriter, - // For error messages - globs_field: &str, -) -> Result<(), Error> { - let license_files_globs: Vec<_> = globs - .iter() - .map(|license_files| { - trace!( - "Including {} at `{}` with `{}`", - globs_field, - src.user_display(), - license_files - ); - parse_portable_glob(license_files) - }) - .collect::>() - .map_err(|err| Error::PortableGlob { - field: globs_field.to_string(), - source: err, - })?; - let matcher = - GlobDirFilter::from_globs(&license_files_globs).map_err(|err| Error::GlobSetTooLarge { - field: globs_field.to_string(), - source: err, - })?; - - wheel_writer.write_directory(target)?; - - for entry in WalkDir::new(src).into_iter().filter_entry(|entry| { - // TODO(konsti): This should be prettier. - let relative = entry - .path() - .strip_prefix(src) - .expect("walkdir starts with root"); - - // Fast path: Don't descend into a directory that can't be included. - matcher.match_directory(relative) - }) { - let entry = entry.map_err(|err| Error::WalkDir { - root: src.to_path_buf(), - err, - })?; - // TODO(konsti): This should be prettier. - let relative = entry - .path() - .strip_prefix(src) - .expect("walkdir starts with root"); - - if !matcher.match_path(relative) { - trace!("Excluding {}: `{}`", globs_field, relative.user_display()); - continue; - }; - - let relative_licenses = Path::new(target) - .join(relative) - .portable_display() - .to_string(); - - if entry.file_type().is_dir() { - wheel_writer.write_directory(&relative_licenses)?; - } else if entry.file_type().is_file() { - debug!("Adding {} file: `{}`", globs_field, relative.user_display()); - wheel_writer.write_file(&relative_licenses, entry.path())?; - } else { - // TODO(konsti): We may want to support symlinks, there is support for installing them. - return Err(Error::UnsupportedFileType( - entry.path().to_path_buf(), - entry.file_type(), - )); - } - } - Ok(()) -} - -/// Build a source distribution from the source tree and place it in the output directory. -pub fn build_source_dist( - source_tree: &Path, - source_dist_directory: &Path, - uv_version: &str, -) -> Result { - let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; - let pyproject_toml = PyProjectToml::parse(&contents)?; - let filename = SourceDistFilename { - name: pyproject_toml.name().clone(), - version: pyproject_toml.version().clone(), - extension: SourceDistExtension::TarGz, - }; - let source_dist_path = source_dist_directory.join(filename.to_string()); - let writer = TarGzWriter::new(&source_dist_path)?; - write_source_dist(source_tree, writer, uv_version)?; - Ok(filename) -} - -/// Shared implementation for building and listing a source distribution. -fn write_source_dist( - source_tree: &Path, - mut writer: impl DirectoryWriter, - uv_version: &str, -) -> Result { - let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; - let pyproject_toml = PyProjectToml::parse(&contents)?; - for warning in pyproject_toml.check_build_system(uv_version) { - warn_user_once!("{warning}"); - } - let settings = pyproject_toml - .settings() - .cloned() - .unwrap_or_else(BuildBackendSettings::default); - - let filename = SourceDistFilename { - name: pyproject_toml.name().clone(), - version: pyproject_toml.version().clone(), - extension: SourceDistExtension::TarGz, - }; - - let top_level = format!( - "{}-{}", - pyproject_toml.name().as_dist_info_name(), - pyproject_toml.version() - ); - - let metadata = pyproject_toml.to_metadata(source_tree)?; - let metadata_email = metadata.core_metadata_format(); - - writer.write_bytes( - &Path::new(&top_level) - .join("PKG-INFO") - .portable_display() - .to_string(), - metadata_email.as_bytes(), - )?; - - let (include_matcher, exclude_matcher) = source_dist_matcher(&pyproject_toml, settings)?; - - let mut files_visited = 0; - for entry in WalkDir::new(source_tree).into_iter().filter_entry(|entry| { - // TODO(konsti): This should be prettier. - let relative = entry - .path() - .strip_prefix(source_tree) - .expect("walkdir starts with root"); - - // Fast path: Don't descend into a directory that can't be included. This is the most - // important performance optimization, it avoids descending into directories such as - // `.venv`. While walkdir is generally cheap, we still avoid traversing large data - // directories that often exist on the top level of a project. This is especially noticeable - // on network file systems with high latencies per operation (while contiguous reading may - // still be fast). - include_matcher.match_directory(relative) && !exclude_matcher.is_match(relative) - }) { - let entry = entry.map_err(|err| Error::WalkDir { - root: source_tree.to_path_buf(), - err, - })?; - - files_visited += 1; - if files_visited > 10000 { - warn_user_once!( - "Visited more than 10,000 files for source distribution build. \ - Consider using more constrained includes or more excludes." - ); - } - // TODO(konsti): This should be prettier. - let relative = entry - .path() - .strip_prefix(source_tree) - .expect("walkdir starts with root"); - - if !include_matcher.match_path(relative) || exclude_matcher.is_match(relative) { - trace!("Excluding: `{}`", relative.user_display()); - continue; - }; - - debug!("Including {}", relative.user_display()); - if entry.file_type().is_dir() { - writer.write_directory( - &Path::new(&top_level) - .join(relative) - .portable_display() - .to_string(), - )?; - } else if entry.file_type().is_file() { - writer.write_file( - &Path::new(&top_level) - .join(relative) - .portable_display() - .to_string(), - entry.path(), - )?; - } else { - return Err(Error::UnsupportedFileType( - relative.to_path_buf(), - entry.file_type(), - )); - } - } - debug!("Visited {files_visited} files for source dist build"); - - writer.close(&top_level)?; - - Ok(filename) -} - -/// Build includes and excludes for source tree walking for source dists. -fn source_dist_matcher( - pyproject_toml: &PyProjectToml, - settings: BuildBackendSettings, -) -> Result<(GlobDirFilter, GlobSet), Error> { - // File and directories to include in the source directory - let mut include_globs = Vec::new(); - let mut includes: Vec = settings.source_include; - // pyproject.toml is always included. - includes.push(globset::escape("pyproject.toml")); - // The wheel must not include any files included by the source distribution (at least until we - // have files generated in the source dist -> wheel build step). - let import_path = &settings - .module_root - .join(pyproject_toml.name().as_dist_info_name().as_ref()) - .portable_display() - .to_string(); - includes.push(format!("{}/**", globset::escape(import_path))); - for include in includes { - let glob = parse_portable_glob(&include).map_err(|err| Error::PortableGlob { - field: "tool.uv.build-backend.source-include".to_string(), - source: err, - })?; - include_globs.push(glob.clone()); - } - - // Include the Readme - if let Some(readme) = pyproject_toml - .readme() - .as_ref() - .and_then(|readme| readme.path()) - { - trace!("Including readme at: `{}`", readme.user_display()); - include_globs.push( - Glob::new(&globset::escape(&readme.portable_display().to_string())) - .expect("escaped globset is parseable"), - ); - } - - // Include the license files - for license_files in pyproject_toml.license_files().into_iter().flatten() { - trace!("Including license files at: `{license_files}`"); - let glob = parse_portable_glob(license_files).map_err(|err| Error::PortableGlob { - field: "project.license-files".to_string(), - source: err, - })?; - include_globs.push(glob); - } - - // Include the data files - for (name, directory) in settings.data.iter() { - let glob = - parse_portable_glob(&format!("{}/**", globset::escape(directory))).map_err(|err| { - Error::PortableGlob { - field: format!("tool.uv.build-backend.data.{name}"), - source: err, - } - })?; - trace!("Including data ({name}) at: `{directory}`"); - include_globs.push(glob); - } - - let include_matcher = - GlobDirFilter::from_globs(&include_globs).map_err(|err| Error::GlobSetTooLarge { - field: "tool.uv.build-backend.source-include".to_string(), - source: err, - })?; - - let mut excludes: Vec = Vec::new(); - excludes.extend(settings.source_exclude); - debug!("Source dist excludes: {:?}", excludes); - let exclude_matcher = build_exclude_matcher(excludes)?; - if exclude_matcher.is_match("pyproject.toml") { - return Err(Error::PyprojectTomlExcluded); - } - Ok((include_matcher, exclude_matcher)) -} - /// Build a globset matcher for excludes. -fn build_exclude_matcher( +pub(crate) fn build_exclude_matcher( excludes: impl IntoIterator>, ) -> Result { let mut exclude_builder = GlobSetBuilder::new(); @@ -900,48 +98,10 @@ fn build_exclude_matcher( Ok(exclude_matcher) } -/// Write the dist-info directory to the output directory without building the wheel. -pub fn metadata( - source_tree: &Path, - metadata_directory: &Path, - uv_version: &str, -) -> Result { - let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; - let pyproject_toml = PyProjectToml::parse(&contents)?; - for warning in pyproject_toml.check_build_system(uv_version) { - warn_user_once!("{warning}"); - } - - let filename = WheelFilename { - name: pyproject_toml.name().clone(), - version: pyproject_toml.version().clone(), - build_tag: None, - python_tag: vec!["py3".to_string()], - abi_tag: vec!["none".to_string()], - platform_tag: vec!["any".to_string()], - }; - - debug!( - "Writing metadata files to {}", - metadata_directory.user_display() - ); - let mut wheel_writer = FilesystemWriter::new(metadata_directory); - let dist_info_dir = write_dist_info( - &mut wheel_writer, - &pyproject_toml, - &filename, - source_tree, - uv_version, - )?; - wheel_writer.close(&dist_info_dir)?; - - Ok(dist_info_dir) -} - /// PEP 517 requires that the metadata directory from the prepare metadata call is identical to the /// build wheel call. This method performs a prudence check that `METADATA` and `entry_points.txt` /// match. -fn check_metadata_directory( +pub(crate) fn check_metadata_directory( source_tree: &Path, metadata_directory: Option<&Path>, pyproject_toml: &PyProjectToml, @@ -986,146 +146,25 @@ fn check_metadata_directory( Ok(()) } -/// Add `METADATA` and `entry_points.txt` to the dist-info directory. -/// -/// Returns the name of the dist-info directory. -fn write_dist_info( - writer: &mut dyn DirectoryWriter, - pyproject_toml: &PyProjectToml, - filename: &WheelFilename, - root: &Path, - uv_version: &str, -) -> Result { - let dist_info_dir = format!( - "{}-{}.dist-info", - pyproject_toml.name().as_dist_info_name(), - pyproject_toml.version() - ); - - writer.write_directory(&dist_info_dir)?; - - // Add `WHEEL`. - let wheel_info = wheel_info(filename, uv_version); - writer.write_bytes(&format!("{dist_info_dir}/WHEEL"), wheel_info.as_bytes())?; - - // Add `entry_points.txt`. - if let Some(entrypoint) = pyproject_toml.to_entry_points()? { - writer.write_bytes( - &format!("{dist_info_dir}/entry_points.txt"), - entrypoint.as_bytes(), - )?; - } - - // Add `METADATA`. - let metadata = pyproject_toml.to_metadata(root)?.core_metadata_format(); - writer.write_bytes(&format!("{dist_info_dir}/METADATA"), metadata.as_bytes())?; - - // `RECORD` is added on closing. - - Ok(dist_info_dir) -} - -/// Returns the `WHEEL` file contents. -fn wheel_info(filename: &WheelFilename, uv_version: &str) -> String { - // https://packaging.python.org/en/latest/specifications/binary-distribution-format/#file-contents - let mut wheel_info = vec![ - ("Wheel-Version", "1.0".to_string()), - ("Generator", format!("uv {uv_version}")), - ("Root-Is-Purelib", "true".to_string()), - ]; - for python_tag in &filename.python_tag { - for abi_tag in &filename.abi_tag { - for platform_tag in &filename.platform_tag { - wheel_info.push(("Tag", format!("{python_tag}-{abi_tag}-{platform_tag}"))); - } - } - } - wheel_info - .into_iter() - .map(|(key, value)| format!("{key}: {value}")) - .join("\n") -} - -/// Write the `RECORD` file. -/// -/// -fn write_record( - writer: &mut dyn Write, - dist_info_dir: &str, - record: Vec, -) -> Result<(), Error> { - let mut record_writer = csv::Writer::from_writer(writer); - for entry in record { - record_writer.write_record(&[ - entry.path, - format!("sha256={}", entry.hash), - entry.size.to_string(), - ])?; - } - - // We can't compute the hash or size for RECORD without modifying it at the same time. - record_writer.write_record(&[ - format!("{dist_info_dir}/RECORD"), - String::new(), - String::new(), - ])?; - record_writer.flush()?; - Ok(()) -} - #[cfg(test)] mod tests { use super::*; + use crate::wheel; use flate2::bufread::GzDecoder; + use fs_err::File; use insta::assert_snapshot; - use std::str::FromStr; + use itertools::Itertools; + use std::io::BufReader; use tempfile::TempDir; use uv_fs::copy_dir_all; - use uv_normalize::PackageName; - use uv_pep440::Version; - - #[test] - fn test_wheel() { - let filename = WheelFilename { - name: PackageName::from_str("foo").unwrap(), - version: Version::from_str("1.2.3").unwrap(), - build_tag: None, - python_tag: vec!["py2".to_string(), "py3".to_string()], - abi_tag: vec!["none".to_string()], - platform_tag: vec!["any".to_string()], - }; - - assert_snapshot!(wheel_info(&filename, "1.0.0+test"), @r" - Wheel-Version: 1.0 - Generator: uv 1.0.0+test - Root-Is-Purelib: true - Tag: py2-none-any - Tag: py3-none-any - "); - } - - #[test] - fn test_record() { - let record = vec![RecordEntry { - path: "built_by_uv/__init__.py".to_string(), - hash: "89f869e53a3a0061a52c0233e6442d4d72de80a8a2d3406d9ea0bfd397ed7865".to_string(), - size: 37, - }]; - - let mut writer = Vec::new(); - write_record(&mut writer, "built_by_uv-0.1.0", record).unwrap(); - assert_snapshot!(String::from_utf8(writer).unwrap(), @r" - built_by_uv/__init__.py,sha256=89f869e53a3a0061a52c0233e6442d4d72de80a8a2d3406d9ea0bfd397ed7865,37 - built_by_uv-0.1.0/RECORD,, - "); - } + use walkdir::WalkDir; /// Snapshot all files from the prepare metadata hook. #[test] fn test_prepare_metadata() { let metadata_dir = TempDir::new().unwrap(); let built_by_uv = Path::new("../../scripts/packages/built-by-uv"); - metadata(built_by_uv, metadata_dir.path(), "1.0.0+test").unwrap(); + wheel::metadata(built_by_uv, metadata_dir.path(), "1.0.0+test").unwrap(); let mut files: Vec<_> = WalkDir::new(metadata_dir.path()) .into_iter() diff --git a/crates/uv-build-backend/src/source_dist.rs b/crates/uv-build-backend/src/source_dist.rs new file mode 100644 index 0000000000000..8966b9eda979b --- /dev/null +++ b/crates/uv-build-backend/src/source_dist.rs @@ -0,0 +1,218 @@ +use crate::fs_write_dispatcher::{FsWriteDispatcher, TarGzWriter}; +use crate::metadata::BuildBackendSettings; +use crate::{Error, PyProjectToml}; +use globset::{Glob, GlobSet}; +use std::path::Path; +use tracing::{debug, trace}; +use uv_distribution_filename::{SourceDistExtension, SourceDistFilename}; +use uv_fs::Simplified; +use uv_globfilter::{parse_portable_glob, GlobDirFilter}; +use uv_warnings::warn_user_once; +use walkdir::WalkDir; + +/// Build a source distribution from the source tree and place it in the output directory. +pub fn build_source_dist( + source_tree: &Path, + source_dist_directory: &Path, + uv_version: &str, +) -> Result { + let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; + let pyproject_toml = PyProjectToml::parse(&contents)?; + let filename = SourceDistFilename { + name: pyproject_toml.name().clone(), + version: pyproject_toml.version().clone(), + extension: SourceDistExtension::TarGz, + }; + let source_dist_path = source_dist_directory.join(filename.to_string()); + let writer = TarGzWriter::new(&source_dist_path)?; + write_source_dist(source_tree, writer, uv_version)?; + Ok(filename) +} + +/// Shared implementation for building and listing a source distribution. +fn write_source_dist( + source_tree: &Path, + mut writer: impl FsWriteDispatcher, + uv_version: &str, +) -> Result { + let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; + let pyproject_toml = PyProjectToml::parse(&contents)?; + for warning in pyproject_toml.check_build_system(uv_version) { + warn_user_once!("{warning}"); + } + let settings = pyproject_toml + .settings() + .cloned() + .unwrap_or_else(BuildBackendSettings::default); + + let filename = SourceDistFilename { + name: pyproject_toml.name().clone(), + version: pyproject_toml.version().clone(), + extension: SourceDistExtension::TarGz, + }; + + let top_level = format!( + "{}-{}", + pyproject_toml.name().as_dist_info_name(), + pyproject_toml.version() + ); + + let metadata = pyproject_toml.to_metadata(source_tree)?; + let metadata_email = metadata.core_metadata_format(); + + writer.write_bytes( + &Path::new(&top_level) + .join("PKG-INFO") + .portable_display() + .to_string(), + metadata_email.as_bytes(), + )?; + + let (include_matcher, exclude_matcher) = source_dist_matcher(&pyproject_toml, settings)?; + + let mut files_visited = 0; + for entry in WalkDir::new(source_tree).into_iter().filter_entry(|entry| { + // TODO(konsti): This should be prettier. + let relative = entry + .path() + .strip_prefix(source_tree) + .expect("walkdir starts with root"); + + // Fast path: Don't descend into a directory that can't be included. This is the most + // important performance optimization, it avoids descending into directories such as + // `.venv`. While walkdir is generally cheap, we still avoid traversing large data + // directories that often exist on the top level of a project. This is especially noticeable + // on network file systems with high latencies per operation (while contiguous reading may + // still be fast). + include_matcher.match_directory(relative) && !exclude_matcher.is_match(relative) + }) { + let entry = entry.map_err(|err| Error::WalkDir { + root: source_tree.to_path_buf(), + err, + })?; + + files_visited += 1; + if files_visited > 10000 { + warn_user_once!( + "Visited more than 10,000 files for source distribution build. \ + Consider using more constrained includes or more excludes." + ); + } + // TODO(konsti): This should be prettier. + let relative = entry + .path() + .strip_prefix(source_tree) + .expect("walkdir starts with root"); + + if !include_matcher.match_path(relative) || exclude_matcher.is_match(relative) { + trace!("Excluding: `{}`", relative.user_display()); + continue; + }; + + debug!("Including {}", relative.user_display()); + if entry.file_type().is_dir() { + writer.write_directory( + &Path::new(&top_level) + .join(relative) + .portable_display() + .to_string(), + )?; + } else if entry.file_type().is_file() { + writer.write_file( + &Path::new(&top_level) + .join(relative) + .portable_display() + .to_string(), + entry.path(), + )?; + } else { + return Err(Error::UnsupportedFileType( + relative.to_path_buf(), + entry.file_type(), + )); + } + } + debug!("Visited {files_visited} files for source dist build"); + + writer.close(&top_level)?; + + Ok(filename) +} + +/// Build includes and excludes for source tree walking for source dists. +fn source_dist_matcher( + pyproject_toml: &PyProjectToml, + settings: BuildBackendSettings, +) -> Result<(GlobDirFilter, GlobSet), Error> { + // File and directories to include in the source directory + let mut include_globs = Vec::new(); + let mut includes: Vec = settings.source_include; + // pyproject.toml is always included. + includes.push(globset::escape("pyproject.toml")); + // The wheel must not include any files included by the source distribution (at least until we + // have files generated in the source dist -> wheel build step). + let import_path = &settings + .module_root + .join(pyproject_toml.name().as_dist_info_name().as_ref()) + .portable_display() + .to_string(); + includes.push(format!("{}/**", globset::escape(import_path))); + for include in includes { + let glob = parse_portable_glob(&include).map_err(|err| Error::PortableGlob { + field: "tool.uv.build-backend.source-include".to_string(), + source: err, + })?; + include_globs.push(glob.clone()); + } + + // Include the Readme + if let Some(readme) = pyproject_toml + .readme() + .as_ref() + .and_then(|readme| readme.path()) + { + trace!("Including readme at: `{}`", readme.user_display()); + include_globs.push( + Glob::new(&globset::escape(&readme.portable_display().to_string())) + .expect("escaped globset is parseable"), + ); + } + + // Include the license files + for license_files in pyproject_toml.license_files().into_iter().flatten() { + trace!("Including license files at: `{license_files}`"); + let glob = parse_portable_glob(license_files).map_err(|err| Error::PortableGlob { + field: "project.license-files".to_string(), + source: err, + })?; + include_globs.push(glob); + } + + // Include the data files + for (name, directory) in settings.data.iter() { + let glob = + parse_portable_glob(&format!("{}/**", globset::escape(directory))).map_err(|err| { + Error::PortableGlob { + field: format!("tool.uv.build-backend.data.{name}"), + source: err, + } + })?; + trace!("Including data ({name}) at: `{directory}`"); + include_globs.push(glob); + } + + let include_matcher = + GlobDirFilter::from_globs(&include_globs).map_err(|err| Error::GlobSetTooLarge { + field: "tool.uv.build-backend.source-include".to_string(), + source: err, + })?; + + let mut excludes: Vec = Vec::new(); + excludes.extend(settings.source_exclude); + debug!("Source dist excludes: {:?}", excludes); + let exclude_matcher = crate::build_exclude_matcher(excludes)?; + if exclude_matcher.is_match("pyproject.toml") { + return Err(Error::PyprojectTomlExcluded); + } + Ok((include_matcher, exclude_matcher)) +} diff --git a/crates/uv-build-backend/src/wheel.rs b/crates/uv-build-backend/src/wheel.rs new file mode 100644 index 0000000000000..c4c4bf2e10478 --- /dev/null +++ b/crates/uv-build-backend/src/wheel.rs @@ -0,0 +1,527 @@ +use crate::fs_write_dispatcher::{FilesystemWriter, FsWriteDispatcher, ZipDirectoryWriter}; +use crate::metadata::BuildBackendSettings; +use crate::{Error, PyProjectToml}; +use fs_err::File; +use itertools::Itertools; +use sha2::{Digest, Sha256}; +use std::io; +use std::io::{Read, Write}; +use std::path::Path; +use tracing::{debug, trace}; +use uv_distribution_filename::WheelFilename; +use uv_fs::Simplified; +use uv_globfilter::{parse_portable_glob, GlobDirFilter}; +use uv_warnings::warn_user_once; +use walkdir::WalkDir; + +/// Write the dist-info directory to the output directory without building the wheel. +pub fn metadata( + source_tree: &Path, + metadata_directory: &Path, + uv_version: &str, +) -> Result { + let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; + let pyproject_toml = PyProjectToml::parse(&contents)?; + for warning in pyproject_toml.check_build_system(uv_version) { + warn_user_once!("{warning}"); + } + + let filename = WheelFilename { + name: pyproject_toml.name().clone(), + version: pyproject_toml.version().clone(), + build_tag: None, + python_tag: vec!["py3".to_string()], + abi_tag: vec!["none".to_string()], + platform_tag: vec!["any".to_string()], + }; + + debug!( + "Writing metadata files to {}", + metadata_directory.user_display() + ); + let mut wheel_writer = FilesystemWriter::new(metadata_directory); + let dist_info_dir = write_dist_info( + &mut wheel_writer, + &pyproject_toml, + &filename, + source_tree, + uv_version, + )?; + wheel_writer.close(&dist_info_dir)?; + + Ok(dist_info_dir) +} + +/// Build a wheel from the source tree and place it in the output directory. +pub fn build_editable( + source_tree: &Path, + wheel_dir: &Path, + metadata_directory: Option<&Path>, + uv_version: &str, +) -> Result { + let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; + let pyproject_toml = PyProjectToml::parse(&contents)?; + for warning in pyproject_toml.check_build_system(uv_version) { + warn_user_once!("{warning}"); + } + let settings = pyproject_toml + .settings() + .cloned() + .unwrap_or_else(BuildBackendSettings::default); + + crate::check_metadata_directory(source_tree, metadata_directory, &pyproject_toml)?; + + let filename = WheelFilename { + name: pyproject_toml.name().clone(), + version: pyproject_toml.version().clone(), + build_tag: None, + python_tag: vec!["py3".to_string()], + abi_tag: vec!["none".to_string()], + platform_tag: vec!["any".to_string()], + }; + + let wheel_path = wheel_dir.join(filename.to_string()); + debug!("Writing wheel at {}", wheel_path.user_display()); + let mut wheel_writer = ZipDirectoryWriter::new_wheel(File::create(&wheel_path)?); + + debug!("Adding pth file to {}", wheel_path.user_display()); + if settings.module_root.is_absolute() { + return Err(Error::AbsoluteModuleRoot(settings.module_root.clone())); + } + let src_root = source_tree.join(settings.module_root); + let module_root = src_root.join(pyproject_toml.name().as_dist_info_name().as_ref()); + if !module_root.join("__init__.py").is_file() { + return Err(Error::MissingModule(module_root)); + } + wheel_writer.write_bytes( + &format!("{}.pth", pyproject_toml.name().as_dist_info_name()), + src_root.as_os_str().as_encoded_bytes(), + )?; + + debug!("Adding metadata files to: `{}`", wheel_path.user_display()); + let dist_info_dir = write_dist_info( + &mut wheel_writer, + &pyproject_toml, + &filename, + source_tree, + uv_version, + )?; + wheel_writer.close(&dist_info_dir)?; + + Ok(filename) +} + +/// Build a wheel from the source tree and place it in the output directory. +pub fn build_wheel( + source_tree: &Path, + wheel_dir: &Path, + metadata_directory: Option<&Path>, + uv_version: &str, +) -> Result { + let contents = fs_err::read_to_string(source_tree.join("pyproject.toml"))?; + let pyproject_toml = PyProjectToml::parse(&contents)?; + for warning in pyproject_toml.check_build_system(uv_version) { + warn_user_once!("{warning}"); + } + let settings = pyproject_toml + .settings() + .cloned() + .unwrap_or_else(BuildBackendSettings::default); + + crate::check_metadata_directory(source_tree, metadata_directory, &pyproject_toml)?; + + let filename = WheelFilename { + name: pyproject_toml.name().clone(), + version: pyproject_toml.version().clone(), + build_tag: None, + python_tag: vec!["py3".to_string()], + abi_tag: vec!["none".to_string()], + platform_tag: vec!["any".to_string()], + }; + + let wheel_path = wheel_dir.join(filename.to_string()); + debug!("Writing wheel at {}", wheel_path.user_display()); + let mut wheel_writer = ZipDirectoryWriter::new_wheel(File::create(&wheel_path)?); + + // Wheel excludes + let mut excludes: Vec = settings.wheel_exclude; + // The wheel must not include any files excluded by the source distribution (at least until we + // have files generated in the source dist -> wheel build step). + for exclude in settings.source_exclude { + // Avoid duplicate entries. + if !excludes.contains(&exclude) { + excludes.push(exclude); + } + } + debug!("Wheel excludes: {:?}", excludes); + let exclude_matcher = crate::build_exclude_matcher(excludes)?; + + debug!("Adding content files to {}", wheel_path.user_display()); + if settings.module_root.is_absolute() { + return Err(Error::AbsoluteModuleRoot(settings.module_root.clone())); + } + let strip_root = source_tree.join(settings.module_root); + let module_root = strip_root.join(pyproject_toml.name().as_dist_info_name().as_ref()); + if !module_root.join("__init__.py").is_file() { + return Err(Error::MissingModule(module_root)); + } + let mut files_visited = 0; + for entry in WalkDir::new(module_root) + .into_iter() + .filter_entry(|entry| !exclude_matcher.is_match(entry.path())) + { + let entry = entry.map_err(|err| Error::WalkDir { + root: source_tree.to_path_buf(), + err, + })?; + + files_visited += 1; + if files_visited > 10000 { + warn_user_once!( + "Visited more than 10,000 files for wheel build. \ + Consider using more constrained includes or more excludes." + ); + } + + // We only want to take the module root, but since excludes start at the source tree root, + // we strip higher than we iterate. + let match_path = entry + .path() + .strip_prefix(source_tree) + .expect("walkdir starts with root"); + let wheel_path = entry + .path() + .strip_prefix(&strip_root) + .expect("walkdir starts with root"); + if exclude_matcher.is_match(match_path) { + trace!("Excluding from module: `{}`", match_path.user_display()); + continue; + } + let wheel_path = wheel_path.portable_display().to_string(); + + debug!("Adding to wheel: `{wheel_path}`"); + + if entry.file_type().is_dir() { + wheel_writer.write_directory(&wheel_path)?; + } else if entry.file_type().is_file() { + wheel_writer.write_file(&wheel_path, entry.path())?; + } else { + // TODO(konsti): We may want to support symlinks, there is support for installing them. + return Err(Error::UnsupportedFileType( + entry.path().to_path_buf(), + entry.file_type(), + )); + } + } + debug!("Visited {files_visited} files for wheel build"); + + // Add the license files + if let Some(license_files) = &pyproject_toml.license_files() { + debug!("Adding license files"); + let license_dir = format!( + "{}-{}.dist-info/licenses/", + pyproject_toml.name().as_dist_info_name(), + pyproject_toml.version() + ); + + wheel_subdir_from_globs( + source_tree, + &license_dir, + license_files, + &mut wheel_writer, + "project.license-files", + )?; + } + + // Add the data files + for (name, directory) in settings.data.iter() { + debug!("Adding {name} data files from: `{directory}`"); + let data_dir = format!( + "{}-{}.data/{}/", + pyproject_toml.name().as_dist_info_name(), + pyproject_toml.version(), + name + ); + + wheel_subdir_from_globs( + &source_tree.join(directory), + &data_dir, + &["**".to_string()], + &mut wheel_writer, + &format!("tool.uv.build-backend.data.{name}"), + )?; + } + + debug!("Adding metadata files to: `{}`", wheel_path.user_display()); + let dist_info_dir = write_dist_info( + &mut wheel_writer, + &pyproject_toml, + &filename, + source_tree, + uv_version, + )?; + wheel_writer.close(&dist_info_dir)?; + + Ok(filename) +} + +/// An entry in the `RECORD` file. +/// +/// +pub(crate) struct RecordEntry { + /// The path to the file relative to the package root. + /// + /// While the spec would allow backslashes, we always use portable paths with forward slashes. + pub(crate) path: String, + /// The SHA256 of the files. + pub(crate) hash: String, + /// The size of the file in bytes. + pub(crate) size: usize, +} + +/// Read the input file and write it both to the hasher and the target file. +/// +/// We're implementing this tee-ing manually since there is no sync `InspectReader` or std tee +/// function. +pub(crate) fn write_hashed( + path: &str, + reader: &mut dyn Read, + writer: &mut dyn Write, +) -> Result { + let mut hasher = Sha256::new(); + let mut size = 0; + // 8KB is the default defined in `std::sys_common::io`. + let mut buffer = vec![0; 8 * 1024]; + loop { + let read = match reader.read(&mut buffer) { + Ok(read) => read, + Err(err) if err.kind() == io::ErrorKind::Interrupted => continue, + Err(err) => return Err(err), + }; + if read == 0 { + // End of file + break; + } + hasher.update(&buffer[..read]); + writer.write_all(&buffer[..read])?; + size += read; + } + Ok(RecordEntry { + path: path.to_string(), + hash: format!("{:x}", hasher.finalize()), + size, + }) +} + +/// Add the files and directories matching from the source tree matching any of the globs in the +/// wheel subdirectory. +fn wheel_subdir_from_globs( + src: &Path, + target: &str, + globs: &[String], + wheel_writer: &mut ZipDirectoryWriter, + // For error messages + globs_field: &str, +) -> Result<(), Error> { + let license_files_globs: Vec<_> = globs + .iter() + .map(|license_files| { + trace!( + "Including {} at `{}` with `{}`", + globs_field, + src.user_display(), + license_files + ); + parse_portable_glob(license_files) + }) + .collect::>() + .map_err(|err| Error::PortableGlob { + field: globs_field.to_string(), + source: err, + })?; + let matcher = + GlobDirFilter::from_globs(&license_files_globs).map_err(|err| Error::GlobSetTooLarge { + field: globs_field.to_string(), + source: err, + })?; + + wheel_writer.write_directory(target)?; + + for entry in WalkDir::new(src).into_iter().filter_entry(|entry| { + // TODO(konsti): This should be prettier. + let relative = entry + .path() + .strip_prefix(src) + .expect("walkdir starts with root"); + + // Fast path: Don't descend into a directory that can't be included. + matcher.match_directory(relative) + }) { + let entry = entry.map_err(|err| Error::WalkDir { + root: src.to_path_buf(), + err, + })?; + // TODO(konsti): This should be prettier. + let relative = entry + .path() + .strip_prefix(src) + .expect("walkdir starts with root"); + + if !matcher.match_path(relative) { + trace!("Excluding {}: `{}`", globs_field, relative.user_display()); + continue; + }; + + let relative_licenses = Path::new(target) + .join(relative) + .portable_display() + .to_string(); + + if entry.file_type().is_dir() { + wheel_writer.write_directory(&relative_licenses)?; + } else if entry.file_type().is_file() { + debug!("Adding {} file: `{}`", globs_field, relative.user_display()); + wheel_writer.write_file(&relative_licenses, entry.path())?; + } else { + // TODO(konsti): We may want to support symlinks, there is support for installing them. + return Err(Error::UnsupportedFileType( + entry.path().to_path_buf(), + entry.file_type(), + )); + } + } + Ok(()) +} + +/// Add `METADATA` and `entry_points.txt` to the dist-info directory. +/// +/// Returns the name of the dist-info directory. +pub(crate) fn write_dist_info( + writer: &mut dyn FsWriteDispatcher, + pyproject_toml: &PyProjectToml, + filename: &WheelFilename, + root: &Path, + uv_version: &str, +) -> Result { + let dist_info_dir = format!( + "{}-{}.dist-info", + pyproject_toml.name().as_dist_info_name(), + pyproject_toml.version() + ); + + writer.write_directory(&dist_info_dir)?; + + // Add `WHEEL`. + let wheel_info = wheel_info(filename, uv_version); + writer.write_bytes(&format!("{dist_info_dir}/WHEEL"), wheel_info.as_bytes())?; + + // Add `entry_points.txt`. + if let Some(entrypoint) = pyproject_toml.to_entry_points()? { + writer.write_bytes( + &format!("{dist_info_dir}/entry_points.txt"), + entrypoint.as_bytes(), + )?; + } + + // Add `METADATA`. + let metadata = pyproject_toml.to_metadata(root)?.core_metadata_format(); + writer.write_bytes(&format!("{dist_info_dir}/METADATA"), metadata.as_bytes())?; + + // `RECORD` is added on closing. + + Ok(dist_info_dir) +} + +/// Returns the `WHEEL` file contents. +fn wheel_info(filename: &WheelFilename, uv_version: &str) -> String { + // https://packaging.python.org/en/latest/specifications/binary-distribution-format/#file-contents + let mut wheel_info = vec![ + ("Wheel-Version", "1.0".to_string()), + ("Generator", format!("uv {uv_version}")), + ("Root-Is-Purelib", "true".to_string()), + ]; + for python_tag in &filename.python_tag { + for abi_tag in &filename.abi_tag { + for platform_tag in &filename.platform_tag { + wheel_info.push(("Tag", format!("{python_tag}-{abi_tag}-{platform_tag}"))); + } + } + } + wheel_info + .into_iter() + .map(|(key, value)| format!("{key}: {value}")) + .join("\n") +} + +/// Write the `RECORD` file. +/// +/// +pub(crate) fn write_record( + writer: &mut dyn Write, + dist_info_dir: &str, + record: Vec, +) -> Result<(), Error> { + let mut record_writer = csv::Writer::from_writer(writer); + for entry in record { + record_writer.write_record(&[ + entry.path, + format!("sha256={}", entry.hash), + entry.size.to_string(), + ])?; + } + + // We can't compute the hash or size for RECORD without modifying it at the same time. + record_writer.write_record(&[ + format!("{dist_info_dir}/RECORD"), + String::new(), + String::new(), + ])?; + record_writer.flush()?; + Ok(()) +} + +#[cfg(test)] +mod tests { + use crate::wheel::{wheel_info, write_record, RecordEntry}; + use insta::assert_snapshot; + use std::str::FromStr; + use uv_distribution_filename::WheelFilename; + use uv_normalize::PackageName; + use uv_pep440::Version; + + #[test] + fn test_wheel() { + let filename = WheelFilename { + name: PackageName::from_str("foo").unwrap(), + version: Version::from_str("1.2.3").unwrap(), + build_tag: None, + python_tag: vec!["py2".to_string(), "py3".to_string()], + abi_tag: vec!["none".to_string()], + platform_tag: vec!["any".to_string()], + }; + + assert_snapshot!(wheel_info(&filename, "1.0.0+test"), @r" + Wheel-Version: 1.0 + Generator: uv 1.0.0+test + Root-Is-Purelib: true + Tag: py2-none-any + Tag: py3-none-any + "); + } + + #[test] + fn test_record() { + let record = vec![RecordEntry { + path: "built_by_uv/__init__.py".to_string(), + hash: "89f869e53a3a0061a52c0233e6442d4d72de80a8a2d3406d9ea0bfd397ed7865".to_string(), + size: 37, + }]; + + let mut writer = Vec::new(); + write_record(&mut writer, "built_by_uv-0.1.0", record).unwrap(); + assert_snapshot!(String::from_utf8(writer).unwrap(), @r" + built_by_uv/__init__.py,sha256=89f869e53a3a0061a52c0233e6442d4d72de80a8a2d3406d9ea0bfd397ed7865,37 + built_by_uv-0.1.0/RECORD,, + "); + } +}