Skip to content

Commit

Permalink
Merge branch 'prefix-dev:main' into cont_on_fail
Browse files Browse the repository at this point in the history
  • Loading branch information
KGB99 authored Aug 20, 2024
2 parents d57b711 + a28628b commit e0ff9b0
Show file tree
Hide file tree
Showing 18 changed files with 223 additions and 145 deletions.
2 changes: 1 addition & 1 deletion Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ members = ["rust-tests"]

[package]
name = "rattler-build"
version = "0.19.0"
version = "0.20.0"
authors = ["rattler-build contributors <[email protected]>"]
homepage = "https://github.com/prefix-dev/rattler-build"
edition = "2021"
Expand Down
38 changes: 31 additions & 7 deletions docs/reference/recipe_file.md
Original file line number Diff line number Diff line change
Expand Up @@ -553,25 +553,49 @@ glibc version is too old.
### Run exports

Packages may have runtime requirements such as shared libraries (e.g. `zlib`), which are required for linking at build time, and for resolving the link at run time.
Such packages use `run_exports` for defining the runtime requirements to let the dependent packages understand the runtime requirements of the package.
With `run_exports` packages runtime requirements can be implicitly added.
`run_exports` are weak by default, these two requirements for the `zlib` package are therefore equivalent:

Example from `zlib`:

```yaml
```yaml title="recipe.yaml for zlib"
requirements:
run_exports:
- ${{ pin_subpackage('libzlib', exact=True) }}
```

Run exports are weak by default. But you can also define strong `run_exports`.
```yaml title="recipe.yaml for zlib"
requirements:
run_exports:
weak:
- ${{ pin_subpackage('libzlib', exact=True) }}
```

```yaml
The alternative to `weak` is `strong`.
For `gcc` this would look like this:

```yaml title="recipe.yaml for gcc"
requirements:
run_exports:
strong:
- ${{ pin_subpackage('libzlib', exact=True) }}
- ${{ pin_subpackage('libgcc', exact=True) }}
```

`weak` exports will only be implicitly added as runtime requirement, if the package is a host dependency.
`strong` exports will be added for both build and host dependencies.
In the following example you can see the implicitly added runtime dependencies.

```yaml title="recipe.yaml of some package using gcc and zlib"
requirements:
build:
- gcc # has a strong run export
host:
- zlib # has a (weak) run export
# - libgcc <-- implicitly added by gcc
run:
# - libgcc <-- implicitly added by gcc
# - libzlib <-- implicitly added by libzlib
```


### Ignore run exports

There maybe cases where an upstream package has a problematic `run_exports` constraint.
Expand Down
2 changes: 1 addition & 1 deletion src/bin/generate-cli-docs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ fn main() {
let target_default_platform = format!("Default value: `{}`", Platform::current());
let help = help.replace(
target_default_platform.as_str(),
format!("Default value: current platform").as_str(),
"Default value: current platform",
);

print!("{}", help);
Expand Down
41 changes: 18 additions & 23 deletions src/build.rs
Original file line number Diff line number Diff line change
@@ -1,19 +1,19 @@
//! The build module contains the code for running the build process for a given [`Output`]
use rattler_conda_types::{Channel, MatchSpec, ParseStrictness};
use std::path::PathBuf;
use std::vec;
//! The build module contains the code for running the build process for a given
//! [`Output`]
use std::{path::PathBuf, vec};

use miette::IntoDiagnostic;
use rattler_conda_types::{Channel, MatchSpec, ParseStrictness};
use rattler_index::index;
use rattler_solve::{ChannelPriority, SolveStrategy};

use crate::metadata::Output;
use crate::package_test::TestConfiguration;
use crate::recipe::parser::TestType;
use crate::render::solver::load_repodatas;
use crate::{package_test, tool_configuration};
use crate::{
metadata::Output, package_test, package_test::TestConfiguration, recipe::parser::TestType,
render::solver::load_repodatas, tool_configuration,
};

/// Check if the build should be skipped because it already exists in any of the channels
/// Check if the build should be skipped because it already exists in any of the
/// channels
pub async fn skip_existing(
mut outputs: Vec<Output>,
tool_configuration: &tool_configuration::Configuration,
Expand Down Expand Up @@ -78,38 +78,32 @@ pub async fn skip_existing(
"{}-{}-{}",
output.name().as_normalized(),
output.version(),
output.build_string().unwrap_or_default()
&output.build_string()
));
if exists {
// The identifier should always be set at this point
tracing::info!(
"Skipping build for {}",
output.identifier().as_deref().unwrap_or("unknown")
);
tracing::info!("Skipping build for {}", output.identifier());
}
!exists
});

Ok(outputs)
}

