diff --git a/README.md b/README.md index 17bbfad278..9d4f9f6cd5 100644 --- a/README.md +++ b/README.md @@ -1712,6 +1712,7 @@ Recipes may be annotated with attributes that change their behavior. | `[confirm]`1.17.0 | Require confirmation prior to executing recipe. | | `[confirm('PROMPT')]`1.23.0 | Require confirmation prior to executing recipe with a custom prompt. | | `[doc('DOC')]`1.27.0 | Set recipe's [documentation comment](#documentation-comments) to `DOC`. | +| `[extension('EXT')]`master | Set shebang recipe script's file extension to `EXT`. `EXT` should include a period if one is desired. | | `[group('NAME')]`1.27.0 | Put recipe in [recipe group](#recipe-groups) `NAME`. | | `[linux]`1.8.0 | Enable recipe on Linux. | | `[macos]`1.8.0 | Enable recipe on MacOS. | diff --git a/src/analyzer.rs b/src/analyzer.rs index 1f1cece394..bd4319bab5 100644 --- a/src/analyzer.rs +++ b/src/analyzer.rs @@ -255,6 +255,20 @@ impl<'src> Analyzer<'src> { continued = line.is_continuation(); } + if !recipe.shebang { + if let Some(attribute) = recipe + .attributes + .iter() + .find(|attribute| matches!(attribute, Attribute::Extension(_))) + { + return Err(recipe.name.error(InvalidAttribute { + item_kind: "Recipe", + item_name: recipe.name.lexeme(), + attribute: attribute.clone(), + })); + } + } + Ok(()) } diff --git a/src/attribute.rs b/src/attribute.rs index 5176364699..699d1495e1 100644 --- a/src/attribute.rs +++ b/src/attribute.rs @@ -11,6 +11,7 @@ use super::*; pub(crate) enum Attribute<'src> { Confirm(Option>), Doc(Option>), + Extension(StringLiteral<'src>), Group(StringLiteral<'src>), Linux, Macos, @@ -27,7 +28,7 @@ impl AttributeDiscriminant { fn argument_range(self) -> RangeInclusive { match self { Self::Confirm | Self::Doc => 0..=1, - Self::Group => 1..=1, + Self::Group | Self::Extension => 1..=1, Self::Linux | Self::Macos | Self::NoCd @@ -46,8 +47,6 @@ impl<'src> Attribute<'src> { name: Name<'src>, argument: Option>, ) -> CompileResult<'src, Self> { - use AttributeDiscriminant::*; - let discriminant = name .lexeme() .parse::() @@ -72,18 +71,19 @@ impl<'src> Attribute<'src> { } Ok(match discriminant { - Confirm => Self::Confirm(argument), - Doc => Self::Doc(argument), - Group => Self::Group(argument.unwrap()), - Linux => Self::Linux, - Macos => Self::Macos, - NoCd => Self::NoCd, - NoExitMessage => Self::NoExitMessage, - NoQuiet => Self::NoQuiet, - PositionalArguments => Self::PositionalArguments, - Private => Self::Private, - Unix => Self::Unix, - Windows => Self::Windows, + AttributeDiscriminant::Confirm => Self::Confirm(argument), + AttributeDiscriminant::Doc => Self::Doc(argument), + AttributeDiscriminant::Extension => Self::Extension(argument.unwrap()), + AttributeDiscriminant::Group => Self::Group(argument.unwrap()), + AttributeDiscriminant::Linux => Self::Linux, + AttributeDiscriminant::Macos => Self::Macos, + AttributeDiscriminant::NoCd => Self::NoCd, + AttributeDiscriminant::NoExitMessage => Self::NoExitMessage, + AttributeDiscriminant::NoQuiet => Self::NoQuiet, + AttributeDiscriminant::PositionalArguments => Self::PositionalArguments, + AttributeDiscriminant::Private => Self::Private, + AttributeDiscriminant::Unix => Self::Unix, + AttributeDiscriminant::Windows => Self::Windows, }) } @@ -93,9 +93,8 @@ impl<'src> Attribute<'src> { fn argument(&self) -> Option<&StringLiteral> { match self { - Self::Confirm(prompt) => prompt.as_ref(), - Self::Doc(doc) => doc.as_ref(), - Self::Group(group) => Some(group), + Self::Confirm(argument) | Self::Doc(argument) => argument.as_ref(), + Self::Extension(argument) | Self::Group(argument) => Some(argument), Self::Linux | Self::Macos | Self::NoCd diff --git a/src/recipe.rs b/src/recipe.rs index a41ce93559..2ba924705e 100644 --- a/src/recipe.rs +++ b/src/recipe.rs @@ -368,7 +368,16 @@ impl<'src, D> Recipe<'src, D> { io_error: error, })?; let mut path = tempdir.path().to_path_buf(); - path.push(shebang.script_filename(self.name())); + + let extension = self.attributes.iter().find_map(|attribute| { + if let Attribute::Extension(extension) = attribute { + Some(extension.cooked.as_str()) + } else { + None + } + }); + + path.push(shebang.script_filename(self.name(), extension)); { let mut f = fs::File::create(&path).map_err(|error| Error::TempdirIo { diff --git a/src/shebang.rs b/src/shebang.rs index 067963f07f..36b79dce45 100644 --- a/src/shebang.rs +++ b/src/shebang.rs @@ -38,12 +38,14 @@ impl<'line> Shebang<'line> { .unwrap_or(self.interpreter) } - pub(crate) fn script_filename(&self, recipe: &str) -> String { - match self.interpreter_filename() { - "cmd" | "cmd.exe" => format!("{recipe}.bat"), - "powershell" | "powershell.exe" | "pwsh" | "pwsh.exe" => format!("{recipe}.ps1"), - _ => recipe.to_owned(), - } + pub(crate) fn script_filename(&self, recipe: &str, extension: Option<&str>) -> String { + let extension = extension.unwrap_or_else(|| match self.interpreter_filename() { + "cmd" | "cmd.exe" => ".bat", + "powershell" | "powershell.exe" | "pwsh" | "pwsh.exe" => ".ps1", + _ => "", + }); + + format!("{recipe}{extension}") } pub(crate) fn include_shebang_line(&self) -> bool { @@ -138,7 +140,9 @@ mod tests { #[test] fn powershell_script_filename() { assert_eq!( - Shebang::new("#!powershell").unwrap().script_filename("foo"), + Shebang::new("#!powershell") + .unwrap() + .script_filename("foo", None), "foo.ps1" ); } @@ -146,7 +150,7 @@ mod tests { #[test] fn pwsh_script_filename() { assert_eq!( - Shebang::new("#!pwsh").unwrap().script_filename("foo"), + Shebang::new("#!pwsh").unwrap().script_filename("foo", None), "foo.ps1" ); } @@ -156,7 +160,7 @@ mod tests { assert_eq!( Shebang::new("#!powershell.exe") .unwrap() - .script_filename("foo"), + .script_filename("foo", None), "foo.ps1" ); } @@ -164,7 +168,9 @@ mod tests { #[test] fn pwsh_exe_script_filename() { assert_eq!( - Shebang::new("#!pwsh.exe").unwrap().script_filename("foo"), + Shebang::new("#!pwsh.exe") + .unwrap() + .script_filename("foo", None), "foo.ps1" ); } @@ -172,7 +178,7 @@ mod tests { #[test] fn cmd_script_filename() { assert_eq!( - Shebang::new("#!cmd").unwrap().script_filename("foo"), + Shebang::new("#!cmd").unwrap().script_filename("foo", None), "foo.bat" ); } @@ -180,14 +186,19 @@ mod tests { #[test] fn cmd_exe_script_filename() { assert_eq!( - Shebang::new("#!cmd.exe").unwrap().script_filename("foo"), + Shebang::new("#!cmd.exe") + .unwrap() + .script_filename("foo", None), "foo.bat" ); } #[test] fn plain_script_filename() { - assert_eq!(Shebang::new("#!bar").unwrap().script_filename("foo"), "foo"); + assert_eq!( + Shebang::new("#!bar").unwrap().script_filename("foo", None), + "foo" + ); } #[test] @@ -211,4 +222,26 @@ mod tests { fn include_shebang_line_other_windows() { assert!(!Shebang::new("#!foo -c").unwrap().include_shebang_line()); } + + #[test] + fn filename_with_extension() { + assert_eq!( + Shebang::new("#!bar") + .unwrap() + .script_filename("foo", Some(".sh")), + "foo.sh" + ); + assert_eq!( + Shebang::new("#!pwsh.exe") + .unwrap() + .script_filename("foo", Some(".sh")), + "foo.sh" + ); + assert_eq!( + Shebang::new("#!cmd.exe") + .unwrap() + .script_filename("foo", Some(".sh")), + "foo.sh" + ); + } } diff --git a/tests/attributes.rs b/tests/attributes.rs index a604777225..65c4d5ad08 100644 --- a/tests/attributes.rs +++ b/tests/attributes.rs @@ -193,3 +193,40 @@ fn doc_multiline() { ) .run(); } + +#[test] +fn extension() { + Test::new() + .justfile( + " + [extension: '.txt'] + baz: + #!/bin/sh + echo $0 + ", + ) + .stdout_regex(r"*baz\.txt\n") + .run(); +} + +#[test] +fn extension_on_linewise_error() { + Test::new() + .justfile( + " + [extension: '.txt'] + baz: + ", + ) + .stderr( + " + error: Recipe `baz` has invalid attribute `extension` + ——▶ justfile:2:1 + │ + 2 │ baz: + │ ^^^ +", + ) + .status(EXIT_FAILURE) + .run(); +}