diff --git a/Cargo.lock b/Cargo.lock index 091de2d4a..3fcc33667 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2915,6 +2915,15 @@ version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" +[[package]] +name = "ordered-float" +version = "2.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68f19d67e5a2795c94e73e0bb1cc1a7edeb2e28efd39e2e1c9b7a40c1108b11c" +dependencies = [ + "num-traits", +] + [[package]] name = "ordered-stream" version = "0.2.0" @@ -3386,6 +3395,7 @@ dependencies = [ "rstest", "serde", "serde-untagged", + "serde-value", "serde_json", "serde_with", "spdx", @@ -4849,6 +4859,16 @@ dependencies = [ "typeid", ] +[[package]] +name = "serde-value" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3a1a3341211875ef120e117ea7fd5228530ae7e7036a779fdc9117be6b3282c" +dependencies = [ + "ordered-float", + "serde", +] + [[package]] name = "serde_derive" version = "1.0.214" diff --git a/Cargo.toml b/Cargo.toml index 6bcdc51d8..f8f5d95e5 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -67,6 +67,7 @@ rstest = "0.19.0" self-replace = "1.3.7" serde = "1.0.198" serde-untagged = "0.1.5" +serde-value = "0.7.0" serde_ignored = "0.1.10" serde_json = "1.0.116" serde_with = "3.7.0" diff --git a/crates/pixi_manifest/Cargo.toml b/crates/pixi_manifest/Cargo.toml index a8cdcfa07..9abf8d73e 100644 --- a/crates/pixi_manifest/Cargo.toml +++ b/crates/pixi_manifest/Cargo.toml @@ -20,6 +20,7 @@ pixi_spec = { workspace = true } regex = { workspace = true } serde = { workspace = true } serde-untagged = { workspace = true } +serde-value = { workspace = true } serde_with = { workspace = true } spdx = { workspace = true } strsim = { workspace = true } @@ -47,5 +48,5 @@ fancy_display = { workspace = true } glob = "0.3.1" insta = { workspace = true, features = ["yaml"] } rstest = { workspace = true } -serde_json = { workspace = true } +serde_json = { workspace = true, features = ["preserve_order"] } tempfile = { workspace = true } diff --git a/crates/pixi_manifest/src/lib.rs b/crates/pixi_manifest/src/lib.rs index fbdabfe57..967558b4c 100644 --- a/crates/pixi_manifest/src/lib.rs +++ b/crates/pixi_manifest/src/lib.rs @@ -11,6 +11,7 @@ mod has_manifest_ref; mod manifests; mod metadata; mod parsed_manifest; +mod preview; pub mod pypi; pub mod pyproject; mod solve_group; @@ -48,6 +49,7 @@ use thiserror::Error; pub use features_ext::FeaturesExt; pub use has_features_iter::HasFeaturesIter; pub use has_manifest_ref::HasManifestRef; +pub use preview::{KnownPreviewFeature, Preview, PreviewFeature}; /// Errors that can occur when getting a feature. #[derive(Debug, Clone, Error, Diagnostic)] diff --git a/crates/pixi_manifest/src/manifests/manifest.rs b/crates/pixi_manifest/src/manifests/manifest.rs index c2a186668..b1dea4787 100644 --- a/crates/pixi_manifest/src/manifests/manifest.rs +++ b/crates/pixi_manifest/src/manifests/manifest.rs @@ -19,6 +19,7 @@ use crate::{ consts, error::{DependencyError, TomlError, UnknownFeature}, manifests::{ManifestSource, TomlManifest}, + preview::Preview, pypi::PyPiPackageName, pyproject::PyProjectManifest, to_options, DependencyOverwriteBehavior, Environment, EnvironmentName, Feature, FeatureName, @@ -722,6 +723,11 @@ impl Manifest { { self.parsed.environments.find(name) } + + /// Returns the preview field of the project + pub fn preview(&self) -> Option<&Preview> { + self.parsed.project.preview.as_ref() + } } #[cfg(test)] diff --git a/crates/pixi_manifest/src/metadata.rs b/crates/pixi_manifest/src/metadata.rs index 2a1dd80ce..ddf1bd7c1 100644 --- a/crates/pixi_manifest/src/metadata.rs +++ b/crates/pixi_manifest/src/metadata.rs @@ -8,6 +8,7 @@ use serde_with::{serde_as, DisplayFromStr}; use url::Url; use super::pypi::pypi_options::PypiOptions; +use crate::preview::Preview; use crate::utils::PixiSpanned; /// Describes the contents of the `[package]` section of the project manifest. @@ -64,4 +65,7 @@ pub struct ProjectMetadata { /// The pypi options supported in the project pub pypi_options: Option, + + /// Preview features + pub preview: Option, } diff --git a/crates/pixi_manifest/src/preview.rs b/crates/pixi_manifest/src/preview.rs new file mode 100644 index 000000000..c7bb190d1 --- /dev/null +++ b/crates/pixi_manifest/src/preview.rs @@ -0,0 +1,203 @@ +//! This module contains the ability to parse the preview features of the project +//! +//! e.g. +//! ```toml +//! [project] +//! # .. other project metadata +//! preview = ["new-resolve"] +//! ``` +//! +//! Features are split into Known and Unknown features. Basically you can use any string as a feature +//! but only the features defined in [`KnownFeature`] can be used. +//! We do this for backwards compatibility with the old features that may have been used in the past. +//! The [`KnownFeature`] enum contains all the known features. Extend this if you want to add support +//! for new features. +use serde::{Deserialize, Deserializer, Serialize}; + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(untagged)] +/// The preview features of the project +pub enum Preview { + /// All preview features are enabled + AllEnabled(bool), // For `preview = true` + /// Specific preview features are enabled + Features(Vec), // For `preview = ["feature"]` +} + +impl Preview { + /// Returns true if all preview features are enabled + pub fn all_enabled(&self) -> bool { + match self { + Preview::AllEnabled(enabled) => *enabled, + Preview::Features(_) => false, + } + } + + /// Returns true if the given preview feature is enabled + pub fn is_enabled(&self, feature: KnownPreviewFeature) -> bool { + match self { + Preview::AllEnabled(_) => true, + Preview::Features(features) => features.iter().any(|f| *f == feature), + } + } + + /// Return all unknown preview features + pub fn unknown_preview_features(&self) -> Vec<&str> { + match self { + Preview::AllEnabled(_) => vec![], + Preview::Features(features) => features + .iter() + .filter_map(|feature| match feature { + PreviewFeature::Unknown(feature) => Some(feature.as_str()), + _ => None, + }) + .collect(), + } + } +} + +impl<'de> Deserialize<'de> for Preview { + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + serde_untagged::UntaggedEnumVisitor::new() + .bool(|bool| Ok(Preview::AllEnabled(bool))) + .seq(|seq| Ok(Preview::Features(seq.deserialize()?))) + .expecting("bool or list of features e.g `true` or `[\"new-resolve\"]`") + .deserialize(deserializer) + } +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(untagged)] +/// A preview feature, can be either a known feature or an unknown feature +pub enum PreviewFeature { + /// This is a known preview feature + Known(KnownPreviewFeature), + /// Unknown preview feature + Unknown(String), +} + +impl PartialEq for PreviewFeature { + fn eq(&self, other: &KnownPreviewFeature) -> bool { + match self { + PreviewFeature::Known(feature) => feature == other, + _ => false, + } + } +} + +#[derive(Debug, Serialize, Deserialize, Clone, Copy, PartialEq)] +#[serde(rename_all = "kebab-case")] +/// Currently supported preview features are listed here +pub enum KnownPreviewFeature { + // Add known features here +} + +impl<'de> Deserialize<'de> for PreviewFeature { + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + let value = serde_value::Value::deserialize(deserializer)?; + let known = KnownPreviewFeature::deserialize(value.clone()).map(PreviewFeature::Known); + if let Ok(feature) = known { + Ok(feature) + } else { + let unknown = String::deserialize(value) + .map(PreviewFeature::Unknown) + .map_err(serde::de::Error::custom)?; + Ok(unknown) + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use insta::assert_snapshot; + use toml_edit::{de::from_str, ser::to_string}; + + /// Fake table to test the `Preview` enum + #[derive(Debug, Serialize, Deserialize)] + struct TopLevel { + preview: Preview, + } + + #[test] + fn test_preview_all_enabled() { + let input = "preview = true"; + let top: TopLevel = from_str(input).expect("should parse as `AllEnabled`"); + assert_eq!(top.preview, Preview::AllEnabled(true)); + + let output = to_string(&top).expect("should serialize back to TOML"); + assert_eq!(output.trim(), input); + } + + #[test] + fn test_preview_with_unknown_feature() { + let input = r#"preview = ["build"]"#; + let top: TopLevel = from_str(input).expect("should parse as `Features` with known feature"); + assert_eq!( + top.preview, + Preview::Features(vec![PreviewFeature::Unknown("build".to_string())]) + ); + + let output = to_string(&top).expect("should serialize back to TOML"); + assert_eq!(output.trim(), input); + } + + #[test] + fn test_insta_error_invalid_bool() { + let input = r#"preview = "not-a-bool""#; + let result: Result = from_str(input); + + assert!(result.is_err()); + assert_snapshot!( + format!("{:?}", result.unwrap_err()), + @r###"Error { inner: TomlError { message: "invalid type: map, expected bool or list of features e.g `true` or `[\"new-resolve\"]`", raw: Some("preview = \"not-a-bool\""), keys: [], span: Some(0..22) } }"### + ); + } + + #[test] + fn test_insta_error_invalid_list_item() { + let input = r#"preview = ["build", 123]"#; + let result: Result = from_str(input); + + assert!(result.is_err()); + assert_snapshot!( + format!("{:?}", result.unwrap_err()), + @r###"Error { inner: TomlError { message: "Invalid type integer `123`. Expected a string\n", raw: Some("preview = [\"build\", 123]"), keys: ["preview"], span: Some(10..24) } }"### + ); + } + + #[test] + fn test_insta_error_invalid_top_level_type() { + let input = r#"preview = 123"#; + let result: Result = from_str(input); + + assert!(result.is_err()); + assert_snapshot!( + format!("{:?}", result.unwrap_err()), + @r###"Error { inner: TomlError { message: "invalid type: integer `123`, expected bool or list of features e.g `true` or `[\"new-resolve\"]`", raw: Some("preview = 123"), keys: ["preview"], span: Some(10..13) } }"### + ); + } + + #[test] + fn test_feature_is_unknown() { + let input = r#"preview = ["new_parsing"]"#; + let top: TopLevel = from_str(input).unwrap(); + match top.preview { + Preview::AllEnabled(_) => unreachable!("this arm should not be used"), + Preview::Features(vec) => { + assert_matches::assert_matches!( + &vec[0], + PreviewFeature::Unknown(s) => { + s == &"new_parsing".to_string() + } + ); + } + } + } +} diff --git a/crates/pixi_manifest/src/validation.rs b/crates/pixi_manifest/src/validation.rs index 309f304bb..c34a38cc1 100644 --- a/crates/pixi_manifest/src/validation.rs +++ b/crates/pixi_manifest/src/validation.rs @@ -132,6 +132,22 @@ impl ParsedManifest { } } + // Warn on any unknown preview features + if let Some(preview) = self.project.preview.as_ref() { + let preview = preview.unknown_preview_features(); + if !preview.is_empty() { + let are = if preview.len() > 1 { "are" } else { "is" }; + let s = if preview.len() > 1 { "s" } else { "" }; + let preview_array = if preview.len() == 1 { + format!("{:?}", preview) + } else { + format!("[{:?}]", preview.iter().format(", ")) + }; + tracing::warn!( + "The preview feature{s}: {preview_array} {are} defined in the manifest but un-used pixi"); + } + } + Ok(()) } diff --git a/docs/reference/project_configuration.md b/docs/reference/project_configuration.md index 8f107fe5e..9897e4bec 100644 --- a/docs/reference/project_configuration.md +++ b/docs/reference/project_configuration.md @@ -769,6 +769,22 @@ When an environment comprises several features (including the default feature): - The `channels` of the environment is the union of the `channels` of all its features. Channel priorities can be specified in each feature, to ensure channels are considered in the right order in the environment. - The `platforms` of the environment is the intersection of the `platforms` of all its features. Be aware that the platforms supported by a feature (including the default feature) will be considered as the `platforms` defined at project level (unless overridden in the feature). This means that it is usually a good idea to set the project `platforms` to all platforms it can support across its environments. +## Preview features +Pixi sometimes introduces new features that are not yet stable, but that we would like for users to test out. These features are called preview features. Preview features are disabled by default and can be enabled by setting the `preview` field in the project manifest. The preview field is an array of strings that specify the preview features to enable, or the boolean value `true` to enable all preview features. + +An example of a preview feature in the project manifest: + +```toml title="Example preview features in the project manifest" +[project] +name = "foo" +channels = [] +platforms = [] +preview = ["new-resolve"] +``` + +Preview features in the documentation will be marked as such on the relevant pages. + + ## Global configuration The global configuration options are documented in the [global configuration](../reference/pixi_configuration.md) section. diff --git a/schema/examples/valid/full.toml b/schema/examples/valid/full.toml index 515b953e9..6ad5e0c4c 100644 --- a/schema/examples/valid/full.toml +++ b/schema/examples/valid/full.toml @@ -12,6 +12,7 @@ license = "MIT" license-file = "LICENSE" name = "project" platforms = ["linux-64", "win-64", "osx-64", "osx-arm64"] +preview = ["new-resolve"] readme = "README.md" repository = "https://github.com/author/project" version = "0.1.0" diff --git a/schema/model.py b/schema/model.py index 2b0062703..5d5dc9586 100644 --- a/schema/model.py +++ b/schema/model.py @@ -88,6 +88,10 @@ class ChannelPriority(str, Enum): strict = "strict" +class KnownPreviewFeature(str, Enum): + """The preview features of the project.""" + + class Project(StrictBaseModel): """The project's metadata information.""" @@ -139,6 +143,9 @@ class Project(StrictBaseModel): pypi_options: PyPIOptions | None = Field( None, alias="pypi-options", description="Options related to PyPI indexes for this project" ) + preview: list[KnownPreviewFeature | str] | bool | None = Field( + None, alias="preview", description="Defines the enabling of preview features of the project" + ) ######################## diff --git a/schema/schema.json b/schema/schema.json index ec81a102b..5016438c2 100644 --- a/schema/schema.json +++ b/schema/schema.json @@ -773,6 +773,30 @@ "$ref": "#/$defs/Platform" } }, + "preview": { + "title": "Preview", + "description": "Defines the enabling of preview features of the project", + "anyOf": [ + { + "type": "array", + "items": { + "anyOf": [ + { + "title": "KnownPreviewFeature", + "description": "The preview features of the project.", + "enum": [] + }, + { + "type": "string" + } + ] + } + }, + { + "type": "boolean" + } + ] + }, "pypi-options": { "$ref": "#/$defs/PyPIOptions", "description": "Options related to PyPI indexes for this project" diff --git a/src/project/mod.rs b/src/project/mod.rs index 30d8e24da..d79167e77 100644 --- a/src/project/mod.rs +++ b/src/project/mod.rs @@ -32,8 +32,8 @@ use pixi_config::{Config, PinningStrategy}; use pixi_consts::consts; use pixi_manifest::{ pypi::PyPiPackageName, DependencyOverwriteBehavior, EnvironmentName, Environments, FeatureName, - FeaturesExt, HasFeaturesIter, HasManifestRef, Manifest, ParsedManifest, PypiDependencyLocation, - SpecType, + FeaturesExt, HasFeaturesIter, HasManifestRef, KnownPreviewFeature, Manifest, ParsedManifest, + PypiDependencyLocation, SpecType, }; use pixi_utils::reqwest::build_reqwest_clients; use pypi_mapping::{ChannelName, CustomMapping, MappingLocation, MappingSource}; @@ -962,6 +962,22 @@ impl Project { Ok(implicit_constraints) } + + /// Returns true if all preview features are enabled + pub fn all_preview_features_enabled(&self) -> bool { + self.manifest + .preview() + .map(|preview| preview.all_enabled()) + .unwrap_or(false) + } + + /// Returns true if the given preview feature is enabled + pub fn is_preview_feature_enabled(&self, feature: KnownPreviewFeature) -> bool { + self.manifest + .preview() + .map(|preview| preview.is_enabled(feature)) + .unwrap_or(false) + } } pub struct UpdateDeps {