Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Cached recipes #1906

Draft
wants to merge 36 commits into
base: master
Choose a base branch
from
Draft
Changes from 1 commit
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
a649e1d
Add ExpressionWalker and refactor Variables
nmay231 Feb 13, 2024
47976a7
Cache recipe by evaluated body, initial
nmay231 Feb 15, 2024
7d24508
Tests
nmay231 Feb 16, 2024
626dd08
Include working directory/project dir in serialized cache
nmay231 Feb 16, 2024
ed9f0aa
Rename RecipeCache.{hash => body_hash}
nmay231 Feb 16, 2024
a1f740c
Shorten cache filename
nmay231 Feb 16, 2024
d885519
Rename JustfileCache.{recipe_caches => recipes}
nmay231 Feb 16, 2024
5a959d1
Use working directory AND justfile path for cache filename
nmay231 Feb 16, 2024
d82e9e1
Remove InvalidCachedRecipe error
nmay231 Mar 7, 2024
c18a0cc
Print 'skipping cached recipe' message by default
nmay231 Mar 7, 2024
6307eb0
Remove `CacheFilename` for now
nmay231 Mar 7, 2024
ed8c05d
Use .as_encoded_bytes() on os strings
nmay231 Mar 7, 2024
bbf7f84
Fix tests
nmay231 Mar 7, 2024
cf3c95a
Make cached_recipes require unstable
nmay231 Mar 7, 2024
36482fd
Put cache files in working directory by default
nmay231 Mar 7, 2024
d5edf65
Added versioning to cache files
nmay231 Mar 7, 2024
f32cc86
Make errors look the same
nmay231 Mar 7, 2024
219c646
Revert "Add ExpressionWalker and refactor Variables"
nmay231 Mar 7, 2024
3a341ea
Clarify error messages
nmay231 Mar 7, 2024
9acd827
Incompatible old versions of cache should also silently be ignored
nmay231 Mar 7, 2024
29d3459
Merge branch 'master' into cached-recipes
nmay231 Mar 12, 2024
0812866
Doc comments
nmay231 Mar 14, 2024
707844c
Move cache_file into Search struct
nmay231 Mar 14, 2024
70fe6ac
Starting to implement cached dependencies
nmay231 Mar 21, 2024
e014285
Return instead of mutate
nmay231 Mar 21, 2024
e9686df
Cached dependencies mostly work
nmay231 Mar 24, 2024
e859bd6
Only impl Recipe::run() for recipes with resolved dependencies
nmay231 Mar 26, 2024
fbc87dd
Finalize cached recipes with deps
nmay231 Mar 26, 2024
8e8e046
Merge branch 'master' into cached-recipes
nmay231 Mar 26, 2024
4961730
Rename src/{cache => justfile_cache}.rs
nmay231 Mar 26, 2024
a17b8dd
Simplify JustfileCache
nmay231 Mar 26, 2024
d12dbc3
Docs
nmay231 Mar 26, 2024
1ca5122
note to add to gitignore
nmay231 Mar 26, 2024
db89a9a
Just make the cache mutable directly, don't use refcell
nmay231 Apr 15, 2024
aa8c2db
Run just check
nmay231 Apr 15, 2024
e666bf5
Merge branch 'master' into cached-recipes
nmay231 Apr 15, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Finalize cached recipes with deps
nmay231 committed Mar 26, 2024

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature. The key has expired.
commit fbc87ddaebc10c9be85bef9ab92c60a774d1b403
10 changes: 9 additions & 1 deletion src/cache.rs
Original file line number Diff line number Diff line change
@@ -30,6 +30,8 @@ pub(crate) struct JustfileCache {
#[derive(Debug, Serialize, Deserialize)]
pub(crate) struct RecipeCache {
pub(crate) body_hash: String,
#[serde(skip)]
pub(crate) hash_changed: bool,
}

impl JustfileCache {
@@ -62,7 +64,13 @@ impl JustfileCache {
}

pub(crate) fn insert_recipe(&mut self, name: String, body_hash: String) {
self.recipes.insert(name, RecipeCache { body_hash });
self.recipes.insert(
name,
RecipeCache {
body_hash,
hash_changed: true,
},
);
}

pub(crate) fn save<'run>(self, search: &Search) -> RunResult<'run, ()> {
32 changes: 13 additions & 19 deletions src/justfile.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use {super::*, serde::Serialize};
use {super::*, serde::Serialize, std::cell::RefCell};

#[derive(Debug)]
struct Invocation<'src: 'run, 'run> {
@@ -272,8 +272,7 @@ impl<'src> Justfile<'src> {
}

let mut ran = Ran::default();
let mut cache = JustfileCache::new(search)?;
let mut recipes_hashes = HashMap::new();
let cache = RefCell::new(JustfileCache::new(search)?);

for invocation in invocations {
let context = RecipeContext {
@@ -284,7 +283,7 @@ impl<'src> Justfile<'src> {
cache: &cache,
};

let new_hashes = Self::run_recipe(
Self::run_recipe(
&invocation
.arguments
.iter()
@@ -297,14 +296,9 @@ impl<'src> Justfile<'src> {
invocation.recipe,
search,
)?;
recipes_hashes.extend(new_hashes);
}

for (name, body_hash) in recipes_hashes {
cache.insert_recipe(name, body_hash);
}

cache.save(search)
cache.into_inner().save(search)
}

pub(crate) fn get_alias(&self, name: &str) -> Option<&Alias<'src>> {
@@ -417,10 +411,9 @@ impl<'src> Justfile<'src> {
ran: &mut Ran<'src>,
recipe: &Recipe<'src>,
search: &Search,
) -> RunResult<'src, HashMap<String, String>> {
let mut recipe_hashes = HashMap::new();
) -> RunResult<'src, ()> {
if ran.has_run(&recipe.namepath, arguments) {
return Ok(recipe_hashes);
return Ok(());
}

if !context.config.yes && !recipe.confirm()? {
@@ -451,16 +444,18 @@ impl<'src> Justfile<'src> {
.map(|argument| evaluator.evaluate_expression(argument))
.collect::<RunResult<Vec<String>>>()?;

let new_hashes = Self::run_recipe(&arguments, context, dotenv, ran, recipe, search)?;
recipe_hashes.extend(new_hashes);
Self::run_recipe(&arguments, context, dotenv, ran, recipe, search)?;
}
}

let updated_hash = recipe.run(context, dotenv, scope.child(), search, &positional)?;
let recipe_hash_changed = updated_hash.is_some();

if let Some(body_hash) = updated_hash {
recipe_hashes.insert(recipe.name.to_string(), body_hash);
context
.cache
.borrow_mut()
.insert_recipe(recipe.name.to_string(), body_hash);
}

if !context.config.no_dependencies && (!recipe.should_cache() || recipe_hash_changed) {
@@ -473,13 +468,12 @@ impl<'src> Justfile<'src> {
evaluated.push(evaluator.evaluate_expression(argument)?);
}

let new_hashes = Self::run_recipe(&evaluated, context, dotenv, &mut ran, recipe, search)?;
recipe_hashes.extend(new_hashes);
Self::run_recipe(&evaluated, context, dotenv, &mut ran, recipe, search)?;
}
}

ran.ran(&recipe.namepath, arguments.to_vec());
Ok(recipe_hashes)
Ok(())
}

pub(crate) fn public_recipes(&self, source_order: bool) -> Vec<&Recipe<'src, Dependency>> {
32 changes: 26 additions & 6 deletions src/recipe.rs
Original file line number Diff line number Diff line change
@@ -160,15 +160,35 @@ impl<'src> Recipe<'src, Dependency<'src>> {
recipe_hash.update(evaluator.evaluate_line(line, false)?.as_bytes());
}
let recipe_hash = recipe_hash.finalize().to_hex();
let recipes = &context.cache.borrow().recipes;

let recipes = &context.cache.recipes;
let updated_hash = recipes.get(self.name()).map_or_else(
|| Some(recipe_hash.to_string()),
recipes.get(self.name()).map_or_else(
|| Ok(Some(recipe_hash.to_string())),
|previous_run| {
(previous_run.body_hash != recipe_hash.as_str()).then(|| recipe_hash.to_string())
let have_deps_changed = self
.dependencies
.iter()
.take(self.priors)
.map(|dep| {
recipes
.get(dep.recipe.name())
.and_then(|previous_run| Some(previous_run.hash_changed))
.ok_or_else(|| {
Error::internal(format!(
"prior dependency `{}` did not run before current recipe `{}`",
dep.recipe.name, self.name
))
})
})
.collect::<Result<Vec<bool>, Error>>()?;

if have_deps_changed.iter().any(|x| *x) || previous_run.body_hash != recipe_hash.as_str() {
Ok(Some(recipe_hash.to_string()))
} else {
Ok(None)
}
},
);
Ok(updated_hash)
)
}

pub(crate) fn run<'run>(
4 changes: 3 additions & 1 deletion src/recipe_context.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
use std::cell::RefCell;

use super::*;

pub(crate) struct RecipeContext<'src: 'run, 'run> {
pub(crate) cache: &'run JustfileCache,
pub(crate) cache: &'run RefCell<JustfileCache>,
pub(crate) config: &'run Config,
pub(crate) scope: &'run Scope<'src, 'run>,
pub(crate) search: &'run Search,