Skip to content

Commit

Permalink
Merge branch 'main' into fix/pixi-build-trigger-rebuild
Browse files Browse the repository at this point in the history
  • Loading branch information
Hofer-Julian authored Dec 5, 2024
2 parents 7a9a850 + 2a1e115 commit 95f767b
Show file tree
Hide file tree
Showing 5 changed files with 200 additions and 2 deletions.
16 changes: 16 additions & 0 deletions crates/pixi_manifest/src/task.rs
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,22 @@ impl Task {
Task::Alias(_) => false,
}
}

/// Returns the inputs of the task.
pub fn inputs(&self) -> Option<&[String]> {
match self {
Task::Execute(exe) => exe.inputs.as_deref(),
_ => None,
}
}

/// Returns the outputs of the task.
pub fn outputs(&self) -> Option<&[String]> {
match self {
Task::Execute(exe) => exe.outputs.as_deref(),
_ => None,
}
}
}

/// A command script executes a single command from the environment
Expand Down
116 changes: 116 additions & 0 deletions src/cli/task.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ use pixi_manifest::task::{quote, Alias, CmdArgs, Execute, Task, TaskName};
use pixi_manifest::EnvironmentName;
use pixi_manifest::FeatureName;
use rattler_conda_types::Platform;
use serde::Serialize;
use serde_with::serde_as;
use std::collections::{BTreeMap, BTreeSet, HashMap, HashSet};
use std::error::Error;
use std::io;
Expand Down Expand Up @@ -135,6 +137,11 @@ pub struct ListArgs {
/// If not specified, the default environment is used.
#[arg(long, short)]
pub environment: Option<String>,

/// List as json instead of a tree
/// If not specified, the default environment is used.
#[arg(long)]
pub json: bool,
}

impl From<AddArgs> for Task {
Expand Down Expand Up @@ -368,6 +375,11 @@ pub fn execute(args: Args) -> miette::Result<()> {
);
}
Operation::List(args) => {
if args.json {
print_tasks_json(&project);
return Ok(());
}

let explicit_environment = args
.environment
.map(|n| EnvironmentName::from_str(n.as_str()))
Expand Down Expand Up @@ -439,3 +451,107 @@ pub fn execute(args: Args) -> miette::Result<()> {
Project::warn_on_discovered_from_env(args.project_config.manifest_path.as_deref());
Ok(())
}

fn print_tasks_json(project: &Project) {
let env_feature_task_map: Vec<EnvTasks> = build_env_feature_task_map(project);

let json_string =
serde_json::to_string_pretty(&env_feature_task_map).expect("Failed to serialize tasks");
println!("{}", json_string);
}

fn build_env_feature_task_map(project: &Project) -> Vec<EnvTasks> {
project
.environments()
.iter()
.sorted_by_key(|env| env.name().to_string())
.filter_map(|env: &Environment<'_>| {
if verify_current_platform_has_required_virtual_packages(env).is_err() {
return None;
}
Some(EnvTasks::from(env))
})
.collect()
}

#[derive(Serialize, Debug)]
struct EnvTasks {
environment: String,
features: Vec<SerializableFeature>,
}

impl From<&Environment<'_>> for EnvTasks {
fn from(env: &Environment) -> Self {
Self {
environment: env.name().to_string(),
features: env
.feature_tasks()
.iter()
.map(|(feature_name, task_map)| {
SerializableFeature::from((*feature_name, task_map))
})
.collect(),
}
}
}

#[derive(Serialize, Debug)]
struct SerializableFeature {
name: String,
tasks: Vec<SerializableTask>,
}

#[derive(Serialize, Debug)]
struct SerializableTask {
name: String,
#[serde(flatten)]
info: TaskInfo,
}

impl From<(&FeatureName, &HashMap<&TaskName, &Task>)> for SerializableFeature {
fn from((feature_name, task_map): (&FeatureName, &HashMap<&TaskName, &Task>)) -> Self {
Self {
name: feature_name.to_string(),
tasks: task_map
.iter()
.map(|(task_name, task)| SerializableTask {
name: task_name.to_string(),
info: TaskInfo::from(*task),
})
.collect(),
}
}
}

/// Collection of task properties for displaying in the UI.
#[serde_as]
#[derive(Serialize, Debug)]
pub struct TaskInfo {
cmd: Option<String>,
description: Option<String>,
depends_on: Vec<TaskName>,
cwd: Option<PathBuf>,
env: Option<IndexMap<String, String>>,
clean_env: bool,
inputs: Option<Vec<String>>,
outputs: Option<Vec<String>>,
}

