This allows you to use the Bun Shell to save and run project-specific commands,
similar to just.
The main advantage over just is that Bun Shell seamlessly strings shell commands onto arbitrarily sophisticated
logic written in TypeScript with powerful Bun-API tools, including almost the entire Node.js-API.
- Usage example
- Installation
- Command line format and flags
- Function runRecipe() syntax restrictions
- Contents of the jb_script/settings folder
- Unpleasant features of Bun Shell and ways to work around them
Recipes are stored in just_bun.ts files in the arms of the switch(recipeName)
of function
export async function runRecipe(recipeName?: string, args = [])
.
Example just_bun.ts file:
import { $ } from "bun";
export async function runRecipe(recipeName?: string, args = []) {
switch (recipeName) {
case 'run': // recipeName
case 'r': // recipeName alias
case undefined: // default: for run without recipeName
await $`cargo run`;
break;
case 'build_release':
case 'b':
await $`cargo build --release`;
await $`echo "Result in: ${__dirname}/target/release"`;
break;
case 'test':
case '# args: [filter] [-nThreads] // e.g.: -1 - in one thread': // comment for list
case 't':
await $`cargo test ${{raw: args.join(' ').replace('-', '-- --test-threads=')}}`;
break;
default:
return console.log(`recipeName error: '${recipeName}'`);
}
}
Now, in the directory containing this just_bun.ts or in its child directory, the terminal command jb -l
will output:
◇ List of recipes in ./just_bun.ts (<absolute path>/just_bun.ts):
1. run | r | <default>
2. build_release | b
3. test | t # args: [filter] [-nThreads] // e.g.: -1 - in one thread
◇ Enter: ( <number> | <name> | <alias> ) [args]. Cancel: <Space>
> ▮
Typing 1
or run
or r
or just <Enter>
at the prompt will do: cargo run
.
And, for example, typing 3 -1
or t -1
will produce: cargo test -- --test-threads=1
.
The same result can be obtained without jb -l
by calling directly in the shell:
jb run
or jb r
or just jb
- for cargo run
or jb t -1
- for cargo test -- --test-threads=1
.
If you prefer, the recipe file name can have a dot in front and any letter case before the extension.
For example, Just_Bun.ts
and .JUST_BUN.ts
are valid names.
Read more about the limitations of the function runRecipe() syntax below.
More information about syntax and working with Bun Shell - follow the link.
- Install Bun if it is not already on your system. Recommended Bun version - not lower than 1.1.21
- Run in terminal in ~/.bun/ directory in bash:
or in PowerShell:
$ mkdir jb_script; cd jb_script; bun i just-bun $ mv -f -t ./ ./node_modules/just-bun/jb_script/**
This will create a jb_script folder with the main.js startup script, a settings subfolder for user settings, and install just-bun in node_modules.PS> mkdir jb_script; cd jb_script; bun i just-bun PS> mv -force ./node_modules/just-bun/jb_script/** ./
- Create a short, convenient alias for your main shell's
bun <absolute path to jb_script>/main.js
command. Here I will denote this aliasjb
.
If you have Rust installed on your system, instead of creating aliases in your shells, you can compile the jb executable (jb.exe for Windows) from the bun_script_alias folder of the repository and place it in the ~/.bun/bin/ folder. The only thing it does is call
bun ~/.bun/jb_script/main.js
with the arguments passed to it.
If desired, you can synchronously rename the runner file and the folder to a more convenient name for calling, for example - in j (j.exe for Windows) and j_script/ respectively.
- At this point everything works, but your code editor requires @types/bun declarations for
autocompletion and type checking from the Bun-API when editing recipe files.
To do this, go to in the terminal in the root directory of the projects in which you will use the recipe files, and enterjb -@
. This will create/update a node_modules/ folder here with @types/bun declarations.
This section can also be read using the command jb -h
or jb --help
Command Line Format variants:
jb [-g] [<recipeName> [args]]
# main usejb -f <path/to/recipe/file>.ts [<recipeName> [args]]
jb -t [<templateSearchLine>]
jb -lf <path/to/recipe/file>.ts
jb <flag>
Flags:
-g
runs a recipe from the global recipe file located in the settings folder. Without the -g flag, the recipe file is searched in the current directory and up the chain of parent directories-f
runs a recipe from any .ts-file specified in <path/to/recipe/file>-t
creates a new recipe file in the current folder based on the template found by the first characters specified in <templateSearchLine>-l
shows the path and numbered list of recipes for the current folder and offers to run the recipe by indicating its number | name | alias and [args]-L
is the same as-l
, but for a global recipe file-lf
is the same as-l
, but for the file <path/to/recipe/file>.ts-o
opens the current recipe file in the editor-O
opens the global recipe file in the editor-p
prints relative and absolute path to the current recipe file-P
prints the absolute path to the global recipe file-@
installs/updates node_modules/ with @types/bun in the folder of the current recipe file, if it doesn't find it, then in the current folder-i
checks and fixes the absolute import path to funcs.ts in the current recipe file and in all recipe files located in the current and child directories-u
update just-bun to latest version-h
,--help
displays help on format command line and flags
These restrictions are imposed solely for the correct display of the list of recipes
displayed using the -l
, -L
, -lf
flags:
The list of recipes is formed according to the text of the first switch statement from recipeName:
switch (recipeName) {...}
and the restrictions apply only to this statement:
-
All
case
expressions must be string literals without line breaks. Only onecase
is allowed per entire switch statement with an undefined literal for the default recipe. -
A chain of several
cases
before a common branch of operations is:- optional recipe comments:
case
starting with "#" - required recipe name: first
case
- is not a comment - optional recipe aliases
case
- comments can be anywhere in the chain.
Such a recipe is displayed in the list in one line: at the beginning - the name of the recipe and its aliases separated by|
, and at the end - the first comment of the recipe (if any). The remaining recipe comments, if any, are displayed as additional lines in the column of first comment .
Acase
with the valueundefined
is listed as<default>
and can be either an alias or the first or onlycase
in the chain. - optional recipe comments:
This is a global recipe file, called from any working directory with the -g
, -L
, -O
, -P
flags.
Used for custom recipes common to all projects. Unlike current recipe files,
its name must be strictly "just_bun.ts": without a dot at the beginning of the name and in lower case.
This is an optional file of user settings that differ from the default values:
- notUpdate - false by default: disables updating on accidental input of the
-u
flag - editor section:
- fileOpen - command line for opening a file in the editor (by default - for VS Code: "code --goto %file%").
If you want to generally prohibit the opening of files using the-o
,-O
flags and those newly created from templates using the-t
flag, The value of this parameter should be set to "none". - fileOpenReport - whether it is necessary to announce in the console about the transmitted command to open a file or about prohibiting fileOpen = "none" (default - false).
- fileOpen - command line for opening a file in the editor (by default - for VS Code: "code --goto %file%").
This is a module containing exportable user constants and functions used in any recipe files. The exported functions initially written in it are used in the starting templates of recipe files from the templates-just_bun/ folder.
Contains custom recipe file templates. They will be called by the first characters of the name
with the command jb -t [<templateSearchLine>]
. If the argument is not passed,
the _.ts template will be used
If you want recipe files newly created using templates to have a dot at the beginning of the name or were non-lower case, replace the ending of the name of this directory "just_bun" with the required one (for example, to create .JUST_BUN.ts files, the directory name should be "templates-.JUST_BUN" ).
For portability, templates import funcs.ts at a relative path. But in newly created recipe files based on them, this path will be replaced with the actual absolute one.
bun Issue #13129
The main Bun Shell $
function does not have the option to pre-print the interpolated command to the console,
which is often desirable when running recipes.
Solution:
You can use a custom function in funcs.ts that decorates $`...`
appropriately.
This is precisely the purpose that the function p$()
originally written in funcs.ts serves,
which is called in the same way: p$`...`
, prints an interpolation of the command
close to $
and then passes the call to $
. Examples of using p$
can be found in the templates.
bun Issue #12225
For example, the command
await $`echo "▶ - play, ■ - stop"`
will output to the console:
\u25B6 - play, \u25A0 - stop
Solution:
Pass parameters and paths containing non-ASCII characters as variables and expressions for interpolation:
await $`echo ${"▶ - play, ■ - stop"}`
will output to the console:
▶ - play, ■ - stop