diff --git a/README.md b/README.md
index cbfaa26b06..1b5adc47f2 100644
--- a/README.md
+++ b/README.md
@@ -1420,6 +1420,18 @@ script:
./{{justfile_directory()}}/scripts/some_script
```
+#### Source and Source Directory
+
+- `source()`master - Retrieves the path of the current source file.
+
+- `source_directory()`master - Retrieves the path of the parent directory of the
+ current source file.
+
+`source()` and `source_directory()` behave the same as `justfile()` and
+`justfile_directory()` in the root `justfile`, but will return the path and
+directory, respectively, of the current `import` or `mod` source file when
+called from within an import or submodule.
+
#### Just Executable
- `just_executable()` - Absolute path to the `just` executable.
diff --git a/src/evaluator.rs b/src/evaluator.rs
index 0ccafb987f..2794b46751 100644
--- a/src/evaluator.rs
+++ b/src/evaluator.rs
@@ -68,25 +68,13 @@ impl<'src, 'run> Evaluator<'src, 'run> {
Expression::Call { thunk } => {
use Thunk::*;
- match thunk {
- Nullary { name, function, .. } => function(self).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- }),
- Unary {
- name,
- function,
- arg,
- ..
- } => {
+ let result = match thunk {
+ Nullary { function, .. } => function(function::Context::new(self, thunk.name())),
+ Unary { function, arg, .. } => {
let arg = self.evaluate_expression(arg)?;
- function(self, &arg).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(function::Context::new(self, thunk.name()), &arg)
}
UnaryOpt {
- name,
function,
args: (a, b),
..
@@ -97,13 +85,9 @@ impl<'src, 'run> Evaluator<'src, 'run> {
None => None,
};
- function(self, &a, b.as_deref()).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(function::Context::new(self, thunk.name()), &a, b.as_deref())
}
UnaryPlus {
- name,
function,
args: (a, rest),
..
@@ -113,26 +97,22 @@ impl<'src, 'run> Evaluator<'src, 'run> {
for arg in rest {
rest_evaluated.push(self.evaluate_expression(arg)?);
}
- function(self, &a, &rest_evaluated).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(
+ function::Context::new(self, thunk.name()),
+ &a,
+ &rest_evaluated,
+ )
}
Binary {
- name,
function,
args: [a, b],
..
} => {
let a = self.evaluate_expression(a)?;
let b = self.evaluate_expression(b)?;
- function(self, &a, &b).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(function::Context::new(self, thunk.name()), &a, &b)
}
BinaryPlus {
- name,
function,
args: ([a, b], rest),
..
@@ -143,13 +123,14 @@ impl<'src, 'run> Evaluator<'src, 'run> {
for arg in rest {
rest_evaluated.push(self.evaluate_expression(arg)?);
}
- function(self, &a, &b, &rest_evaluated).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(
+ function::Context::new(self, thunk.name()),
+ &a,
+ &b,
+ &rest_evaluated,
+ )
}
Ternary {
- name,
function,
args: [a, b, c],
..
@@ -157,12 +138,14 @@ impl<'src, 'run> Evaluator<'src, 'run> {
let a = self.evaluate_expression(a)?;
let b = self.evaluate_expression(b)?;
let c = self.evaluate_expression(c)?;
- function(self, &a, &b, &c).map_err(|message| Error::FunctionCall {
- function: *name,
- message,
- })
+ function(function::Context::new(self, thunk.name()), &a, &b, &c)
}
- }
+ };
+
+ result.map_err(|message| Error::FunctionCall {
+ function: thunk.name(),
+ message,
+ })
}
Expression::StringLiteral { string_literal } => Ok(string_literal.cooked.clone()),
Expression::Backtick { contents, token } => {
diff --git a/src/function.rs b/src/function.rs
index 0321508ac8..52fcf00e19 100644
--- a/src/function.rs
+++ b/src/function.rs
@@ -11,13 +11,24 @@ use {
};
pub(crate) enum Function {
- Nullary(fn(&Evaluator) -> Result),
- Unary(fn(&Evaluator, &str) -> Result),
- UnaryOpt(fn(&Evaluator, &str, Option<&str>) -> Result),
- UnaryPlus(fn(&Evaluator, &str, &[String]) -> Result),
- Binary(fn(&Evaluator, &str, &str) -> Result),
- BinaryPlus(fn(&Evaluator, &str, &str, &[String]) -> Result),
- Ternary(fn(&Evaluator, &str, &str, &str) -> Result),
+ Nullary(fn(Context) -> Result),
+ Unary(fn(Context, &str) -> Result),
+ UnaryOpt(fn(Context, &str, Option<&str>) -> Result),
+ UnaryPlus(fn(Context, &str, &[String]) -> Result),
+ Binary(fn(Context, &str, &str) -> Result),
+ BinaryPlus(fn(Context, &str, &str, &[String]) -> Result),
+ Ternary(fn(Context, &str, &str, &str) -> Result),
+}
+
+pub(crate) struct Context<'src: 'run, 'run> {
+ pub(crate) evaluator: &'run Evaluator<'src, 'run>,
+ pub(crate) name: Name<'src>,
+}
+
+impl<'src: 'run, 'run> Context<'src, 'run> {
+ pub(crate) fn new(evaluator: &'run Evaluator<'src, 'run>, name: Name<'src>) -> Self {
+ Self { evaluator, name }
+ }
}
pub(crate) fn get(name: &str) -> Option {
@@ -72,6 +83,8 @@ pub(crate) fn get(name: &str) -> Option {
"shoutykebabcase" => Unary(shoutykebabcase),
"shoutysnakecase" => Unary(shoutysnakecase),
"snakecase" => Unary(snakecase),
+ "source_directory" => Nullary(source_directory),
+ "source_file" => Nullary(source_file),
"titlecase" => Unary(titlecase),
"trim" => Unary(trim),
"trim_end" => Unary(trim_end),
@@ -103,18 +116,23 @@ impl Function {
}
}
-fn absolute_path(evaluator: &Evaluator, path: &str) -> Result {
- let abs_path_unchecked = evaluator.search.working_directory.join(path).lexiclean();
+fn absolute_path(context: Context, path: &str) -> Result {
+ let abs_path_unchecked = context
+ .evaluator
+ .search
+ .working_directory
+ .join(path)
+ .lexiclean();
match abs_path_unchecked.to_str() {
Some(absolute_path) => Ok(absolute_path.to_owned()),
None => Err(format!(
"Working directory is not valid unicode: {}",
- evaluator.search.working_directory.display()
+ context.evaluator.search.working_directory.display()
)),
}
}
-fn append(_evaluator: &Evaluator, suffix: &str, s: &str) -> Result {
+fn append(_context: Context, suffix: &str, s: &str) -> Result {
Ok(
s.split_whitespace()
.map(|s| format!("{s}{suffix}"))
@@ -123,16 +141,16 @@ fn append(_evaluator: &Evaluator, suffix: &str, s: &str) -> Result Result {
+fn arch(_context: Context) -> Result {
Ok(target::arch().to_owned())
}
-fn blake3(_evaluator: &Evaluator, s: &str) -> Result {
+fn blake3(_context: Context, s: &str) -> Result {
Ok(blake3::hash(s.as_bytes()).to_string())
}
-fn blake3_file(evaluator: &Evaluator, path: &str) -> Result {
- let path = evaluator.search.working_directory.join(path);
+fn blake3_file(context: Context, path: &str) -> Result {
+ let path = context.evaluator.search.working_directory.join(path);
let mut hasher = blake3::Hasher::new();
hasher
.update_mmap_rayon(&path)
@@ -140,7 +158,7 @@ fn blake3_file(evaluator: &Evaluator, path: &str) -> Result {
Ok(hasher.finalize().to_string())
}
-fn canonicalize(_evaluator: &Evaluator, path: &str) -> Result {
+fn canonicalize(_context: Context, path: &str) -> Result {
let canonical =
std::fs::canonicalize(path).map_err(|err| format!("I/O error canonicalizing path: {err}"))?;
@@ -152,7 +170,7 @@ fn canonicalize(_evaluator: &Evaluator, path: &str) -> Result {
})
}
-fn capitalize(_evaluator: &Evaluator, s: &str) -> Result {
+fn capitalize(_context: Context, s: &str) -> Result {
let mut capitalized = String::new();
for (i, c) in s.chars().enumerate() {
if i == 0 {
@@ -164,7 +182,7 @@ fn capitalize(_evaluator: &Evaluator, s: &str) -> Result {
Ok(capitalized)
}
-fn choose(_evaluator: &Evaluator, n: &str, alphabet: &str) -> Result {
+fn choose(_context: Context, n: &str, alphabet: &str) -> Result {
if alphabet.is_empty() {
return Err("empty alphabet".into());
}
@@ -188,7 +206,7 @@ fn choose(_evaluator: &Evaluator, n: &str, alphabet: &str) -> Result Result {
+fn clean(_context: Context, path: &str) -> Result {
Ok(Path::new(path).lexiclean().to_str().unwrap().to_owned())
}
@@ -208,7 +226,7 @@ fn dir(name: &'static str, f: fn() -> Option) -> Result
}
}
-fn encode_uri_component(_evaluator: &Evaluator, s: &str) -> Result {
+fn encode_uri_component(_context: Context, s: &str) -> Result {
static PERCENT_ENCODE: percent_encoding::AsciiSet = percent_encoding::NON_ALPHANUMERIC
.remove(b'-')
.remove(b'_')
@@ -222,10 +240,10 @@ fn encode_uri_component(_evaluator: &Evaluator, s: &str) -> Result Result {
+fn env_var(context: Context, key: &str) -> Result {
use std::env::VarError::*;
- if let Some(value) = evaluator.dotenv.get(key) {
+ if let Some(value) = context.evaluator.dotenv.get(key) {
return Ok(value.clone());
}
@@ -238,10 +256,10 @@ fn env_var(evaluator: &Evaluator, key: &str) -> Result {
}
}
-fn env_var_or_default(evaluator: &Evaluator, key: &str, default: &str) -> Result {
+fn env_var_or_default(context: Context, key: &str, default: &str) -> Result {
use std::env::VarError::*;
- if let Some(value) = evaluator.dotenv.get(key) {
+ if let Some(value) = context.evaluator.dotenv.get(key) {
return Ok(value.clone());
}
@@ -254,48 +272,49 @@ fn env_var_or_default(evaluator: &Evaluator, key: &str, default: &str) -> Result
}
}
-fn env(evaluator: &Evaluator, key: &str, default: Option<&str>) -> Result {
+fn env(context: Context, key: &str, default: Option<&str>) -> Result {
match default {
- Some(val) => env_var_or_default(evaluator, key, val),
- None => env_var(evaluator, key),
+ Some(val) => env_var_or_default(context, key, val),
+ None => env_var(context, key),
}
}
-fn error(_evaluator: &Evaluator, message: &str) -> Result {
+fn error(_context: Context, message: &str) -> Result {
Err(message.to_owned())
}
-fn extension(_evaluator: &Evaluator, path: &str) -> Result {
+fn extension(_context: Context, path: &str) -> Result {
Utf8Path::new(path)
.extension()
.map(str::to_owned)
.ok_or_else(|| format!("Could not extract extension from `{path}`"))
}
-fn file_name(_evaluator: &Evaluator, path: &str) -> Result {
+fn file_name(_context: Context, path: &str) -> Result {
Utf8Path::new(path)
.file_name()
.map(str::to_owned)
.ok_or_else(|| format!("Could not extract file name from `{path}`"))
}
-fn file_stem(_evaluator: &Evaluator, path: &str) -> Result {
+fn file_stem(_context: Context, path: &str) -> Result {
Utf8Path::new(path)
.file_stem()
.map(str::to_owned)
.ok_or_else(|| format!("Could not extract file stem from `{path}`"))
}
-fn invocation_directory(evaluator: &Evaluator) -> Result {
+fn invocation_directory(context: Context) -> Result {
Platform::convert_native_path(
- &evaluator.search.working_directory,
- &evaluator.config.invocation_directory,
+ &context.evaluator.search.working_directory,
+ &context.evaluator.config.invocation_directory,
)
.map_err(|e| format!("Error getting shell path: {e}"))
}
-fn invocation_directory_native(evaluator: &Evaluator) -> Result {
- evaluator
+fn invocation_directory_native(context: Context) -> Result {
+ context
+ .evaluator
.config
.invocation_directory
.to_str()
@@ -303,12 +322,12 @@ fn invocation_directory_native(evaluator: &Evaluator) -> Result
.ok_or_else(|| {
format!(
"Invocation directory is not valid unicode: {}",
- evaluator.config.invocation_directory.display()
+ context.evaluator.config.invocation_directory.display()
)
})
}
-fn prepend(_evaluator: &Evaluator, prefix: &str, s: &str) -> Result {
+fn prepend(_context: Context, prefix: &str, s: &str) -> Result {
Ok(
s.split_whitespace()
.map(|s| format!("{prefix}{s}"))
@@ -317,7 +336,7 @@ fn prepend(_evaluator: &Evaluator, prefix: &str, s: &str) -> Result Result {
+fn join(_context: Context, base: &str, with: &str, and: &[String]) -> Result {
let mut result = Utf8Path::new(base).join(with);
for arg in and {
result.push(arg);
@@ -325,7 +344,7 @@ fn join(_evaluator: &Evaluator, base: &str, with: &str, and: &[String]) -> Resul
Ok(result.to_string())
}
-fn just_executable(_evaluator: &Evaluator) -> Result {
+fn just_executable(_context: Context) -> Result {
let exe_path =
env::current_exe().map_err(|e| format!("Error getting current executable: {e}"))?;
@@ -337,12 +356,13 @@ fn just_executable(_evaluator: &Evaluator) -> Result {
})
}
-fn just_pid(_evaluator: &Evaluator) -> Result {
+fn just_pid(_context: Context) -> Result {
Ok(std::process::id().to_string())
}
-fn justfile(evaluator: &Evaluator) -> Result {
- evaluator
+fn justfile(context: Context) -> Result {
+ context
+ .evaluator
.search
.justfile
.to_str()
@@ -350,16 +370,16 @@ fn justfile(evaluator: &Evaluator) -> Result {
.ok_or_else(|| {
format!(
"Justfile path is not valid unicode: {}",
- evaluator.search.justfile.display()
+ context.evaluator.search.justfile.display()
)
})
}
-fn justfile_directory(evaluator: &Evaluator) -> Result {
- let justfile_directory = evaluator.search.justfile.parent().ok_or_else(|| {
+fn justfile_directory(context: Context) -> Result {
+ let justfile_directory = context.evaluator.search.justfile.parent().ok_or_else(|| {
format!(
"Could not resolve justfile directory. Justfile `{}` had no parent.",
- evaluator.search.justfile.display()
+ context.evaluator.search.justfile.display()
)
})?;
@@ -374,41 +394,42 @@ fn justfile_directory(evaluator: &Evaluator) -> Result {
})
}
-fn kebabcase(_evaluator: &Evaluator, s: &str) -> Result {
+fn kebabcase(_context: Context, s: &str) -> Result {
Ok(s.to_kebab_case())
}
-fn lowercamelcase(_evaluator: &Evaluator, s: &str) -> Result {
+fn lowercamelcase(_context: Context, s: &str) -> Result {
Ok(s.to_lower_camel_case())
}
-fn lowercase(_evaluator: &Evaluator, s: &str) -> Result {
+fn lowercase(_context: Context, s: &str) -> Result {
Ok(s.to_lowercase())
}
-fn num_cpus(_evaluator: &Evaluator) -> Result {
+fn num_cpus(_context: Context) -> Result {
let num = num_cpus::get();
Ok(num.to_string())
}
-fn os(_evaluator: &Evaluator) -> Result {
+fn os(_context: Context) -> Result {
Ok(target::os().to_owned())
}
-fn os_family(_evaluator: &Evaluator) -> Result {
+fn os_family(_context: Context) -> Result {
Ok(target::family().to_owned())
}
-fn parent_directory(_evaluator: &Evaluator, path: &str) -> Result {
+fn parent_directory(_context: Context, path: &str) -> Result {
Utf8Path::new(path)
.parent()
.map(Utf8Path::to_string)
.ok_or_else(|| format!("Could not extract parent directory from `{path}`"))
}
-fn path_exists(evaluator: &Evaluator, path: &str) -> Result {
+fn path_exists(context: Context, path: &str) -> Result {
Ok(
- evaluator
+ context
+ .evaluator
.search
.working_directory
.join(path)
@@ -417,16 +438,16 @@ fn path_exists(evaluator: &Evaluator, path: &str) -> Result {
)
}
-fn quote(_evaluator: &Evaluator, s: &str) -> Result {
+fn quote(_context: Context, s: &str) -> Result {
Ok(format!("'{}'", s.replace('\'', "'\\''")))
}
-fn replace(_evaluator: &Evaluator, s: &str, from: &str, to: &str) -> Result {
+fn replace(_context: Context, s: &str, from: &str, to: &str) -> Result {
Ok(s.replace(from, to))
}
fn replace_regex(
- _evaluator: &Evaluator,
+ _context: Context,
s: &str,
regex: &str,
replacement: &str,
@@ -439,7 +460,7 @@ fn replace_regex(
)
}
-fn sha256(_evaluator: &Evaluator, s: &str) -> Result {
+fn sha256(_context: Context, s: &str) -> Result {
use sha2::{Digest, Sha256};
let mut hasher = Sha256::new();
hasher.update(s);
@@ -447,9 +468,9 @@ fn sha256(_evaluator: &Evaluator, s: &str) -> Result {
Ok(format!("{hash:x}"))
}
-fn sha256_file(evaluator: &Evaluator, path: &str) -> Result {
+fn sha256_file(context: Context, path: &str) -> Result {
use sha2::{Digest, Sha256};
- let path = evaluator.search.working_directory.join(path);
+ let path = context.evaluator.search.working_directory.join(path);
let mut hasher = Sha256::new();
let mut file =
fs::File::open(&path).map_err(|err| format!("Failed to open `{}`: {err}", path.display()))?;
@@ -459,73 +480,112 @@ fn sha256_file(evaluator: &Evaluator, path: &str) -> Result {
Ok(format!("{hash:x}"))
}
-fn shell(evaluator: &Evaluator, command: &str, args: &[String]) -> Result {
+fn shell(context: Context, command: &str, args: &[String]) -> Result {
let args = iter::once(command)
.chain(args.iter().map(String::as_str))
.collect::>();
- evaluator
+ context
+ .evaluator
.run_command(command, &args)
.map_err(|output_error| output_error.to_string())
}
-fn shoutykebabcase(_evaluator: &Evaluator, s: &str) -> Result {
+fn shoutykebabcase(_context: Context, s: &str) -> Result {
Ok(s.to_shouty_kebab_case())
}
-fn shoutysnakecase(_evaluator: &Evaluator, s: &str) -> Result {
+fn shoutysnakecase(_context: Context, s: &str) -> Result {
Ok(s.to_shouty_snake_case())
}
-fn snakecase(_evaluator: &Evaluator, s: &str) -> Result {
+fn snakecase(_context: Context, s: &str) -> Result {
Ok(s.to_snake_case())
}
-fn titlecase(_evaluator: &Evaluator, s: &str) -> Result {
+fn source_directory(context: Context) -> Result {
+ context
+ .evaluator
+ .search
+ .justfile
+ .parent()
+ .unwrap()
+ .join(context.name.token.path)
+ .parent()
+ .unwrap()
+ .to_str()
+ .map(str::to_owned)
+ .ok_or_else(|| {
+ format!(
+ "Source file path not valid unicode: {}",
+ context.name.token.path.display(),
+ )
+ })
+}
+
+fn source_file(context: Context) -> Result {
+ context
+ .evaluator
+ .search
+ .justfile
+ .parent()
+ .unwrap()
+ .join(context.name.token.path)
+ .to_str()
+ .map(str::to_owned)
+ .ok_or_else(|| {
+ format!(
+ "Source file path not valid unicode: {}",
+ context.name.token.path.display(),
+ )
+ })
+}
+
+fn titlecase(_context: Context, s: &str) -> Result {
Ok(s.to_title_case())
}
-fn trim(_evaluator: &Evaluator, s: &str) -> Result {
+fn trim(_context: Context, s: &str) -> Result {
Ok(s.trim().to_owned())
}
-fn trim_end(_evaluator: &Evaluator, s: &str) -> Result {
+fn trim_end(_context: Context, s: &str) -> Result {
Ok(s.trim_end().to_owned())
}
-fn trim_end_match(_evaluator: &Evaluator, s: &str, pat: &str) -> Result {
+fn trim_end_match(_context: Context, s: &str, pat: &str) -> Result {
Ok(s.strip_suffix(pat).unwrap_or(s).to_owned())
}
-fn trim_end_matches(_evaluator: &Evaluator, s: &str, pat: &str) -> Result {
+fn trim_end_matches(_context: Context, s: &str, pat: &str) -> Result {
Ok(s.trim_end_matches(pat).to_owned())
}
-fn trim_start(_evaluator: &Evaluator, s: &str) -> Result {
+fn trim_start(_context: Context, s: &str) -> Result {
Ok(s.trim_start().to_owned())
}
-fn trim_start_match(_evaluator: &Evaluator, s: &str, pat: &str) -> Result {
+fn trim_start_match(_context: Context, s: &str, pat: &str) -> Result {
Ok(s.strip_prefix(pat).unwrap_or(s).to_owned())
}
-fn trim_start_matches(_evaluator: &Evaluator, s: &str, pat: &str) -> Result {
+fn trim_start_matches(_context: Context, s: &str, pat: &str) -> Result {
Ok(s.trim_start_matches(pat).to_owned())
}
-fn uppercamelcase(_evaluator: &Evaluator, s: &str) -> Result {
+fn uppercamelcase(_context: Context, s: &str) -> Result {
Ok(s.to_upper_camel_case())
}
-fn uppercase(_evaluator: &Evaluator, s: &str) -> Result {
+fn uppercase(_context: Context, s: &str) -> Result {
Ok(s.to_uppercase())
}
-fn uuid(_evaluator: &Evaluator) -> Result {
+fn uuid(_context: Context) -> Result {
Ok(uuid::Uuid::new_v4().to_string())
}
-fn without_extension(_evaluator: &Evaluator, path: &str) -> Result {
+fn without_extension(_context: Context, path: &str) -> Result {
let parent = Utf8Path::new(path)
.parent()
.ok_or_else(|| format!("Could not extract parent from `{path}`"))?;
@@ -539,11 +599,7 @@ fn without_extension(_evaluator: &Evaluator, path: &str) -> Result=0.1.0")
-fn semver_matches(
- _evaluator: &Evaluator,
- version: &str,
- requirement: &str,
-) -> Result {
+fn semver_matches(_context: Context, version: &str, requirement: &str) -> Result {
Ok(
requirement
.parse::()
diff --git a/src/thunk.rs b/src/thunk.rs
index 87d66a3dbe..2ab203abb9 100644
--- a/src/thunk.rs
+++ b/src/thunk.rs
@@ -6,48 +6,48 @@ pub(crate) enum Thunk<'src> {
Nullary {
name: Name<'src>,
#[derivative(Debug = "ignore", PartialEq = "ignore")]
- function: fn(&Evaluator) -> Result,
+ function: fn(function::Context) -> Result,
},
Unary {
name: Name<'src>,
#[derivative(Debug = "ignore", PartialEq = "ignore")]
- function: fn(&Evaluator, &str) -> Result,
+ function: fn(function::Context, &str) -> Result,
arg: Box>,
},
UnaryOpt {
name: Name<'src>,
#[derivative(Debug = "ignore", PartialEq = "ignore")]
- function: fn(&Evaluator, &str, Option<&str>) -> Result,
+ function: fn(function::Context, &str, Option<&str>) -> Result,
args: (Box>, Box