/// Run the build for the given output. This will fetch the sources, resolve the dependencies,
/// and execute the build script. Returns the path to the resulting package.
/// Run the build for the given output. This will fetch the sources, resolve the
/// dependencies, and execute the build script. Returns the path to the
/// resulting package.
pub async fn run_build(
output: Output,
tool_configuration: &tool_configuration::Configuration,
) -> miette::Result<(Output, PathBuf)> {
if output.build_string().is_none() {
miette::bail!("Build string is not set for {:?}", output.name());
}

output
.build_configuration
.directories
.create_build_dir()
.into_diagnostic()?;

let span = tracing::info_span!("Running build for", recipe = output.identifier().unwrap());
let span = tracing::info_span!("Running build for", recipe = output.identifier());
let _enter = span.enter();
output.record_build_start();

Expand Down Expand Up @@ -148,7 +142,8 @@ pub async fn run_build(

// We run all the package content tests
for test in output.recipe.tests() {
// TODO we could also run each of the (potentially multiple) test scripts and collect the errors
// TODO we could also run each of the (potentially multiple) test scripts and
// collect the errors
if let TestType::PackageContents { package_contents } = test {
package_contents
.run_test(&paths_json, &output.build_configuration.target_platform)
Expand Down
4 changes: 2 additions & 2 deletions src/env_vars.rs
Original file line number Diff line number Diff line change
Expand Up @@ -292,8 +292,8 @@ pub fn vars(output: &Output, build_state: &str) -> HashMap<String, String> {
.recipe
.build()
.string()
.map(|s| s.to_string())
.unwrap_or_else(|| hash.to_string())
.resolve(&hash, output.recipe.build().number)
.into_owned()
);
insert!(vars, "PKG_HASH", hash);

Expand Down
4 changes: 2 additions & 2 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -271,8 +271,8 @@ pub async fn get_build_output(
build_string: recipe
.build()
.string()
.expect("Shouldn't be unset, needs major refactoring, for handling this better")
.to_owned(),
.resolve(&hash, recipe.build().number())
.into_owned(),
},
);

Expand Down
72 changes: 41 additions & 31 deletions src/metadata.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
//! All the metadata that makes up a recipe file
use std::{
borrow::Cow,
collections::BTreeMap,
fmt::{self, Display, Formatter},
io::Write,
Expand Down Expand Up @@ -72,7 +73,8 @@ pub struct Directories {
/// Exposed as `$PREFIX` (or `%PREFIX%` on Windows) in the build script
pub host_prefix: PathBuf,
/// The build prefix is the directory where build dependencies are installed
/// Exposed as `$BUILD_PREFIX` (or `%BUILD_PREFIX%` on Windows) in the build script
/// Exposed as `$BUILD_PREFIX` (or `%BUILD_PREFIX%` on Windows) in the build
/// script
pub build_prefix: PathBuf,
/// The work directory is the directory where the source code is copied to
pub work_dir: PathBuf,
Expand Down Expand Up @@ -210,7 +212,8 @@ fn default_true() -> bool {
pub struct PackagingSettings {
/// The archive type, currently supported are `tar.bz2` and `conda`
pub archive_type: ArchiveType,
/// The compression level from 1-9 or -7-22 for `tar.bz2` and `conda` archives
/// The compression level from 1-9 or -7-22 for `tar.bz2` and `conda`
/// archives
pub compression_level: i32,
}

Expand All @@ -235,7 +238,8 @@ impl PackagingSettings {
pub struct BuildConfiguration {
/// The target platform for the build
pub target_platform: Platform,
/// The host platform (usually target platform, but for `noarch` it's the build platform)
/// The host platform (usually target platform, but for `noarch` it's the
/// build platform)
pub host_platform: Platform,
/// The build platform (the platform that the build is running on)
pub build_platform: Platform,
Expand All @@ -253,14 +257,17 @@ pub struct BuildConfiguration {
pub solve_strategy: SolveStrategy,
/// The timestamp to use for the build
pub timestamp: chrono::DateTime<chrono::Utc>,
/// All subpackages coming from this output or other outputs from the same recipe
/// All subpackages coming from this output or other outputs from the same
/// recipe
pub subpackages: BTreeMap<PackageName, PackageIdentifier>,
/// Package format (.tar.bz2 or .conda)
pub packaging_settings: PackagingSettings,
/// Whether to store the recipe and build instructions in the final package or not
/// Whether to store the recipe and build instructions in the final package
/// or not
#[serde(skip_serializing, default = "default_true")]
pub store_recipe: bool,
/// Whether to set additional environment variables to force colors in the build script or not
/// Whether to set additional environment variables to force colors in the
/// build script or not
#[serde(skip_serializing, default = "default_true")]
pub force_colors: bool,
}
Expand Down Expand Up @@ -301,21 +308,24 @@ pub struct BuildSummary {
pub failed: bool,
}

/// A output. This is the central element that is passed to the `run_build` function
/// and fully specifies all the options and settings to run the build.
/// A output. This is the central element that is passed to the `run_build`
/// function and fully specifies all the options and settings to run the build.
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct Output {
/// The rendered recipe that is used to build this output
pub recipe: Recipe,
/// The build configuration for this output (e.g. target_platform, channels, and other settings)
/// The build configuration for this output (e.g. target_platform, channels,
/// and other settings)
pub build_configuration: BuildConfiguration,
/// The finalized dependencies for this output. If this is `None`, the dependencies have not been resolved yet.
/// During the `run_build` functions, the dependencies are resolved and this field is filled.
/// The finalized dependencies for this output. If this is `None`, the
/// dependencies have not been resolved yet. During the `run_build`
/// functions, the dependencies are resolved and this field is filled.
pub finalized_dependencies: Option<FinalizedDependencies>,
/// The finalized dependencies from the cache (if there is a cache instruction)
/// The finalized dependencies from the cache (if there is a cache
/// instruction)
pub finalized_cache_dependencies: Option<FinalizedDependencies>,
/// The finalized sources for this output. Contain the exact git hashes for the sources that are used
/// to build this output.
/// The finalized sources for this output. Contain the exact git hashes for
/// the sources that are used to build this output.
pub finalized_sources: Option<Vec<Source>>,

/// Summary of the build
Expand All @@ -341,9 +351,13 @@ impl Output {
self.recipe.package().version()
}

/// The build string is usually set automatically as the hash of the variant configuration.
pub fn build_string(&self) -> Option<&str> {
self.recipe.build().string()
/// The build string is either the build string from the recipe or computed
/// from the hash and build number.
pub fn build_string(&self) -> Cow<'_, str> {
self.recipe
.build()
.string
.resolve(&self.build_configuration.hash, self.recipe.build().number)
}

/// The channels to use when resolving dependencies
Expand All @@ -358,13 +372,13 @@ impl Output {
}

/// retrieve an identifier for this output ({name}-{version}-{build_string})
pub fn identifier(&self) -> Option<String> {
Some(format!(
pub fn identifier(&self) -> String {
format!(
"{}-{}-{}",
self.name().as_normalized(),
self.version(),
self.build_string()?
))
&self.build_string()
)
}

/// Record a warning during the build
Expand Down Expand Up @@ -420,7 +434,8 @@ impl Output {
}

/// Search for the resolved package with the given name in the host prefix
/// Returns a tuple of the package and a boolean indicating whether the package is directly requested
/// Returns a tuple of the package and a boolean indicating whether the
/// package is directly requested
pub fn find_resolved_package(&self, name: &str) -> Option<(&RepoDataRecord, bool)> {
let host = self.finalized_dependencies.as_ref()?.host.as_ref()?;
let record = host
Expand All @@ -442,7 +457,7 @@ impl Output {
/// Print a nice summary of the build
pub fn log_build_summary(&self) -> Result<(), std::io::Error> {
let summary = self.build_summary.lock().unwrap();
let identifier = self.identifier().unwrap_or_default();
let identifier = self.identifier();
let span = tracing::info_span!("Build summary for", recipe = identifier);
let _enter = span.enter();

Expand Down Expand Up @@ -574,11 +589,7 @@ impl Output {
table
};

writeln!(
f,
"Variant configuration (hash: {}):",
self.build_string().unwrap_or_default()
)?;
writeln!(f, "Variant configuration (hash: {}):", self.build_string())?;
let mut table = template();
if table_format != comfy_table::presets::UTF8_FULL {
table.set_header(vec!["Key", "Value"]);
Expand Down Expand Up @@ -642,7 +653,6 @@ mod tests {

#[cfg(test)]
mod test {
use rstest::*;
use std::str::FromStr;

use chrono::TimeZone;
Expand All @@ -652,11 +662,11 @@ mod test {
VersionWithSource,
};
use rattler_digest::{parse_digest_from_hex, Md5, Sha256};
use rstest::*;
use url::Url;

use crate::render::resolved_dependencies::{self, SourceDependency};

use super::{Directories, Output};
use crate::render::resolved_dependencies::{self, SourceDependency};

#[test]
fn test_directories_yaml_rendering() {
Expand Down
3 changes: 2 additions & 1 deletion src/opt.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ use url::Url;

/// Application subcommands.
#[derive(Parser)]
#[allow(clippy::large_enum_variant)]
pub enum SubCommands {
/// Build a package from a recipe
Build(BuildOpts),
Expand Down Expand Up @@ -363,7 +364,7 @@ fn is_dir(dir: &str) -> Result<PathBuf, String> {
/// Parse a single key-value pair
fn parse_key_val(s: &str) -> Result<(String, Value), Box<dyn Error + Send + Sync + 'static>> {
let (key, value) = s
.split_once("=")
.split_once('=')
.ok_or_else(|| format!("invalid KEY=value: no `=` found in `{}`", s))?;
Ok((key.to_string(), json!(value)))
}
Expand Down
Loading

0 comments on commit e0ff9b0

Please sign in to comment.