Skip to content

Commit

Permalink
deprecate release.toml
Browse files Browse the repository at this point in the history
When loading a project, issue a deprecation warning if Release.toml is
found. If it is found, ensure that its version field is aligned with
Twoliter.toml
  • Loading branch information
webern committed Dec 1, 2023
1 parent 2ef4226 commit 16775d6
Showing 1 changed file with 86 additions and 13 deletions.
99 changes: 86 additions & 13 deletions twoliter/src/project.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,13 @@ use crate::docker::ImageArchUri;
use crate::schema_version::SchemaVersion;
use anyhow::{ensure, Context, Result};
use async_recursion::async_recursion;
use log::{debug, trace};
use log::{debug, info, trace, warn};
use non_empty_string::NonEmptyString;
use serde::{Deserialize, Serialize};
use sha2::{Digest, Sha512};
use std::path::{Path, PathBuf};
use tokio::fs;
use toml::Table;

/// Common functionality in commands, if the user gave a path to the `Twoliter.toml` file,
/// we use it, otherwise we search for the file. Returns the `Project` and the path at which it was
Expand All @@ -25,7 +26,7 @@ pub(crate) async fn load_or_find_project(user_path: Option<PathBuf>) -> Result<P
}

/// Represents the structure of a `Twoliter.toml` project file.
#[derive(Debug, Clone, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize)]
#[derive(Debug, Clone, Eq, PartialEq, Ord, PartialOrd, Serialize)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct Project {
#[serde(skip)]
Expand Down Expand Up @@ -53,20 +54,11 @@ impl Project {
let data = fs::read_to_string(&path)
.await
.context(format!("Unable to read project file '{}'", path.display()))?;
let mut project: Self = toml::from_str(&data).context(format!(
let unvalidated: UnvalidatedProject = toml::from_str(&data).context(format!(
"Unable to deserialize project file '{}'",
path.display()
))?;
project.filepath = path.into();
project.project_dir = project
.filepath
.parent()
.context(format!(
"Unable to find the parent directory of '{}'",
project.filepath.display(),
))?
.into();
Ok(project)
unvalidated.validate(path).await
}

/// Recursively search for a file named `Twoliter.toml` starting in `dir`. If it is not found,
Expand Down Expand Up @@ -163,6 +155,87 @@ impl ImageName {
}
}

/// This is used to `Deserialize` a project, then run validation code before returning a valid
/// [`Project`]. See [this serde issue] to understand why this is necessary. See [`Project`] for
/// field documentation.
#[derive(Debug, Clone, Deserialize)]
#[serde(rename_all = "kebab-case")]
struct UnvalidatedProject {
schema_version: SchemaVersion<1>,
release_version: String,
sdk: Option<ImageName>,
toolchain: Option<ImageName>,
}

impl UnvalidatedProject {
/// Constructs a [`Project`] from an [`UnvalidatedProject`] after validating fields.
async fn validate(self, path: impl AsRef<Path>) -> Result<Project> {
let filepath: PathBuf = path.as_ref().into();
let project_dir = filepath
.parent()
.context(format!(
"Unable to find the parent directory of '{}'",
filepath.display(),
))?
.to_path_buf();

self.check_release_toml(&project_dir).await?;

Ok(Project {
filepath,
project_dir,
schema_version: self.schema_version,
release_version: self.release_version,
sdk: self.sdk,
toolchain: self.toolchain,
})
}

/// Issues a warning if `Release.toml` is found and, if so, ensures that it contains the same
/// version (i.e. `release-version`) as the `Twoliter.toml` project file.
async fn check_release_toml(&self, project_dir: &Path) -> Result<()> {
let path = project_dir.join("Release.toml");
if !path.exists() || !path.is_file() {
// There is no Release.toml file. This is a good thing!
trace!("This project does not have a Release.toml file (this is not a problem)");
return Ok(());
}
warn!(
"A Release.toml file was found. Release.toml is deprecated. Please remove it from \
your project."
);
let content = fs::read_to_string(&path).await.context(format!(
"Error while checking Release.toml file at '{}'",
path.display()
))?;
let toml: Table = match toml::from_str(&content) {
Ok(toml) => toml,
Err(e) => {
warn!(
"Unable to parse Release.toml to ensure that its version matches the \
release-version in Twoliter.toml: {e}",
);
return Ok(());
}
};
let version = match toml.get("version") {
Some(version) => version,
None => {
info!("Release.toml does not contain a version key. Ignoring it.");
return Ok(());
}
}
.to_string();
ensure!(
version == self.release_version,
"The version found in Release.toml, '{version}', does not match the release-version \
found in Twoliter.toml '{}'",
self.release_version
);
Ok(())
}
}

#[cfg(test)]
mod test {
use super::*;
Expand Down

0 comments on commit 16775d6

Please sign in to comment.