impl From<&Task> for TaskInfo {
fn from(task: &Task) -> Self {
TaskInfo {
cmd: task.as_single_command().map(|cmd| cmd.to_string()),
description: task.description().map(|desc| desc.to_string()),
depends_on: task.depends_on().to_vec(),
cwd: task.working_directory().map(PathBuf::from),
env: task.env().cloned(),
clean_env: task.clean_env(),
inputs: task
.inputs()
.map(|inputs| inputs.iter().map(String::from).collect()),
outputs: task
.outputs()
.map(|outputs| outputs.iter().map(String::from).collect()),
}
}
}
6 changes: 6 additions & 0 deletions src/install_pypi/plan/test/harness.rs
Original file line number Diff line number Diff line change
Expand Up @@ -190,6 +190,11 @@ impl MockedSitePackages {
}
}

#[allow(dead_code)]
pub fn base_dir(&self) -> &Path {
self.fake_site_packages.path()
}

/// Create INSTALLER and METADATA files for the installed dist
/// these are checked for the installer and requires python
fn create_file_backing(
Expand Down Expand Up @@ -516,6 +521,7 @@ pub fn fake_pyproject_toml(
// Set the modification time if it is provided
if let Some(modification_time) = modification_time {
pyproject_toml.set_modified(modification_time).unwrap();
pyproject_toml.sync_all().unwrap();
}
(temp_dir, pyproject_toml)
}
44 changes: 42 additions & 2 deletions src/install_pypi/plan/test/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -320,8 +320,7 @@ fn test_local_source_newer_than_local_metadata() {
.unwrap();
pyproject.sync_all().unwrap();

// pyproject.toml file is older than the cache, all else is the same
// so we do not expect a re-installation
// We expect a reinstall, because the pyproject.toml file is newer than the cache
let plan = harness::install_planner();
let install_plan = plan
.plan(&site_packages, NoCache, &required.to_borrowed())
Expand All @@ -332,6 +331,47 @@ fn test_local_source_newer_than_local_metadata() {
);
}

#[test]
fn test_local_source_older_than_local_metadata() {
let (fake, pyproject) = harness::fake_pyproject_toml(Some(
std::time::SystemTime::now() - std::time::Duration::from_secs(60 * 60 * 24),
));
let site_packages = MockedSitePackages::new().add_directory(
"aiofiles",
"0.6.0",
fake.path().to_path_buf(),
false,
// Set the metadata mtime to now explicitly
InstalledDistOptions::default().with_metadata_mtime(std::time::SystemTime::now()),
);
// Requires following package
let required = RequiredPackages::new().add_directory(
"aiofiles",
"0.6.0",
fake.path().to_path_buf(),
false,
);

let dist_info = site_packages
.base_dir()
.join(format!("{}-{}.dist-info", "aiofiles", "0.6.0"))
.join("METADATA");
// Sanity check that these timestamps are different
assert_ne!(
pyproject.metadata().unwrap().modified().unwrap(),
dist_info.metadata().unwrap().modified().unwrap()
);

// Install plan should not reinstall anything
let plan = harness::install_planner();
let install_plan = plan
.plan(&site_packages, NoCache, &required.to_borrowed())
.expect("should install");
assert_eq!(install_plan.reinstalls.len(), 0);
assert_eq!(install_plan.local.len(), 0);
assert_eq!(install_plan.remote.len(), 0);
}

/// When we have an editable package installed and we require a non-editable package
/// we should reinstall the non-editable package
#[test]
Expand Down
20 changes: 20 additions & 0 deletions src/project/environment.rs
Original file line number Diff line number Diff line change
Expand Up @@ -227,6 +227,26 @@ impl<'p> Environment<'p> {
}
}

/// Returns a map of all the features and their tasks for this environment.
///
/// Resolves for the best platform target.
pub(crate) fn feature_tasks(
&self,
) -> HashMap<&'p FeatureName, HashMap<&'p TaskName, &'p Task>> {
self.features()
.map(|feature| {
(
&feature.name,
feature
.targets
.resolve(Some(self.best_platform()))
.flat_map(|target| target.tasks.iter())
.collect::<HashMap<_, _>>(),
)
})
.collect()
}

/// Returns the system requirements for this environment.
///
/// The system requirements of the environment are the union of the system
Expand Down

0 comments on commit 95f767b

Please sign in to comment.